Compare commits
86 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
c7fcfa8534 | ||
|
|
aa012b992e | ||
|
|
538182e12c | ||
|
|
13d554acbe | ||
|
|
4fcf90edae | ||
|
|
fc970799a1 | ||
|
|
78d7e85352 | ||
|
|
c503855262 | ||
|
|
77acfb4593 | ||
|
|
f6d0cf95fd | ||
|
|
25b570d7e9 | ||
|
|
297b40d526 | ||
|
|
bd2cb06789 | ||
|
|
3c5682150d | ||
|
|
e32b2ab6bd | ||
|
|
1df5dc2e93 | ||
|
|
36594014c9 | ||
|
|
8f9eccdd61 | ||
|
|
dd03989709 | ||
|
|
b2a11eaf2a | ||
|
|
25c2a435f5 | ||
|
|
3d1a21094a | ||
|
|
2a0d29a390 | ||
|
|
db955e671f | ||
|
|
ab44c38298 | ||
|
|
93e2db7085 | ||
|
|
b1ef9251d4 | ||
|
|
b19afbfe93 | ||
|
|
a6762d500c | ||
|
|
c4174a7167 | ||
|
|
2f9fe7003a | ||
|
|
c86436afa6 | ||
|
|
29e912e90b | ||
|
|
e29de04513 | ||
|
|
c6945514cb | ||
|
|
847cd7ed2b | ||
|
|
fbf897b047 | ||
|
|
45c7b8008a | ||
|
|
4788775f8c | ||
|
|
bcb8546e91 | ||
|
|
befa5a2b54 | ||
|
|
0ba64fc318 | ||
|
|
d16bd4bcdb | ||
|
|
33a532ee01 | ||
|
|
cdc6b773c2 | ||
|
|
17d5f1814a | ||
|
|
5f8be70066 | ||
|
|
5530eca17d | ||
|
|
c8d9e25085 | ||
|
|
a241119bf7 | ||
|
|
3e3f8d5c2f | ||
|
|
9aeb78fc36 | ||
|
|
7863456d52 | ||
|
|
428b658160 | ||
|
|
350cf80bbf | ||
|
|
d92051d40f | ||
|
|
aa2621f6c6 | ||
|
|
2b1d6f0e7a | ||
|
|
992676722d | ||
|
|
451c754068 | ||
|
|
84ad46751a | ||
|
|
ec5392d566 | ||
|
|
7d8fdbc1fc | ||
|
|
fdfcb9c1df | ||
|
|
a988459dcd | ||
|
|
51e0b46256 | ||
|
|
eb06aeeb26 | ||
|
|
8bb5daf60e | ||
|
|
ac80d1b3ed | ||
|
|
2e17cfab4f | ||
|
|
3f4e1665b1 | ||
|
|
b9b0897d95 | ||
|
|
6337f8bc01 | ||
|
|
ccb8072759 | ||
|
|
d73111cda1 | ||
|
|
ff521ecb1a | ||
|
|
61b4980077 | ||
|
|
100d090853 | ||
|
|
bd60de5995 | ||
|
|
274eecdf18 | ||
|
|
8fa00f826c | ||
|
|
a64fe5b210 | ||
|
|
5c59dc425f | ||
|
|
8ad14d8ea4 | ||
|
|
8a1f9ef44e | ||
|
|
5cd1e11379 |
27
.github/ISSUE_TEMPLATE/bug_report.md
vendored
27
.github/ISSUE_TEMPLATE/bug_report.md
vendored
@@ -1,27 +0,0 @@
|
||||
---
|
||||
name: Bug report
|
||||
about: Create a report to help us improve
|
||||
|
||||
---
|
||||
|
||||
The following program `sample.go` triggers a panic:
|
||||
|
||||
```go
|
||||
package main
|
||||
|
||||
func main() {
|
||||
// add a sample
|
||||
}
|
||||
```
|
||||
|
||||
Expected result:
|
||||
```console
|
||||
$ go run ./sample.go
|
||||
// output
|
||||
```
|
||||
|
||||
Got:
|
||||
```console
|
||||
$ yaegi ./sample.go
|
||||
// output
|
||||
```
|
||||
66
.github/ISSUE_TEMPLATE/bug_report.yml
vendored
Normal file
66
.github/ISSUE_TEMPLATE/bug_report.yml
vendored
Normal file
@@ -0,0 +1,66 @@
|
||||
name: Bug Report
|
||||
description: Create a report to help us improve
|
||||
|
||||
body:
|
||||
- type: markdown
|
||||
attributes:
|
||||
value: |
|
||||
⚠️ Make sure to browse the opened and closed issues before submit your issue.
|
||||
|
||||
- type: textarea
|
||||
id: sample
|
||||
attributes:
|
||||
label: "The following program `sample.go` triggers an unexpected result"
|
||||
value: |
|
||||
package main
|
||||
|
||||
func main() {
|
||||
// add a sample
|
||||
}
|
||||
render: go
|
||||
validations:
|
||||
required: true
|
||||
|
||||
- type: textarea
|
||||
id: expected
|
||||
attributes:
|
||||
label: Expected result
|
||||
description: |-
|
||||
```console
|
||||
$ go run ./sample.go
|
||||
// output
|
||||
```
|
||||
placeholder: $ go run ./sample.go
|
||||
render: console
|
||||
validations:
|
||||
required: true
|
||||
|
||||
- type: textarea
|
||||
id: got
|
||||
attributes:
|
||||
label: Got
|
||||
description: |-
|
||||
```console
|
||||
$ yaegi ./sample.go
|
||||
// output
|
||||
```
|
||||
placeholder: $ yaegi ./sample.go
|
||||
render: console
|
||||
validations:
|
||||
required: true
|
||||
|
||||
- type: input
|
||||
id: version
|
||||
attributes:
|
||||
label: Yaegi Version
|
||||
description: Can be a tag or a hash.
|
||||
validations:
|
||||
required: true
|
||||
|
||||
- type: textarea
|
||||
id: additional
|
||||
attributes:
|
||||
label: Additional Notes
|
||||
description: Use [Markdown syntax](https://help.github.com/articles/github-flavored-markdown) if needed.
|
||||
validations:
|
||||
required: false
|
||||
8
.github/ISSUE_TEMPLATE/config.yml
vendored
Normal file
8
.github/ISSUE_TEMPLATE/config.yml
vendored
Normal file
@@ -0,0 +1,8 @@
|
||||
blank_issues_enabled: false
|
||||
contact_links:
|
||||
- name: Questions
|
||||
url: https://community.traefik.io/c/yaegi
|
||||
about: If you have a question, or are looking for advice, please post on our discussions forum!
|
||||
- name: Documentation
|
||||
url: https://pkg.go.dev/github.com/traefik/yaegi
|
||||
about: Please take a look to our documenation.
|
||||
19
.github/ISSUE_TEMPLATE/feature_request.md
vendored
19
.github/ISSUE_TEMPLATE/feature_request.md
vendored
@@ -1,19 +0,0 @@
|
||||
---
|
||||
name: Feature request
|
||||
about: Propose a change to Yaegi!
|
||||
---
|
||||
|
||||
<!-- ⚠️ If you do not respect this template your issue will be closed. -->
|
||||
<!-- ⚠️ Make sure to browse the opened and closed issues before submit your issue. -->
|
||||
|
||||
#### Proposal
|
||||
|
||||
<!-- Write your feature request in the form of a proposal to be considered for implementation -->
|
||||
|
||||
#### Background
|
||||
|
||||
<!-- Describe the background problem or need that led to this feature request -->
|
||||
|
||||
#### Workarounds
|
||||
|
||||
<!-- Are there any current workarounds that you're using that others in similar positions should know about? -->
|
||||
32
.github/ISSUE_TEMPLATE/feature_request.yml
vendored
Normal file
32
.github/ISSUE_TEMPLATE/feature_request.yml
vendored
Normal file
@@ -0,0 +1,32 @@
|
||||
name: Feature request
|
||||
description: Propose a change to Yaegi
|
||||
|
||||
body:
|
||||
- type: markdown
|
||||
attributes:
|
||||
value: |
|
||||
⚠️ Make sure to browse the opened and closed issues before submit your issue.
|
||||
|
||||
- type: textarea
|
||||
id: proposal
|
||||
attributes:
|
||||
label: Proposal
|
||||
description: Write your feature request in the form of a proposal to be considered for implementation.
|
||||
validations:
|
||||
required: true
|
||||
|
||||
- type: textarea
|
||||
id: background
|
||||
attributes:
|
||||
label: Background
|
||||
description: Describe the background problem or need that led to this feature request.
|
||||
validations:
|
||||
required: true
|
||||
|
||||
- type: textarea
|
||||
id: workarounds
|
||||
attributes:
|
||||
label: Workarounds
|
||||
description: Are there any current workarounds that you're using that others in similar positions should know about?
|
||||
validations:
|
||||
required: true
|
||||
72
.github/workflows/go-cross.yml
vendored
Normal file
72
.github/workflows/go-cross.yml
vendored
Normal file
@@ -0,0 +1,72 @@
|
||||
name: Build Cross OS
|
||||
|
||||
on:
|
||||
push:
|
||||
branches:
|
||||
- master
|
||||
pull_request:
|
||||
|
||||
jobs:
|
||||
|
||||
cross:
|
||||
name: Go
|
||||
runs-on: ${{ matrix.os }}
|
||||
defaults:
|
||||
run:
|
||||
working-directory: ${{ github.workspace }}/go/src/github.com/traefik/yaegi
|
||||
|
||||
strategy:
|
||||
matrix:
|
||||
go-version: [ 1.15, 1.16 ]
|
||||
os: [ubuntu-latest, macos-latest, windows-latest]
|
||||
|
||||
include:
|
||||
- os: ubuntu-latest
|
||||
go-path-suffix: /go
|
||||
- os: macos-latest
|
||||
go-path-suffix: /go
|
||||
- os: windows-latest
|
||||
go-path-suffix: \go
|
||||
|
||||
steps:
|
||||
# https://github.com/marketplace/actions/setup-go-environment
|
||||
- name: Set up Go ${{ matrix.go-version }}
|
||||
uses: actions/setup-go@v2
|
||||
with:
|
||||
go-version: ${{ matrix.go-version }}
|
||||
|
||||
# https://github.com/marketplace/actions/checkout
|
||||
- name: Checkout code
|
||||
uses: actions/checkout@v2
|
||||
with:
|
||||
path: go/src/github.com/traefik/yaegi
|
||||
|
||||
# https://github.com/marketplace/actions/cache
|
||||
- name: Cache Go modules
|
||||
uses: actions/cache@v2
|
||||
with:
|
||||
# In order:
|
||||
# * Module download cache
|
||||
# * Build cache (Linux)
|
||||
# * Build cache (Mac)
|
||||
# * Build cache (Windows)
|
||||
path: |
|
||||
~/go/pkg/mod
|
||||
~/.cache/go-build
|
||||
~/Library/Caches/go-build
|
||||
%LocalAppData%\go-build
|
||||
key: ${{ runner.os }}-${{ matrix.go-version }}-go-${{ hashFiles('**/go.sum') }}
|
||||
restore-keys: |
|
||||
${{ runner.os }}-${{ matrix.go-version }}-go-
|
||||
|
||||
- name: Setup GOPATH
|
||||
run: go env -w GOPATH=${{ github.workspace }}${{ matrix.go-path-suffix }}
|
||||
|
||||
# TODO fail on windows
|
||||
# - name: Tests
|
||||
# run: go test -v -cover ./...
|
||||
# env:
|
||||
# GOPATH: ${{ github.workspace }}${{ matrix.go-path }}
|
||||
|
||||
- name: Build
|
||||
run: go build -race -v -ldflags "-s -w" -trimpath
|
||||
110
.github/workflows/main.yml
vendored
Normal file
110
.github/workflows/main.yml
vendored
Normal file
@@ -0,0 +1,110 @@
|
||||
name: Main
|
||||
|
||||
on:
|
||||
push:
|
||||
branches:
|
||||
- master
|
||||
pull_request:
|
||||
|
||||
env:
|
||||
GO_VERSION: 1.16
|
||||
GOLANGCI_LINT_VERSION: v1.41.1
|
||||
|
||||
jobs:
|
||||
|
||||
linting:
|
||||
name: Linting
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- name: Set up Go ${{ env.GO_VERSION }}
|
||||
uses: actions/setup-go@v2
|
||||
with:
|
||||
go-version: ${{ env.GO_VERSION }}
|
||||
|
||||
- name: Check out code
|
||||
uses: actions/checkout@v2
|
||||
with:
|
||||
fetch-depth: 0
|
||||
|
||||
- name: Check and get dependencies
|
||||
run: |
|
||||
go mod tidy
|
||||
git diff --exit-code go.mod
|
||||
# git diff --exit-code go.sum
|
||||
go mod download
|
||||
|
||||
- name: Install golangci-lint ${{ env.GOLANGCI_LINT_VERSION }}
|
||||
run: curl -sSfL https://raw.githubusercontent.com/golangci/golangci-lint/master/install.sh | sh -s -- -b $(go env GOPATH)/bin ${GOLANGCI_LINT_VERSION}
|
||||
|
||||
- name: Run golangci-lint ${{ env.GOLANGCI_LINT_VERSION }}
|
||||
run: make check
|
||||
|
||||
generate:
|
||||
name: Checks code and generated code
|
||||
runs-on: ubuntu-latest
|
||||
needs: linting
|
||||
strategy:
|
||||
matrix:
|
||||
go-version: [ 1.15, 1.16 ]
|
||||
steps:
|
||||
- name: Set up Go ${{ matrix.go-version }}
|
||||
uses: actions/setup-go@v2
|
||||
with:
|
||||
go-version: ${{ matrix.go-version }}
|
||||
|
||||
- name: Check out code
|
||||
uses: actions/checkout@v2
|
||||
with:
|
||||
fetch-depth: 0
|
||||
|
||||
- name: Check generated code
|
||||
run: |
|
||||
rm -f interp/op.go
|
||||
make generate
|
||||
git update-index -q --refresh
|
||||
CHANGED=$(git diff-index --name-only HEAD --)
|
||||
test -z "$CHANGED" || echo $CHANGED
|
||||
test -z "$CHANGED"
|
||||
|
||||
main:
|
||||
name: Build and Test
|
||||
runs-on: ubuntu-latest
|
||||
needs: linting
|
||||
defaults:
|
||||
run:
|
||||
working-directory: ${{ github.workspace }}/go/src/github.com/traefik/yaegi
|
||||
strategy:
|
||||
matrix:
|
||||
go-version: [ 1.15, 1.16 ]
|
||||
|
||||
steps:
|
||||
- name: Set up Go ${{ matrix.go-version }}
|
||||
uses: actions/setup-go@v2
|
||||
with:
|
||||
go-version: ${{ matrix.go-version }}
|
||||
|
||||
- name: Check out code
|
||||
uses: actions/checkout@v2
|
||||
with:
|
||||
path: go/src/github.com/traefik/yaegi
|
||||
fetch-depth: 0
|
||||
|
||||
# https://github.com/marketplace/actions/cache
|
||||
- name: Cache Go modules
|
||||
uses: actions/cache@v2
|
||||
with:
|
||||
path: ./_test/tmp
|
||||
key: ${{ runner.os }}-yaegi-${{ hashFiles('**//_test/tmp/') }}
|
||||
restore-keys: |
|
||||
${{ runner.os }}-yaegi-
|
||||
|
||||
- name: Setup GOPATH
|
||||
run: go env -w GOPATH=${{ github.workspace }}/go
|
||||
|
||||
- name: Build
|
||||
run: go build -v ./...
|
||||
|
||||
- name: Run tests
|
||||
run: make tests
|
||||
env:
|
||||
GOPATH: ${{ github.workspace }}/go
|
||||
42
.github/workflows/release.yml
vendored
Normal file
42
.github/workflows/release.yml
vendored
Normal file
@@ -0,0 +1,42 @@
|
||||
name: Release
|
||||
|
||||
on:
|
||||
push:
|
||||
tags:
|
||||
- v[0-9]+.[0-9]+*
|
||||
|
||||
env:
|
||||
GO_VERSION: 1.16
|
||||
|
||||
jobs:
|
||||
|
||||
release:
|
||||
name: Create a release
|
||||
runs-on: ubuntu-latest
|
||||
|
||||
steps:
|
||||
- name: Set up Go ${{ env.GO_VERSION }}
|
||||
uses: actions/setup-go@v2
|
||||
with:
|
||||
go-version: ${{ env.GO_VERSION }}
|
||||
|
||||
- name: Check out code
|
||||
uses: actions/checkout@v2
|
||||
with:
|
||||
fetch-depth: 0
|
||||
|
||||
- name: Cache Go modules
|
||||
uses: actions/cache@v2
|
||||
with:
|
||||
path: ~/go/pkg/mod
|
||||
key: ${{ runner.os }}-go-${{ hashFiles('**/go.sum') }}
|
||||
restore-keys: |
|
||||
${{ runner.os }}-go-
|
||||
|
||||
- name: Run GoReleaser
|
||||
uses: goreleaser/goreleaser-action@v2
|
||||
with:
|
||||
version: latest
|
||||
args: release --rm-dist
|
||||
env:
|
||||
GITHUB_TOKEN: ${{ secrets.GH_TOKEN_REPO }}
|
||||
@@ -23,26 +23,37 @@
|
||||
[linters]
|
||||
enable-all = true
|
||||
disable = [
|
||||
"maligned",
|
||||
"golint", # deprecated
|
||||
"scopelint", # deprecated
|
||||
"interfacer", # deprecated
|
||||
"maligned", # deprecated
|
||||
"lll",
|
||||
"gas",
|
||||
"dupl",
|
||||
"prealloc",
|
||||
"scopelint",
|
||||
"gocyclo",
|
||||
"cyclop",
|
||||
"gochecknoinits",
|
||||
"gochecknoglobals",
|
||||
"wsl",
|
||||
"nlreturn",
|
||||
"godox",
|
||||
"funlen",
|
||||
"gocognit",
|
||||
"stylecheck",
|
||||
"gomnd",
|
||||
"testpackage",
|
||||
"paralleltest",
|
||||
"tparallel",
|
||||
"goerr113",
|
||||
"wrapcheck",
|
||||
"nestif",
|
||||
"exhaustive",
|
||||
"nlreturn",
|
||||
"exhaustivestruct",
|
||||
"forbidigo",
|
||||
"ifshort",
|
||||
"forcetypeassert",
|
||||
"errorlint", # TODO: must be reactivate before fixes
|
||||
]
|
||||
|
||||
[issues]
|
||||
@@ -54,6 +65,9 @@
|
||||
[[issues.exclude-rules]]
|
||||
path = "interp/.+_test\\.go"
|
||||
linters = ["goconst"]
|
||||
[[issues.exclude-rules]]
|
||||
path = "interp/.+_test\\.go"
|
||||
text = "var-declaration:"
|
||||
|
||||
[[issues.exclude-rules]]
|
||||
path = "interp/interp.go"
|
||||
@@ -61,3 +75,6 @@
|
||||
[[issues.exclude-rules]]
|
||||
path = "interp/interp.go"
|
||||
text = "`out` can be `io.Writer`"
|
||||
[[issues.exclude-rules]]
|
||||
path = "interp/interp_eval_test.go"
|
||||
linters = ["thelper"]
|
||||
|
||||
@@ -47,7 +47,7 @@ archives:
|
||||
- LICENSE
|
||||
|
||||
brews:
|
||||
- github:
|
||||
- tap:
|
||||
owner: traefik
|
||||
name: homebrew-tap
|
||||
commit_author:
|
||||
|
||||
60
.travis.yml
60
.travis.yml
@@ -1,60 +0,0 @@
|
||||
language: go
|
||||
|
||||
dist: xenial
|
||||
|
||||
branches:
|
||||
only:
|
||||
- master
|
||||
- /^v\d+\.\d+(\.\d+)?(-\S*)?$/
|
||||
|
||||
notifications:
|
||||
email:
|
||||
on_success: never
|
||||
on_failure: change
|
||||
|
||||
cache:
|
||||
directories:
|
||||
- $GOPATH/pkg/mod
|
||||
|
||||
matrix:
|
||||
fast_finish: true
|
||||
include:
|
||||
- go: 1.14.x
|
||||
- go: 1.15.x
|
||||
env: STABLE=true
|
||||
|
||||
env:
|
||||
global:
|
||||
- GO111MODULE=on
|
||||
|
||||
go_import_path: github.com/traefik/yaegi
|
||||
|
||||
before_install:
|
||||
# Install linters and misspell
|
||||
- curl -sfL https://install.goreleaser.com/github.com/golangci/golangci-lint.sh | bash -s -- -b $GOPATH/bin ${GOLANGCI_LINT_VERSION}
|
||||
- golangci-lint --version
|
||||
|
||||
install:
|
||||
- echo "TRAVIS_GO_VERSION=$TRAVIS_GO_VERSION"
|
||||
- go mod download
|
||||
|
||||
before_script:
|
||||
- rm -f interp/op.go
|
||||
- make generate
|
||||
- git update-index -q --refresh
|
||||
- CHANGED=$(git diff-index --name-only HEAD --)
|
||||
- test -z "$CHANGED" || echo $CHANGED
|
||||
- test -z "$CHANGED"
|
||||
|
||||
script:
|
||||
- make check
|
||||
- go build -v ./...
|
||||
- make tests
|
||||
|
||||
deploy:
|
||||
- provider: script
|
||||
skip_cleanup: true
|
||||
script: curl -sL https://git.io/goreleaser | bash
|
||||
on:
|
||||
tags: true
|
||||
condition: $STABLE = true
|
||||
10
README.md
10
README.md
@@ -3,9 +3,9 @@
|
||||
</p>
|
||||
|
||||
[](https://github.com/traefik/yaegi/releases)
|
||||
[](https://travis-ci.com/traefik/yaegi)
|
||||
[](https://godoc.org/github.com/traefik/yaegi)
|
||||
[](https://community.containo.us/c/yaegi)
|
||||
[](https://github.com/traefik/yaegi/actions/workflows/main.yml)
|
||||
[](https://pkg.go.dev/mod/github.com/traefik/yaegi)
|
||||
[](https://community.traefik.io/c/yaegi)
|
||||
|
||||
Yaegi is Another Elegant Go Interpreter.
|
||||
It powers executable Go scripts and plugins, in embedded interpreters or interactive shells, on top of the Go runtime.
|
||||
@@ -18,7 +18,7 @@ It powers executable Go scripts and plugins, in embedded interpreters or interac
|
||||
* Works everywhere Go works
|
||||
* All Go & runtime resources accessible from script (with control)
|
||||
* Security: `unsafe` and `syscall` packages neither used nor exported by default
|
||||
* Support Go 1.13 and Go 1.14 (the latest 2 major releases)
|
||||
* Support Go 1.15 and Go 1.16 (the latest 2 major releases)
|
||||
|
||||
## Install
|
||||
|
||||
@@ -186,7 +186,7 @@ Beside the known [bugs] which are supposed to be fixed in the short term, there
|
||||
[Apache 2.0][License].
|
||||
|
||||
[specs]: https://golang.org/ref/spec
|
||||
[docs]: https://godoc.org/github.com/traefik/yaegi
|
||||
[docs]: https://pkg.go.dev/github.com/traefik/yaegi
|
||||
[license]: https://github.com/traefik/yaegi/blob/master/LICENSE
|
||||
[github]: https://github.com/traefik/yaegi
|
||||
[bugs]: https://github.com/traefik/yaegi/issues?q=is%3Aissue+is%3Aopen+label%3Abug
|
||||
|
||||
@@ -1,10 +1,41 @@
|
||||
package main
|
||||
|
||||
func main() {
|
||||
b := 2
|
||||
var a interface{} = 5 + b
|
||||
b := 2 // int
|
||||
|
||||
var c int = 5 + b
|
||||
println(c)
|
||||
|
||||
var d int32 = 6 + int32(b)
|
||||
println(d)
|
||||
|
||||
var a interface{} = 7 + b
|
||||
println(a.(int))
|
||||
|
||||
var e int32 = 2
|
||||
var f interface{} = 8 + e
|
||||
println(f.(int32))
|
||||
|
||||
a = 9 + e
|
||||
println(a.(int32))
|
||||
|
||||
var g int = 2
|
||||
a = 10 + g
|
||||
println(a.(int))
|
||||
|
||||
// multiple assignment
|
||||
var foo interface{}
|
||||
foo, a = "hello", 11 + g
|
||||
println(a.(int))
|
||||
println(foo.(string))
|
||||
}
|
||||
|
||||
// Output:
|
||||
// 7
|
||||
// 8
|
||||
// 9
|
||||
// 10
|
||||
// 11
|
||||
// 12
|
||||
// 13
|
||||
// hello
|
||||
|
||||
@@ -2,6 +2,7 @@ package main
|
||||
|
||||
import (
|
||||
"encoding/xml"
|
||||
"errors"
|
||||
"fmt"
|
||||
)
|
||||
|
||||
@@ -10,7 +11,30 @@ type Email struct {
|
||||
Addr string
|
||||
}
|
||||
|
||||
func f(s string, r interface{}) error {
|
||||
func f(r interface{}) error {
|
||||
return withPointerAsInterface(&r)
|
||||
}
|
||||
|
||||
func withPointerAsInterface(r interface{}) error {
|
||||
_ = (r).(*interface{})
|
||||
rp, ok := (r).(*interface{})
|
||||
if !ok {
|
||||
return errors.New("cannot assert to *interface{}")
|
||||
}
|
||||
em, ok := (*rp).(*Email)
|
||||
if !ok {
|
||||
return errors.New("cannot assert to *Email")
|
||||
}
|
||||
em.Where = "work"
|
||||
em.Addr = "bob@work.com"
|
||||
return nil
|
||||
}
|
||||
|
||||
func ff(s string, r interface{}) error {
|
||||
return xml.Unmarshal([]byte(s), r)
|
||||
}
|
||||
|
||||
func fff(s string, r interface{}) error {
|
||||
return xml.Unmarshal([]byte(s), &r)
|
||||
}
|
||||
|
||||
@@ -21,9 +45,19 @@ func main() {
|
||||
</Email>
|
||||
`
|
||||
v := Email{}
|
||||
err := f(data, &v)
|
||||
err := f(&v)
|
||||
fmt.Println(err, v)
|
||||
|
||||
vv := Email{}
|
||||
err = ff(data, &vv)
|
||||
fmt.Println(err, vv)
|
||||
|
||||
vvv := Email{}
|
||||
err = ff(data, &vvv)
|
||||
fmt.Println(err, vvv)
|
||||
}
|
||||
|
||||
// Ouput:
|
||||
// Output:
|
||||
// <nil> {work bob@work.com}
|
||||
// <nil> {work bob@work.com}
|
||||
// <nil> {work bob@work.com}
|
||||
|
||||
24
_test/addr3.go
Normal file
24
_test/addr3.go
Normal file
@@ -0,0 +1,24 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
)
|
||||
|
||||
func main() {
|
||||
var a interface{}
|
||||
a = 2
|
||||
fmt.Println(a)
|
||||
|
||||
var b *interface{}
|
||||
b = &a
|
||||
fmt.Println(*b)
|
||||
|
||||
var c **interface{}
|
||||
c = &b
|
||||
fmt.Println(**c)
|
||||
}
|
||||
|
||||
// Output:
|
||||
// 2
|
||||
// 2
|
||||
// 2
|
||||
114
_test/addr4.go
Normal file
114
_test/addr4.go
Normal file
@@ -0,0 +1,114 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"encoding/json"
|
||||
"fmt"
|
||||
"log"
|
||||
)
|
||||
|
||||
const jsonData = `[
|
||||
"foo",
|
||||
"bar"
|
||||
]`
|
||||
|
||||
const jsonData2 = `[
|
||||
{"foo": "foo"},
|
||||
{"bar": "bar"}
|
||||
]`
|
||||
|
||||
const jsonData3 = `{
|
||||
"foo": "foo",
|
||||
"bar": "bar"
|
||||
}`
|
||||
|
||||
func fromSlice() {
|
||||
var a []interface{}
|
||||
var c, d interface{}
|
||||
c = 2
|
||||
d = 3
|
||||
a = []interface{}{c, d}
|
||||
|
||||
if err := json.Unmarshal([]byte(jsonData), &a); err != nil {
|
||||
log.Fatalln(err)
|
||||
}
|
||||
|
||||
for k, v := range a {
|
||||
fmt.Println(k, ":", v)
|
||||
}
|
||||
}
|
||||
|
||||
func fromEmpty() {
|
||||
var a interface{}
|
||||
var c, d interface{}
|
||||
c = 2
|
||||
d = 3
|
||||
a = []interface{}{c, d}
|
||||
|
||||
if err := json.Unmarshal([]byte(jsonData), &a); err != nil {
|
||||
log.Fatalln(err)
|
||||
}
|
||||
|
||||
b := a.([]interface{})
|
||||
|
||||
for k, v := range b {
|
||||
fmt.Println(k, ":", v)
|
||||
}
|
||||
}
|
||||
|
||||
func sliceOfObjects() {
|
||||
var a interface{}
|
||||
|
||||
if err := json.Unmarshal([]byte(jsonData2), &a); err != nil {
|
||||
log.Fatalln(err)
|
||||
}
|
||||
|
||||
b := a.([]interface{})
|
||||
|
||||
for k, v := range b {
|
||||
fmt.Println(k, ":", v)
|
||||
}
|
||||
}
|
||||
|
||||
func intoMap() {
|
||||
var a interface{}
|
||||
|
||||
if err := json.Unmarshal([]byte(jsonData3), &a); err != nil {
|
||||
log.Fatalln(err)
|
||||
}
|
||||
|
||||
b := a.(map[string]interface{})
|
||||
|
||||
seenFoo := false
|
||||
for k, v := range b {
|
||||
vv := v.(string)
|
||||
if vv != "foo" {
|
||||
if seenFoo {
|
||||
fmt.Println(k, ":", vv)
|
||||
break
|
||||
}
|
||||
kk := k
|
||||
vvv := vv
|
||||
defer fmt.Println(kk, ":", vvv)
|
||||
continue
|
||||
}
|
||||
seenFoo = true
|
||||
fmt.Println(k, ":", vv)
|
||||
}
|
||||
}
|
||||
|
||||
func main() {
|
||||
fromSlice()
|
||||
fromEmpty()
|
||||
sliceOfObjects()
|
||||
intoMap()
|
||||
}
|
||||
|
||||
// Output:
|
||||
// 0 : foo
|
||||
// 1 : bar
|
||||
// 0 : foo
|
||||
// 1 : bar
|
||||
// 0 : map[foo:foo]
|
||||
// 1 : map[bar:bar]
|
||||
// foo : foo
|
||||
// bar : bar
|
||||
62
_test/addr5.go
Normal file
62
_test/addr5.go
Normal file
@@ -0,0 +1,62 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"encoding/json"
|
||||
"fmt"
|
||||
"net/url"
|
||||
)
|
||||
|
||||
func main() {
|
||||
body := []byte(`{
|
||||
"BODY_1": "VALUE_1",
|
||||
"BODY_2": "VALUE_2",
|
||||
"BODY_3": null,
|
||||
"BODY_4": {
|
||||
"BODY_1": "VALUE_1",
|
||||
"BODY_2": "VALUE_2",
|
||||
"BODY_3": null
|
||||
},
|
||||
"BODY_5": [
|
||||
"VALUE_1",
|
||||
"VALUE_2",
|
||||
"VALUE_3"
|
||||
]
|
||||
}`)
|
||||
|
||||
values := url.Values{}
|
||||
|
||||
var rawData map[string]interface{}
|
||||
err := json.Unmarshal(body, &rawData)
|
||||
if err != nil {
|
||||
fmt.Println("can't parse body")
|
||||
return
|
||||
}
|
||||
|
||||
for key, val := range rawData {
|
||||
switch val.(type) {
|
||||
case string, bool, float64:
|
||||
values.Add(key, fmt.Sprint(val))
|
||||
case nil:
|
||||
values.Add(key, "")
|
||||
case map[string]interface{}, []interface{}:
|
||||
jsonVal, err := json.Marshal(val)
|
||||
if err != nil {
|
||||
fmt.Println("can't encode json")
|
||||
return
|
||||
}
|
||||
values.Add(key, string(jsonVal))
|
||||
}
|
||||
}
|
||||
fmt.Println(values.Get("BODY_1"))
|
||||
fmt.Println(values.Get("BODY_2"))
|
||||
fmt.Println(values.Get("BODY_3"))
|
||||
fmt.Println(values.Get("BODY_4"))
|
||||
fmt.Println(values.Get("BODY_5"))
|
||||
}
|
||||
|
||||
// Output:
|
||||
// VALUE_1
|
||||
// VALUE_2
|
||||
//
|
||||
// {"BODY_1":"VALUE_1","BODY_2":"VALUE_2","BODY_3":null}
|
||||
// ["VALUE_1","VALUE_2","VALUE_3"]
|
||||
14
_test/append3.go
Normal file
14
_test/append3.go
Normal file
@@ -0,0 +1,14 @@
|
||||
package main
|
||||
|
||||
import "fmt"
|
||||
|
||||
func main() {
|
||||
a := []int{1, 2}
|
||||
b := [2]int{3, 4}
|
||||
fmt.Println(append(a, b[:]...))
|
||||
fmt.Println(append(a, []int{5, 6}...))
|
||||
}
|
||||
|
||||
// Output:
|
||||
// [1 2 3 4]
|
||||
// [1 2 5 6]
|
||||
12
_test/append4.go
Normal file
12
_test/append4.go
Normal file
@@ -0,0 +1,12 @@
|
||||
package main
|
||||
|
||||
import "fmt"
|
||||
|
||||
func main() {
|
||||
a := []*int{}
|
||||
a = append(a, nil)
|
||||
fmt.Println(a)
|
||||
}
|
||||
|
||||
// Output:
|
||||
// [<nil>]
|
||||
@@ -2,7 +2,6 @@ package main
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"reflect"
|
||||
"time"
|
||||
)
|
||||
|
||||
@@ -10,6 +9,10 @@ type MyWriter interface {
|
||||
Write(p []byte) (i int, err error)
|
||||
}
|
||||
|
||||
type DummyWriter interface {
|
||||
Write(p []byte) (i int, err error)
|
||||
}
|
||||
|
||||
type TestStruct struct{}
|
||||
|
||||
func (t TestStruct) Write(p []byte) (n int, err error) {
|
||||
@@ -25,14 +28,18 @@ type MyStringer interface {
|
||||
String() string
|
||||
}
|
||||
|
||||
type DummyStringer interface {
|
||||
String() string
|
||||
}
|
||||
|
||||
func usesStringer(s MyStringer) {
|
||||
fmt.Println(s.String())
|
||||
}
|
||||
|
||||
func main() {
|
||||
aType := reflect.TypeOf((*MyWriter)(nil)).Elem()
|
||||
|
||||
var t interface{}
|
||||
// TODO(mpl): restore when we can deal with empty interface.
|
||||
// var t interface{}
|
||||
var t DummyWriter
|
||||
t = TestStruct{}
|
||||
var tw MyWriter
|
||||
var ok bool
|
||||
@@ -45,8 +52,6 @@ func main() {
|
||||
}
|
||||
n, _ := t.(MyWriter).Write([]byte("hello world"))
|
||||
fmt.Println(n)
|
||||
bType := reflect.TypeOf(TestStruct{})
|
||||
fmt.Println(bType.Implements(aType))
|
||||
|
||||
// not redundant with the above, because it goes through a slightly different code path.
|
||||
if _, ok := t.(MyWriter); !ok {
|
||||
@@ -56,6 +61,8 @@ func main() {
|
||||
fmt.Println("TestStruct implements MyWriter")
|
||||
}
|
||||
|
||||
// TODO(mpl): restore
|
||||
/*
|
||||
t = 42
|
||||
foo, ok := t.(MyWriter)
|
||||
if !ok {
|
||||
@@ -70,8 +77,10 @@ func main() {
|
||||
} else {
|
||||
fmt.Println("42 implements MyWriter")
|
||||
}
|
||||
*/
|
||||
|
||||
var tt interface{}
|
||||
// var tt interface{}
|
||||
var tt DummyStringer
|
||||
tt = time.Nanosecond
|
||||
var myD MyStringer
|
||||
myD, ok = tt.(MyStringer)
|
||||
@@ -82,9 +91,6 @@ func main() {
|
||||
usesStringer(myD)
|
||||
}
|
||||
fmt.Println(tt.(MyStringer).String())
|
||||
cType := reflect.TypeOf((*MyStringer)(nil)).Elem()
|
||||
dType := reflect.TypeOf(time.Nanosecond)
|
||||
fmt.Println(dType.Implements(cType))
|
||||
|
||||
if _, ok := tt.(MyStringer); !ok {
|
||||
fmt.Println("time.Nanosecond does not implement MyStringer")
|
||||
@@ -92,6 +98,8 @@ func main() {
|
||||
fmt.Println("time.Nanosecond implements MyStringer")
|
||||
}
|
||||
|
||||
// TODO(mpl): restore
|
||||
/*
|
||||
tt = 42
|
||||
bar, ok := tt.(MyStringer)
|
||||
if !ok {
|
||||
@@ -106,20 +114,15 @@ func main() {
|
||||
} else {
|
||||
fmt.Println("42 implements MyStringer")
|
||||
}
|
||||
*/
|
||||
}
|
||||
|
||||
// Output:
|
||||
// TestStruct implements MyWriter
|
||||
// 11
|
||||
// 11
|
||||
// true
|
||||
// TestStruct implements MyWriter
|
||||
// 42 does not implement MyWriter
|
||||
// 42 does not implement MyWriter
|
||||
// time.Nanosecond implements MyStringer
|
||||
// 1ns
|
||||
// 1ns
|
||||
// true
|
||||
// time.Nanosecond implements MyStringer
|
||||
// 42 does not implement MyStringer
|
||||
// 42 does not implement MyStringer
|
||||
|
||||
@@ -12,6 +12,10 @@ func (t TestStruct) String() string {
|
||||
return "hello world"
|
||||
}
|
||||
|
||||
type DummyStringer interface{
|
||||
String() string
|
||||
}
|
||||
|
||||
func main() {
|
||||
aType := reflect.TypeOf((*fmt.Stringer)(nil)).Elem()
|
||||
|
||||
@@ -52,7 +56,9 @@ func main() {
|
||||
return
|
||||
}
|
||||
|
||||
var tt interface{}
|
||||
// TODO(mpl): restore when fixed
|
||||
// var tt interface{}
|
||||
var tt DummyStringer
|
||||
tt = TestStruct{}
|
||||
ss, ok := tt.(fmt.Stringer)
|
||||
if !ok {
|
||||
@@ -61,9 +67,6 @@ func main() {
|
||||
}
|
||||
fmt.Println(ss.String())
|
||||
fmt.Println(tt.(fmt.Stringer).String())
|
||||
// TODO(mpl): uncomment when fixed
|
||||
// cType := reflect.TypeOf(TestStruct{})
|
||||
// fmt.Println(cType.Implements(aType))
|
||||
|
||||
if _, ok := tt.(fmt.Stringer); !ok {
|
||||
fmt.Println("TestStuct does not implement fmt.Stringer")
|
||||
|
||||
@@ -4,10 +4,16 @@ func f(a []int) interface{} {
|
||||
return cap(a)
|
||||
}
|
||||
|
||||
func g(a []int) int {
|
||||
return cap(a)
|
||||
}
|
||||
|
||||
func main() {
|
||||
a := []int{1, 2}
|
||||
println(g(a))
|
||||
println(f(a).(int))
|
||||
}
|
||||
|
||||
// Output:
|
||||
// 2
|
||||
// 2
|
||||
|
||||
18
_test/closure10.go
Normal file
18
_test/closure10.go
Normal file
@@ -0,0 +1,18 @@
|
||||
package main
|
||||
|
||||
func main() {
|
||||
foos := []func(){}
|
||||
|
||||
for i := 0; i < 3; i++ {
|
||||
a, b := i, i
|
||||
foos = append(foos, func() { println(i, a, b) })
|
||||
}
|
||||
foos[0]()
|
||||
foos[1]()
|
||||
foos[2]()
|
||||
}
|
||||
|
||||
// Output:
|
||||
// 3 0 0
|
||||
// 3 1 1
|
||||
// 3 2 2
|
||||
22
_test/closure11.go
Normal file
22
_test/closure11.go
Normal file
@@ -0,0 +1,22 @@
|
||||
package main
|
||||
|
||||
type T struct {
|
||||
F func()
|
||||
}
|
||||
|
||||
func main() {
|
||||
foos := []T{}
|
||||
|
||||
for i := 0; i < 3; i++ {
|
||||
a := i
|
||||
foos = append(foos, T{func() { println(i, a) }})
|
||||
}
|
||||
foos[0].F()
|
||||
foos[1].F()
|
||||
foos[2].F()
|
||||
}
|
||||
|
||||
// Output:
|
||||
// 3 0
|
||||
// 3 1
|
||||
// 3 2
|
||||
25
_test/closure12.go
Normal file
25
_test/closure12.go
Normal file
@@ -0,0 +1,25 @@
|
||||
package main
|
||||
|
||||
import "fmt"
|
||||
|
||||
type T struct {
|
||||
F func()
|
||||
}
|
||||
|
||||
func main() {
|
||||
foos := []T{}
|
||||
|
||||
for i := 0; i < 3; i++ {
|
||||
a := i
|
||||
n := fmt.Sprintf("i=%d", i)
|
||||
foos = append(foos, T{func() { println(i, a, n) }})
|
||||
}
|
||||
foos[0].F()
|
||||
foos[1].F()
|
||||
foos[2].F()
|
||||
}
|
||||
|
||||
// Output:
|
||||
// 3 0 i=0
|
||||
// 3 1 i=1
|
||||
// 3 2 i=2
|
||||
18
_test/closure9.go
Normal file
18
_test/closure9.go
Normal file
@@ -0,0 +1,18 @@
|
||||
package main
|
||||
|
||||
func main() {
|
||||
foos := []func(){}
|
||||
|
||||
for i := 0; i < 3; i++ {
|
||||
a := i
|
||||
foos = append(foos, func() { println(i, a) })
|
||||
}
|
||||
foos[0]()
|
||||
foos[1]()
|
||||
foos[2]()
|
||||
}
|
||||
|
||||
// Output:
|
||||
// 3 0
|
||||
// 3 1
|
||||
// 3 2
|
||||
@@ -32,6 +32,7 @@ func interfaceAsInterfaces() {
|
||||
println("nope")
|
||||
return
|
||||
}
|
||||
fmt.Println(d)
|
||||
|
||||
for _, v := range d {
|
||||
fmt.Println(v)
|
||||
@@ -46,5 +47,6 @@ func main() {
|
||||
// Output:
|
||||
// 2
|
||||
// 3
|
||||
// [2 3]
|
||||
// 2
|
||||
// 3
|
||||
|
||||
30
_test/composite17.go
Normal file
30
_test/composite17.go
Normal file
@@ -0,0 +1,30 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"html/template"
|
||||
)
|
||||
|
||||
var str = `{{ stringOr .Data "test" }}`
|
||||
|
||||
func main() {
|
||||
_, err := template.New("test").
|
||||
Funcs(template.FuncMap{
|
||||
"stringOr": stringOr,
|
||||
}).
|
||||
Parse(str)
|
||||
if err != nil {
|
||||
println(err.Error())
|
||||
return
|
||||
}
|
||||
println("success")
|
||||
}
|
||||
|
||||
func stringOr(v, def string) string {
|
||||
if v == "" {
|
||||
return def
|
||||
}
|
||||
return v
|
||||
}
|
||||
|
||||
// Output:
|
||||
// success
|
||||
15
_test/const22.go
Normal file
15
_test/const22.go
Normal file
@@ -0,0 +1,15 @@
|
||||
package main
|
||||
|
||||
const (
|
||||
numDec uint8 = (1 << iota) / 2
|
||||
numHex
|
||||
numOct
|
||||
numFloat
|
||||
)
|
||||
|
||||
func main() {
|
||||
println(13 & (numHex | numOct))
|
||||
}
|
||||
|
||||
// Output:
|
||||
// 1
|
||||
12
_test/const23.go
Normal file
12
_test/const23.go
Normal file
@@ -0,0 +1,12 @@
|
||||
package main
|
||||
|
||||
const maxlen = len("hello")
|
||||
|
||||
var gfm = [maxlen]byte{}
|
||||
|
||||
func main() {
|
||||
println(len(gfm))
|
||||
}
|
||||
|
||||
// Output:
|
||||
// 5
|
||||
14
_test/const24.go
Normal file
14
_test/const24.go
Normal file
@@ -0,0 +1,14 @@
|
||||
package main
|
||||
|
||||
var aa = [...]int{1, 2, 3}
|
||||
|
||||
const maxlen = cap(aa)
|
||||
|
||||
var gfm = [maxlen]byte{}
|
||||
|
||||
func main() {
|
||||
println(len(gfm))
|
||||
}
|
||||
|
||||
// Output:
|
||||
// 3
|
||||
11
_test/const25.go
Normal file
11
_test/const25.go
Normal file
@@ -0,0 +1,11 @@
|
||||
package main
|
||||
|
||||
const (
|
||||
FGBlack Attribute = iota + 30
|
||||
)
|
||||
|
||||
type Attribute int
|
||||
|
||||
func main() {
|
||||
println(FGBlack)
|
||||
}
|
||||
9
_test/d1/d1.go
Normal file
9
_test/d1/d1.go
Normal file
@@ -0,0 +1,9 @@
|
||||
package d1
|
||||
|
||||
type T struct {
|
||||
Name string
|
||||
}
|
||||
|
||||
func (t *T) F() { println(t.Name) }
|
||||
|
||||
func NewT(s string) *T { return &T{s} }
|
||||
8
_test/d2/d2.go
Normal file
8
_test/d2/d2.go
Normal file
@@ -0,0 +1,8 @@
|
||||
package d2
|
||||
|
||||
import "github.com/traefik/yaegi/_test/d1"
|
||||
|
||||
var (
|
||||
X = d1.NewT("test")
|
||||
F = X.F
|
||||
)
|
||||
11
_test/d3.go
Normal file
11
_test/d3.go
Normal file
@@ -0,0 +1,11 @@
|
||||
package main
|
||||
|
||||
import "github.com/traefik/yaegi/_test/d2"
|
||||
|
||||
func main() {
|
||||
f := d2.F
|
||||
f()
|
||||
}
|
||||
|
||||
// Output:
|
||||
// test
|
||||
@@ -2,10 +2,15 @@ package main
|
||||
|
||||
func f1(a int) interface{} { return a + 1 }
|
||||
|
||||
func f2(a int64) interface{} { return a + 1 }
|
||||
|
||||
func main() {
|
||||
c := f1(3)
|
||||
println(c.(int))
|
||||
b := f2(3)
|
||||
println(b.(int64))
|
||||
}
|
||||
|
||||
// Output:
|
||||
// 4
|
||||
// 4
|
||||
|
||||
@@ -2,7 +2,13 @@ package main
|
||||
|
||||
func f1(a int) int { return a + 1 }
|
||||
|
||||
func f2(a int) interface{} { return f1(a) }
|
||||
func f2(a int) interface{} {
|
||||
// TODO: re-enable the optimized case below, once we've figured out why it
|
||||
// interferes with the empty interface model.
|
||||
// return f1(a)
|
||||
var foo interface{} = f1(a)
|
||||
return foo
|
||||
}
|
||||
|
||||
func main() {
|
||||
c := f2(3)
|
||||
|
||||
17
_test/interface48.go
Normal file
17
_test/interface48.go
Normal file
@@ -0,0 +1,17 @@
|
||||
package main
|
||||
|
||||
import "fmt"
|
||||
|
||||
type I1 interface{ A }
|
||||
|
||||
type A = I2
|
||||
|
||||
type I2 interface{ F() I1 }
|
||||
|
||||
func main() {
|
||||
var i I1
|
||||
fmt.Println(i)
|
||||
}
|
||||
|
||||
// Output:
|
||||
// <nil>
|
||||
45
_test/interface49.go
Normal file
45
_test/interface49.go
Normal file
@@ -0,0 +1,45 @@
|
||||
package main
|
||||
|
||||
type Descriptor interface {
|
||||
ParentFile() FileDescriptor
|
||||
}
|
||||
|
||||
type FileDescriptor interface {
|
||||
Enums() EnumDescriptors
|
||||
Services() ServiceDescriptors
|
||||
}
|
||||
|
||||
type EnumDescriptors interface {
|
||||
Get(i int) EnumDescriptor
|
||||
}
|
||||
|
||||
type EnumDescriptor interface {
|
||||
Values() EnumValueDescriptors
|
||||
}
|
||||
|
||||
type EnumValueDescriptors interface {
|
||||
Get(i int) EnumValueDescriptor
|
||||
}
|
||||
|
||||
type EnumValueDescriptor interface {
|
||||
Descriptor
|
||||
}
|
||||
|
||||
type ServiceDescriptors interface {
|
||||
Get(i int) ServiceDescriptor
|
||||
}
|
||||
|
||||
type ServiceDescriptor interface {
|
||||
Descriptor
|
||||
isServiceDescriptor
|
||||
}
|
||||
|
||||
type isServiceDescriptor interface{ ProtoType(ServiceDescriptor) }
|
||||
|
||||
func main() {
|
||||
var d Descriptor
|
||||
println(d == nil)
|
||||
}
|
||||
|
||||
// Output:
|
||||
// true
|
||||
13
_test/interface50.go
Normal file
13
_test/interface50.go
Normal file
@@ -0,0 +1,13 @@
|
||||
package main
|
||||
|
||||
func main() {
|
||||
a := true
|
||||
var b interface{} = 5
|
||||
println(b.(int))
|
||||
b = a == true
|
||||
println(b.(bool))
|
||||
}
|
||||
|
||||
// Output:
|
||||
// 5
|
||||
// true
|
||||
23
_test/interface51.go
Normal file
23
_test/interface51.go
Normal file
@@ -0,0 +1,23 @@
|
||||
package main
|
||||
|
||||
type Error interface {
|
||||
error
|
||||
Message() string
|
||||
}
|
||||
|
||||
type T struct {
|
||||
Msg string
|
||||
}
|
||||
|
||||
func (t *T) Error() string { return t.Msg }
|
||||
func (t *T) Message() string { return "message:" + t.Msg }
|
||||
|
||||
func newError() Error { return &T{"test"} }
|
||||
|
||||
func main() {
|
||||
e := newError()
|
||||
println(e.Error())
|
||||
}
|
||||
|
||||
// Output:
|
||||
// test
|
||||
17
_test/interface52.go
Normal file
17
_test/interface52.go
Normal file
@@ -0,0 +1,17 @@
|
||||
package main
|
||||
|
||||
import "testing"
|
||||
|
||||
func main() {
|
||||
t := testing.T{}
|
||||
var tb testing.TB
|
||||
tb = &t
|
||||
if tb.TempDir() == "" {
|
||||
println("FAIL")
|
||||
return
|
||||
}
|
||||
println("PASS")
|
||||
}
|
||||
|
||||
// Output:
|
||||
// PASS
|
||||
40
_test/issue-1007.go
Normal file
40
_test/issue-1007.go
Normal file
@@ -0,0 +1,40 @@
|
||||
package main
|
||||
|
||||
type TypeA struct {
|
||||
B TypeB
|
||||
}
|
||||
|
||||
type TypeB struct {
|
||||
C1 *TypeC
|
||||
C2 *TypeC
|
||||
}
|
||||
|
||||
type TypeC struct {
|
||||
Val string
|
||||
D *TypeD
|
||||
D2 *TypeD
|
||||
}
|
||||
|
||||
type TypeD struct {
|
||||
Name string
|
||||
}
|
||||
|
||||
func build() *TypeA {
|
||||
return &TypeA{
|
||||
B: TypeB{
|
||||
C2: &TypeC{Val: "22"},
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
func Bar(s string) string {
|
||||
a := build()
|
||||
return s + "-" + a.B.C2.Val
|
||||
}
|
||||
|
||||
func main() {
|
||||
println(Bar("test"))
|
||||
}
|
||||
|
||||
// Output:
|
||||
// test-22
|
||||
22
_test/issue-1010.go
Normal file
22
_test/issue-1010.go
Normal file
@@ -0,0 +1,22 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"encoding/json"
|
||||
"fmt"
|
||||
)
|
||||
|
||||
type MyJsonMarshaler struct{ n int }
|
||||
|
||||
func (m MyJsonMarshaler) MarshalJSON() ([]byte, error) {
|
||||
return []byte(fmt.Sprintf(`{"num": %d}`, m.n)), nil
|
||||
}
|
||||
|
||||
func main() {
|
||||
ch := make(chan json.Marshaler, 1)
|
||||
ch <- MyJsonMarshaler{2}
|
||||
m, err := json.Marshal(<-ch)
|
||||
fmt.Println(string(m), err)
|
||||
}
|
||||
|
||||
// Output:
|
||||
// {"num":2} <nil>
|
||||
17
_test/issue-1022.go
Normal file
17
_test/issue-1022.go
Normal file
@@ -0,0 +1,17 @@
|
||||
package main
|
||||
|
||||
import "fmt"
|
||||
|
||||
func main() {
|
||||
defer func() {
|
||||
r := recover()
|
||||
if r != nil {
|
||||
fmt.Println(r)
|
||||
}
|
||||
}()
|
||||
|
||||
panic("Ho Ho Ho!")
|
||||
}
|
||||
|
||||
// Output:
|
||||
// Ho Ho Ho!
|
||||
23
_test/issue-1052.go
Normal file
23
_test/issue-1052.go
Normal file
@@ -0,0 +1,23 @@
|
||||
package main
|
||||
|
||||
import "fmt"
|
||||
|
||||
func main() {
|
||||
a, b := 1, 1
|
||||
for i := 0; i < 10; i++ {
|
||||
fmt.Println(a)
|
||||
a, b = b, a+b
|
||||
}
|
||||
}
|
||||
|
||||
// Output:
|
||||
// 1
|
||||
// 1
|
||||
// 2
|
||||
// 3
|
||||
// 5
|
||||
// 8
|
||||
// 13
|
||||
// 21
|
||||
// 34
|
||||
// 55
|
||||
20
_test/issue-1065.go
Normal file
20
_test/issue-1065.go
Normal file
@@ -0,0 +1,20 @@
|
||||
package main
|
||||
|
||||
import "fmt"
|
||||
|
||||
type AST struct {
|
||||
Num int
|
||||
Children []AST
|
||||
}
|
||||
|
||||
func newAST(num int, root AST, children ...AST) AST {
|
||||
return AST{num, append([]AST{root}, children...)}
|
||||
}
|
||||
|
||||
func main() {
|
||||
ast := newAST(1, AST{}, AST{})
|
||||
fmt.Println(ast)
|
||||
}
|
||||
|
||||
// Output:
|
||||
// {1 [{0 []} {0 []}]}
|
||||
19
_test/issue-1068.go
Normal file
19
_test/issue-1068.go
Normal file
@@ -0,0 +1,19 @@
|
||||
package main
|
||||
|
||||
type I interface {
|
||||
Hello()
|
||||
}
|
||||
|
||||
type T struct{}
|
||||
|
||||
func (t T) Hello() { println("hello") }
|
||||
|
||||
type I2 I
|
||||
|
||||
func main() {
|
||||
var i I2 = T{}
|
||||
i.Hello()
|
||||
}
|
||||
|
||||
// Output:
|
||||
// hello
|
||||
14
_test/issue-1088.go
Normal file
14
_test/issue-1088.go
Normal file
@@ -0,0 +1,14 @@
|
||||
package main
|
||||
|
||||
import "fmt"
|
||||
|
||||
func main() {
|
||||
for i, ch := range "日本語" {
|
||||
fmt.Printf("%#U starts at byte position %d\n", ch, i)
|
||||
}
|
||||
}
|
||||
|
||||
// Output:
|
||||
// U+65E5 '日' starts at byte position 0
|
||||
// U+672C '本' starts at byte position 3
|
||||
// U+8A9E '語' starts at byte position 6
|
||||
13
_test/issue-1089.go
Normal file
13
_test/issue-1089.go
Normal file
@@ -0,0 +1,13 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"time"
|
||||
)
|
||||
|
||||
func main() {
|
||||
fmt.Println(`"` + time.RFC3339Nano + `"`)
|
||||
}
|
||||
|
||||
// Output:
|
||||
// "2006-01-02T15:04:05.999999999Z07:00"
|
||||
13
_test/issue-1093.go
Normal file
13
_test/issue-1093.go
Normal file
@@ -0,0 +1,13 @@
|
||||
package main
|
||||
|
||||
func b() string {
|
||||
return "b"
|
||||
}
|
||||
|
||||
func main() {
|
||||
var x int
|
||||
x = "a" + b()
|
||||
}
|
||||
|
||||
// Error:
|
||||
// 9:6: cannot use type string as type int in assignment
|
||||
12
_test/issue-1094.go
Normal file
12
_test/issue-1094.go
Normal file
@@ -0,0 +1,12 @@
|
||||
package main
|
||||
|
||||
import "fmt"
|
||||
|
||||
func main() {
|
||||
var x interface{}
|
||||
x = "a" + fmt.Sprintf("b")
|
||||
fmt.Printf("%v %T\n", x, x)
|
||||
}
|
||||
|
||||
// Output:
|
||||
// ab string
|
||||
17
_test/issue-1101.go
Normal file
17
_test/issue-1101.go
Normal file
@@ -0,0 +1,17 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"net/http"
|
||||
)
|
||||
|
||||
func main() {
|
||||
method := "POST"
|
||||
switch method {
|
||||
case http.MethodPost:
|
||||
fmt.Println("It's a post!")
|
||||
}
|
||||
}
|
||||
|
||||
// Output:
|
||||
// It's a post!
|
||||
24
_test/issue-1115.go
Normal file
24
_test/issue-1115.go
Normal file
@@ -0,0 +1,24 @@
|
||||
package main
|
||||
|
||||
import "fmt"
|
||||
|
||||
func main() {
|
||||
outer:
|
||||
for y := 0; y < 10; y++ {
|
||||
for x := 0; x < 10; x++ {
|
||||
if x == 5 && y == 5 {
|
||||
break outer
|
||||
}
|
||||
}
|
||||
fmt.Println(y)
|
||||
}
|
||||
fmt.Println("Yay! I finished!")
|
||||
}
|
||||
|
||||
// Output:
|
||||
// 0
|
||||
// 1
|
||||
// 2
|
||||
// 3
|
||||
// 4
|
||||
// Yay! I finished!
|
||||
23
_test/issue-1126.go
Normal file
23
_test/issue-1126.go
Normal file
@@ -0,0 +1,23 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"errors"
|
||||
"fmt"
|
||||
"strings"
|
||||
)
|
||||
|
||||
func main() {
|
||||
err := errors.New("hello there")
|
||||
|
||||
switch true {
|
||||
case err == nil:
|
||||
break
|
||||
case strings.Contains(err.Error(), "hello"):
|
||||
fmt.Println("True!")
|
||||
default:
|
||||
fmt.Println("False!")
|
||||
}
|
||||
}
|
||||
|
||||
// Output:
|
||||
// True!
|
||||
10
_test/issue-1128.go
Normal file
10
_test/issue-1128.go
Normal file
@@ -0,0 +1,10 @@
|
||||
package main
|
||||
|
||||
import "net"
|
||||
|
||||
func main() {
|
||||
c := append(net.Buffers{}, []byte{})
|
||||
println(len(c))
|
||||
}
|
||||
|
||||
// Output: 1
|
||||
20
_test/issue-1134.go
Normal file
20
_test/issue-1134.go
Normal file
@@ -0,0 +1,20 @@
|
||||
package main
|
||||
|
||||
type I interface {
|
||||
Hello()
|
||||
}
|
||||
|
||||
type T struct {
|
||||
Name string
|
||||
Child []*T
|
||||
}
|
||||
|
||||
func (t *T) Hello() { println("Hello", t.Name) }
|
||||
|
||||
func main() {
|
||||
var i I = new(T)
|
||||
i.Hello()
|
||||
}
|
||||
|
||||
// Output:
|
||||
// Hello
|
||||
22
_test/issue-1136.go
Normal file
22
_test/issue-1136.go
Normal file
@@ -0,0 +1,22 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"io"
|
||||
)
|
||||
|
||||
type T struct {
|
||||
r io.Reader
|
||||
}
|
||||
|
||||
func (t *T) Read(p []byte) (n int, err error) { n, err = t.r.Read(p); return }
|
||||
|
||||
func main() {
|
||||
x := io.LimitedReader{}
|
||||
y := io.Reader(&x)
|
||||
y = &T{y}
|
||||
fmt.Println(y.Read([]byte("")))
|
||||
}
|
||||
|
||||
// Output:
|
||||
// 0 EOF
|
||||
14
_test/issue-1145.go
Normal file
14
_test/issue-1145.go
Normal file
@@ -0,0 +1,14 @@
|
||||
package main
|
||||
|
||||
import "sync"
|
||||
|
||||
type F func()
|
||||
|
||||
func main() {
|
||||
var wg sync.WaitGroup
|
||||
var f F = wg.Done
|
||||
println(f != nil)
|
||||
}
|
||||
|
||||
// Output:
|
||||
// true
|
||||
25
_test/issue-1156.go
Normal file
25
_test/issue-1156.go
Normal file
@@ -0,0 +1,25 @@
|
||||
package main
|
||||
|
||||
type myInterface interface {
|
||||
myFunc() string
|
||||
}
|
||||
|
||||
type V struct{}
|
||||
|
||||
func (v *V) myFunc() string { return "hello" }
|
||||
|
||||
type U struct {
|
||||
v myInterface
|
||||
}
|
||||
|
||||
func (u *U) myFunc() string { return u.v.myFunc() }
|
||||
|
||||
func main() {
|
||||
x := V{}
|
||||
y := myInterface(&x)
|
||||
y = &U{y}
|
||||
println(y.myFunc())
|
||||
}
|
||||
|
||||
// Output:
|
||||
// hello
|
||||
41
_test/issue-1163.go
Normal file
41
_test/issue-1163.go
Normal file
@@ -0,0 +1,41 @@
|
||||
package main
|
||||
|
||||
import "fmt"
|
||||
|
||||
type WidgetEvent struct {
|
||||
Nothing string
|
||||
}
|
||||
|
||||
type WidgetControl interface {
|
||||
HandleEvent(e *WidgetEvent)
|
||||
}
|
||||
|
||||
type Button struct{}
|
||||
|
||||
func (b *Button) HandleEvent(e *WidgetEvent) {
|
||||
}
|
||||
|
||||
type WindowEvent struct {
|
||||
Something int
|
||||
}
|
||||
|
||||
type Window struct {
|
||||
Widget WidgetControl
|
||||
}
|
||||
|
||||
func (w *Window) HandleEvent(e *WindowEvent) {
|
||||
}
|
||||
|
||||
func main() {
|
||||
window := &Window{
|
||||
Widget: &Button{},
|
||||
}
|
||||
windowevent := &WindowEvent{}
|
||||
// The next line uses the signature from the wrong method, resulting in an error.
|
||||
// Renaming one of the clashing method names fixes the problem.
|
||||
window.HandleEvent(windowevent)
|
||||
fmt.Println("OK!")
|
||||
}
|
||||
|
||||
// Output:
|
||||
// OK!
|
||||
24
_test/issue-1166.go
Normal file
24
_test/issue-1166.go
Normal file
@@ -0,0 +1,24 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"io"
|
||||
)
|
||||
|
||||
type T []byte
|
||||
|
||||
func (t *T) Write(p []byte) (n int, err error) { *t = append(*t, p...); return len(p), nil }
|
||||
|
||||
func foo(w io.Writer) {
|
||||
a := w.(*T)
|
||||
fmt.Fprint(a, "test")
|
||||
fmt.Printf("%s\n", *a)
|
||||
}
|
||||
|
||||
func main() {
|
||||
x := T{}
|
||||
foo(&x)
|
||||
}
|
||||
|
||||
// Output:
|
||||
// test
|
||||
19
_test/issue-1167.go
Normal file
19
_test/issue-1167.go
Normal file
@@ -0,0 +1,19 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"crypto/ecdsa"
|
||||
"crypto/elliptic"
|
||||
"crypto/rand"
|
||||
)
|
||||
|
||||
func main() {
|
||||
key, err := ecdsa.GenerateKey(elliptic.P256(), rand.Reader)
|
||||
if err != nil {
|
||||
panic(err)
|
||||
}
|
||||
pub := key.Public().(*ecdsa.PublicKey)
|
||||
println(pub.Params().Name)
|
||||
}
|
||||
|
||||
// Output:
|
||||
// P-256
|
||||
10
_test/issue-1173.go
Normal file
10
_test/issue-1173.go
Normal file
@@ -0,0 +1,10 @@
|
||||
package main
|
||||
|
||||
var real = func() { println("Hello") }
|
||||
|
||||
func main() {
|
||||
real()
|
||||
}
|
||||
|
||||
// Output:
|
||||
// Hello
|
||||
18
_test/issue-1175.go
Normal file
18
_test/issue-1175.go
Normal file
@@ -0,0 +1,18 @@
|
||||
package main
|
||||
|
||||
type Level int8
|
||||
|
||||
const (
|
||||
a Level = -1
|
||||
b Level = 5
|
||||
d = b - a + 1
|
||||
)
|
||||
|
||||
type counters [d]int
|
||||
|
||||
func main() {
|
||||
println(len(counters{}))
|
||||
}
|
||||
|
||||
// Output:
|
||||
// 7
|
||||
13
_test/issue-1177.go
Normal file
13
_test/issue-1177.go
Normal file
@@ -0,0 +1,13 @@
|
||||
package main
|
||||
|
||||
type counters [3][16]int
|
||||
|
||||
func main() {
|
||||
cs := &counters{}
|
||||
p := &cs[0][1]
|
||||
*p = 2
|
||||
println(cs[0][1])
|
||||
}
|
||||
|
||||
// Output:
|
||||
// 2
|
||||
23
_test/issue-1179.go
Normal file
23
_test/issue-1179.go
Normal file
@@ -0,0 +1,23 @@
|
||||
package main
|
||||
|
||||
type I interface {
|
||||
F()
|
||||
}
|
||||
|
||||
type T struct {
|
||||
Name string
|
||||
}
|
||||
|
||||
func (t *T) F() { println("in F", t.Name) }
|
||||
|
||||
func NewI(s string) I { return newT(s) }
|
||||
|
||||
func newT(s string) *T { return &T{s} }
|
||||
|
||||
func main() {
|
||||
i := NewI("test")
|
||||
i.F()
|
||||
}
|
||||
|
||||
// Output:
|
||||
// in F test
|
||||
10
_test/issue-1181.go
Normal file
10
_test/issue-1181.go
Normal file
@@ -0,0 +1,10 @@
|
||||
package main
|
||||
|
||||
func main() {
|
||||
a, b := 1, 2
|
||||
a, b = b, -a
|
||||
println(a, b)
|
||||
}
|
||||
|
||||
// Output:
|
||||
// 2 -1
|
||||
20
_test/issue-1185.go
Normal file
20
_test/issue-1185.go
Normal file
@@ -0,0 +1,20 @@
|
||||
package main
|
||||
|
||||
import "io"
|
||||
|
||||
type B []byte
|
||||
|
||||
func (b B) Write(p []byte) (n int, err error) {
|
||||
b = p
|
||||
return len(p), nil
|
||||
}
|
||||
|
||||
func main() {
|
||||
b := B{}
|
||||
a := make([]io.Writer, 0)
|
||||
a = append(a, b)
|
||||
println(len(a))
|
||||
}
|
||||
|
||||
// Output:
|
||||
// 1
|
||||
15
_test/issue-981.go
Normal file
15
_test/issue-981.go
Normal file
@@ -0,0 +1,15 @@
|
||||
package main
|
||||
|
||||
import "fmt"
|
||||
|
||||
func main() {
|
||||
dp := make(map[int]int)
|
||||
dp[0] = 1
|
||||
for i := 1; i < 10; i++ {
|
||||
dp[i] = dp[i-1] + dp[i-2]
|
||||
}
|
||||
fmt.Printf("%v\n", dp)
|
||||
}
|
||||
|
||||
// Output:
|
||||
// map[0:1 1:1 2:2 3:3 4:5 5:8 6:13 7:21 8:34 9:55]
|
||||
27
_test/map30.go
Normal file
27
_test/map30.go
Normal file
@@ -0,0 +1,27 @@
|
||||
package main
|
||||
|
||||
import "strings"
|
||||
|
||||
func f(s string) string { return "hello " + s }
|
||||
|
||||
func g(s string) string { return "hi " + s }
|
||||
|
||||
var methods = map[string]func(string) string{
|
||||
"f": f,
|
||||
"h": strings.ToLower,
|
||||
}
|
||||
|
||||
func main() {
|
||||
methods["i"] = strings.ToUpper
|
||||
methods["g"] = g
|
||||
println(methods["f"]("test"))
|
||||
println(methods["g"]("test"))
|
||||
println(methods["i"]("test"))
|
||||
println(methods["h"]("TEST"))
|
||||
}
|
||||
|
||||
// Output:
|
||||
// hello test
|
||||
// hi test
|
||||
// TEST
|
||||
// test
|
||||
@@ -12,10 +12,16 @@ type Hi interface {
|
||||
Hello() string
|
||||
}
|
||||
|
||||
type Hey interface {
|
||||
Hello() string
|
||||
}
|
||||
|
||||
func (r *Root) Hello() string { return "Hello " + r.Name }
|
||||
|
||||
func main() {
|
||||
var one interface{} = &One{Root{Name: "test2"}}
|
||||
// TODO(mpl): restore when type assertions work again.
|
||||
// var one interface{} = &One{Root{Name: "test2"}}
|
||||
var one Hey = &One{Root{Name: "test2"}}
|
||||
println(one.(Hi).Hello())
|
||||
}
|
||||
|
||||
|
||||
26
_test/method36.go
Normal file
26
_test/method36.go
Normal file
@@ -0,0 +1,26 @@
|
||||
package main
|
||||
|
||||
type I interface{ Hello() }
|
||||
|
||||
type T struct{ Name string }
|
||||
|
||||
func (t *T) Hello() { println("Hello", t.Name) }
|
||||
|
||||
type FT func(i I)
|
||||
|
||||
type ST struct{ Handler FT }
|
||||
|
||||
func newF() FT {
|
||||
return func(i I) {
|
||||
i.Hello()
|
||||
}
|
||||
}
|
||||
|
||||
func main() {
|
||||
st := &ST{}
|
||||
st.Handler = newF()
|
||||
st.Handler(&T{"test"})
|
||||
}
|
||||
|
||||
// Output:
|
||||
// Hello test
|
||||
19
_test/method37.go
Normal file
19
_test/method37.go
Normal file
@@ -0,0 +1,19 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"net"
|
||||
"os"
|
||||
)
|
||||
|
||||
func writeBufs(bufs ...[]byte) error {
|
||||
b := net.Buffers(bufs)
|
||||
_, err := b.WriteTo(os.Stdout)
|
||||
return err
|
||||
}
|
||||
|
||||
func main() {
|
||||
writeBufs([]byte("hello"))
|
||||
}
|
||||
|
||||
// Output:
|
||||
// hello
|
||||
28
_test/ret8.go
Normal file
28
_test/ret8.go
Normal file
@@ -0,0 +1,28 @@
|
||||
package main
|
||||
|
||||
import "fmt"
|
||||
|
||||
type CustomError string
|
||||
|
||||
func (s CustomError) Error() string {
|
||||
return string(s)
|
||||
}
|
||||
|
||||
func NewCustomError(errorText string) CustomError {
|
||||
return CustomError(errorText)
|
||||
}
|
||||
|
||||
func fail() (err error) {
|
||||
return NewCustomError("Everything is going wrong!")
|
||||
}
|
||||
|
||||
func main() {
|
||||
fmt.Println(fail())
|
||||
var myError error
|
||||
myError = NewCustomError("ok")
|
||||
fmt.Println(myError)
|
||||
}
|
||||
|
||||
// Output:
|
||||
// Everything is going wrong!
|
||||
// ok
|
||||
@@ -7,7 +7,7 @@ import (
|
||||
|
||||
const (
|
||||
period = 100 * time.Millisecond
|
||||
precision = 5 * time.Millisecond
|
||||
precision = 7 * time.Millisecond
|
||||
)
|
||||
|
||||
func main() {
|
||||
|
||||
41
_test/struct59.go
Normal file
41
_test/struct59.go
Normal file
@@ -0,0 +1,41 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
)
|
||||
|
||||
type A struct {
|
||||
B map[string]*B
|
||||
C map[string]*C
|
||||
}
|
||||
|
||||
type C struct {
|
||||
D *D
|
||||
E *E
|
||||
}
|
||||
|
||||
type D struct {
|
||||
F *F
|
||||
G []G
|
||||
}
|
||||
|
||||
type E struct {
|
||||
H []H
|
||||
F *F
|
||||
}
|
||||
|
||||
type B struct{}
|
||||
type F struct{}
|
||||
type G struct{}
|
||||
type H struct{}
|
||||
|
||||
func main() {
|
||||
conf := &A{
|
||||
B: make(map[string]*B),
|
||||
C: make(map[string]*C),
|
||||
}
|
||||
fmt.Println(conf)
|
||||
}
|
||||
|
||||
// Output:
|
||||
// &{map[] map[]}
|
||||
@@ -2,12 +2,26 @@ package main
|
||||
|
||||
func main() {
|
||||
var a interface{}
|
||||
a = []int{3}
|
||||
switch a.(type) {
|
||||
case []int:
|
||||
println("a is []int")
|
||||
case []string:
|
||||
println("a is []string")
|
||||
}
|
||||
|
||||
var b interface{}
|
||||
b = []string{"hello"}
|
||||
switch b.(type) {
|
||||
case []int:
|
||||
println("b is []int")
|
||||
case []string:
|
||||
println("b is []string")
|
||||
}
|
||||
println("bye")
|
||||
}
|
||||
|
||||
// Output:
|
||||
// a is []int
|
||||
// b is []string
|
||||
// bye
|
||||
|
||||
11
_test/time16.go
Normal file
11
_test/time16.go
Normal file
@@ -0,0 +1,11 @@
|
||||
package main
|
||||
|
||||
import "time"
|
||||
|
||||
func main() {
|
||||
localTime := time.ANSIC
|
||||
println(localTime)
|
||||
}
|
||||
|
||||
// Output:
|
||||
// Mon Jan _2 15:04:05 2006
|
||||
@@ -12,11 +12,13 @@ type S struct {
|
||||
}
|
||||
|
||||
func main() {
|
||||
size := unsafe.Sizeof(S{})
|
||||
align := unsafe.Alignof(S{})
|
||||
x := S{}
|
||||
size := unsafe.Sizeof(x)
|
||||
align := unsafe.Alignof(x.Y)
|
||||
offset := unsafe.Offsetof(x.Z)
|
||||
|
||||
fmt.Println(size, align)
|
||||
fmt.Println(size, align, offset)
|
||||
}
|
||||
|
||||
// Output:
|
||||
// 24 8
|
||||
// 24 8 16
|
||||
|
||||
@@ -7,7 +7,7 @@ import (
|
||||
"fmt"
|
||||
"io"
|
||||
"os"
|
||||
"path"
|
||||
"path/filepath"
|
||||
"strings"
|
||||
|
||||
"github.com/traefik/yaegi/extract"
|
||||
@@ -50,7 +50,7 @@ func extractCmd(arg []string) error {
|
||||
}
|
||||
|
||||
if name == "" {
|
||||
name = path.Base(wd)
|
||||
name = filepath.Base(wd)
|
||||
}
|
||||
ext := extract.Extractor{
|
||||
Dest: name,
|
||||
@@ -102,7 +102,7 @@ func genLicense(fname string) (string, error) {
|
||||
|
||||
f, err := os.Open(fname)
|
||||
if err != nil {
|
||||
return "", fmt.Errorf("could not open LICENSE file: %v", err)
|
||||
return "", fmt.Errorf("could not open LICENSE file: %w", err)
|
||||
}
|
||||
defer func() { _ = f.Close() }()
|
||||
|
||||
@@ -116,7 +116,7 @@ func genLicense(fname string) (string, error) {
|
||||
license.WriteString("//" + txt + "\n")
|
||||
}
|
||||
if sc.Err() != nil {
|
||||
return "", fmt.Errorf("could not scan LICENSE file: %v", err)
|
||||
return "", fmt.Errorf("could not scan LICENSE file: %w", err)
|
||||
}
|
||||
|
||||
return license.String(), nil
|
||||
|
||||
@@ -6,6 +6,7 @@ import (
|
||||
"go/build"
|
||||
"io/ioutil"
|
||||
"os"
|
||||
"reflect"
|
||||
"strconv"
|
||||
"strings"
|
||||
|
||||
@@ -18,6 +19,7 @@ import (
|
||||
|
||||
func run(arg []string) error {
|
||||
var interactive bool
|
||||
var noAutoImport bool
|
||||
var tags string
|
||||
var cmd string
|
||||
var err error
|
||||
@@ -33,6 +35,7 @@ func run(arg []string) error {
|
||||
rflag.BoolVar(&useUnrestricted, "unrestricted", useUnrestricted, "include unrestricted symbols")
|
||||
rflag.StringVar(&tags, "tags", "", "set a list of build tags")
|
||||
rflag.BoolVar(&useUnsafe, "unsafe", useUnsafe, "include unsafe symbols")
|
||||
rflag.BoolVar(&noAutoImport, "noautoimport", false, "do not auto import pre-compiled packages. Import names that would result in collisions (e.g. rand from crypto/rand and rand from math/rand) are automatically renamed (crypto_rand and math_rand)")
|
||||
rflag.StringVar(&cmd, "e", "", "set the command to be executed (instead of script or/and shell)")
|
||||
rflag.Usage = func() {
|
||||
fmt.Println("Usage: yaegi run [options] [path] [args]")
|
||||
@@ -45,48 +48,68 @@ func run(arg []string) error {
|
||||
args := rflag.Args()
|
||||
|
||||
i := interp.New(interp.Options{GoPath: build.Default.GOPATH, BuildTags: strings.Split(tags, ",")})
|
||||
i.Use(stdlib.Symbols)
|
||||
i.Use(interp.Symbols)
|
||||
if err := i.Use(stdlib.Symbols); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := i.Use(interp.Symbols); err != nil {
|
||||
return err
|
||||
}
|
||||
if useSyscall {
|
||||
i.Use(syscall.Symbols)
|
||||
if err := i.Use(syscall.Symbols); err != nil {
|
||||
return err
|
||||
}
|
||||
// Using a environment var allows a nested interpreter to import the syscall package.
|
||||
if err := os.Setenv("YAEGI_SYSCALL", "1"); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
if useUnsafe {
|
||||
i.Use(unsafe.Symbols)
|
||||
if err := i.Use(unsafe.Symbols); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := os.Setenv("YAEGI_UNSAFE", "1"); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
if useUnrestricted {
|
||||
// Use of unrestricted symbols should always follow stdlib and syscall symbols, to update them.
|
||||
i.Use(unrestricted.Symbols)
|
||||
if err := i.Use(unrestricted.Symbols); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := os.Setenv("YAEGI_UNRESTRICTED", "1"); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
|
||||
if cmd != "" {
|
||||
_, err = i.Eval(cmd)
|
||||
if !noAutoImport {
|
||||
i.ImportUsed()
|
||||
}
|
||||
var v reflect.Value
|
||||
v, err = i.Eval(cmd)
|
||||
if len(args) == 0 && v.IsValid() {
|
||||
fmt.Println(v)
|
||||
}
|
||||
}
|
||||
|
||||
if len(args) == 0 {
|
||||
if interactive || cmd == "" {
|
||||
if cmd == "" || interactive {
|
||||
showError(err)
|
||||
if !noAutoImport {
|
||||
i.ImportUsed()
|
||||
}
|
||||
_, err = i.REPL()
|
||||
}
|
||||
return err
|
||||
}
|
||||
|
||||
// Skip first os arg to set command line as expected by interpreted main
|
||||
// Skip first os arg to set command line as expected by interpreted main.
|
||||
path := args[0]
|
||||
os.Args = arg
|
||||
flag.CommandLine = flag.NewFlagSet(path, flag.ExitOnError)
|
||||
|
||||
if isFile(path) {
|
||||
err = runFile(i, path)
|
||||
err = runFile(i, path, noAutoImport)
|
||||
} else {
|
||||
_, err = i.EvalPath(path)
|
||||
}
|
||||
@@ -106,7 +129,7 @@ func isFile(path string) bool {
|
||||
return err == nil && fi.Mode().IsRegular()
|
||||
}
|
||||
|
||||
func runFile(i *interp.Interpreter, path string) error {
|
||||
func runFile(i *interp.Interpreter, path string, noAutoImport bool) error {
|
||||
b, err := ioutil.ReadFile(path)
|
||||
if err != nil {
|
||||
return err
|
||||
@@ -115,6 +138,9 @@ func runFile(i *interp.Interpreter, path string) error {
|
||||
if s := string(b); strings.HasPrefix(s, "#!") {
|
||||
// Allow executable go scripts, Have the same behavior as in interactive mode.
|
||||
s = strings.Replace(s, "#!", "//", 1)
|
||||
if !noAutoImport {
|
||||
i.ImportUsed()
|
||||
}
|
||||
_, err = i.Eval(s)
|
||||
return err
|
||||
}
|
||||
|
||||
@@ -117,23 +117,33 @@ func test(arg []string) (err error) {
|
||||
}
|
||||
|
||||
i := interp.New(interp.Options{GoPath: build.Default.GOPATH, BuildTags: strings.Split(tags, ",")})
|
||||
i.Use(stdlib.Symbols)
|
||||
i.Use(interp.Symbols)
|
||||
if err := i.Use(stdlib.Symbols); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := i.Use(interp.Symbols); err != nil {
|
||||
return err
|
||||
}
|
||||
if useSyscall {
|
||||
i.Use(syscall.Symbols)
|
||||
if err := i.Use(syscall.Symbols); err != nil {
|
||||
return err
|
||||
}
|
||||
// Using a environment var allows a nested interpreter to import the syscall package.
|
||||
if err := os.Setenv("YAEGI_SYSCALL", "1"); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
if useUnrestricted {
|
||||
i.Use(unrestricted.Symbols)
|
||||
if err := i.Use(unrestricted.Symbols); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := os.Setenv("YAEGI_UNRESTRICTED", "1"); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
if useUnsafe {
|
||||
i.Use(unsafe.Symbols)
|
||||
if err := i.Use(unsafe.Symbols); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := os.Setenv("YAEGI_UNSAFE", "1"); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
@@ -50,9 +50,9 @@ func TestYaegiCmdCancel(t *testing.T) {
|
||||
|
||||
yaegi := filepath.Join(tmp, "yaegi")
|
||||
build := exec.Command("go", "build", "-race", "-o", yaegi, ".")
|
||||
err = build.Run()
|
||||
out, err := build.CombinedOutput()
|
||||
if err != nil {
|
||||
t.Fatalf("failed to build yaegi command: %v", err)
|
||||
t.Fatalf("failed to build yaegi command: %v: %s", err, out)
|
||||
}
|
||||
|
||||
// Test src must be terminated by a single newline.
|
||||
@@ -82,7 +82,7 @@ func TestYaegiCmdCancel(t *testing.T) {
|
||||
if err != nil {
|
||||
t.Errorf("failed pipe test source to yaegi command: %v", err)
|
||||
}
|
||||
Sleep(200 * time.Millisecond)
|
||||
Sleep(500 * time.Millisecond)
|
||||
err = cmd.Process.Signal(os.Interrupt)
|
||||
if err != nil {
|
||||
t.Errorf("failed to send os.Interrupt to yaegi command: %v", err)
|
||||
|
||||
@@ -9,7 +9,9 @@ import (
|
||||
|
||||
func TestFunctionCall(t *testing.T) {
|
||||
i := interp.New(interp.Options{GoPath: "./_pkg"})
|
||||
i.Use(stdlib.Symbols)
|
||||
if err := i.Use(stdlib.Symbols); err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
||||
_, err := i.Eval(`import "foo/bar"`)
|
||||
if err != nil {
|
||||
|
||||
@@ -10,7 +10,9 @@ import (
|
||||
|
||||
func TestGetFunc(t *testing.T) {
|
||||
i := interp.New(interp.Options{GoPath: "./_gopath/"})
|
||||
i.Use(stdlib.Symbols)
|
||||
if err := i.Use(stdlib.Symbols); err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
||||
if _, err := i.Eval(`import "github.com/foo/bar"`); err != nil {
|
||||
t.Fatal(err)
|
||||
|
||||
@@ -104,18 +104,21 @@ func TestPackages(t *testing.T) {
|
||||
for _, test := range testCases {
|
||||
test := test
|
||||
t.Run(test.desc, func(t *testing.T) {
|
||||
goPath, err := filepath.Abs(test.goPath)
|
||||
goPath, err := filepath.Abs(filepath.FromSlash(test.goPath))
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
||||
var stdout, stderr bytes.Buffer
|
||||
i := interp.New(interp.Options{GoPath: goPath, Stdout: &stdout, Stderr: &stderr})
|
||||
i.Use(stdlib.Symbols) // Use binary standard library
|
||||
// Use binary standard library
|
||||
if err := i.Use(stdlib.Symbols); err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
||||
var msg string
|
||||
if test.evalFile != "" {
|
||||
if _, err := i.EvalPath(test.evalFile); err != nil {
|
||||
if _, err := i.EvalPath(filepath.FromSlash(test.evalFile)); err != nil {
|
||||
fatalStderrf(t, "%v", err)
|
||||
}
|
||||
msg = stdout.String()
|
||||
@@ -146,6 +149,8 @@ func TestPackages(t *testing.T) {
|
||||
}
|
||||
|
||||
func fatalStderrf(t *testing.T, format string, args ...interface{}) {
|
||||
t.Helper()
|
||||
|
||||
fmt.Fprintf(os.Stderr, format+"\n", args...)
|
||||
t.FailNow()
|
||||
}
|
||||
@@ -159,7 +164,7 @@ func TestPackagesError(t *testing.T) {
|
||||
{
|
||||
desc: "different packages in the same directory",
|
||||
goPath: "./_pkg9/",
|
||||
expected: "1:21: import \"github.com/foo/pkg\" error: found packages pkg and pkgfalse in _pkg9/src/github.com/foo/pkg",
|
||||
expected: `1:21: import "github.com/foo/pkg" error: found packages pkg and pkgfalse in ` + filepath.FromSlash("_pkg9/src/github.com/foo/pkg"),
|
||||
},
|
||||
}
|
||||
|
||||
@@ -168,7 +173,10 @@ func TestPackagesError(t *testing.T) {
|
||||
t.Run(test.desc, func(t *testing.T) {
|
||||
// Init go interpreter
|
||||
i := interp.New(interp.Options{GoPath: test.goPath})
|
||||
i.Use(stdlib.Symbols) // Use binary standard library
|
||||
// Use binary standard library
|
||||
if err := i.Use(stdlib.Symbols); err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
||||
// Load pkg from sources
|
||||
_, err := i.Eval(`import "github.com/foo/pkg"`)
|
||||
|
||||
@@ -25,7 +25,7 @@ import (
|
||||
"text/template"
|
||||
)
|
||||
|
||||
const model = `// Code generated by 'yaegi extract {{.PkgName}}'. DO NOT EDIT.
|
||||
const model = `// Code generated by 'yaegi extract {{.ImportPath}}'. DO NOT EDIT.
|
||||
|
||||
{{.License}}
|
||||
|
||||
@@ -39,7 +39,7 @@ import (
|
||||
"{{$key}}"
|
||||
{{- end}}
|
||||
{{- end}}
|
||||
"{{.PkgName}}"
|
||||
"{{.ImportPath}}"
|
||||
"reflect"
|
||||
)
|
||||
|
||||
@@ -74,6 +74,7 @@ func init() {
|
||||
{{range $key, $value := .Wrap -}}
|
||||
// {{$value.Name}} is an interface wrapper for {{$key}} type
|
||||
type {{$value.Name}} struct {
|
||||
IValue interface{}
|
||||
{{range $m := $value.Method -}}
|
||||
W{{$m.Name}} func{{$m.Param}} {{$m.Result}}
|
||||
{{end}}
|
||||
@@ -167,8 +168,8 @@ func (e *Extractor) genContent(importPath string, p *types.Package) ([]byte, err
|
||||
continue
|
||||
}
|
||||
|
||||
pname := path.Base(importPath) + "." + name
|
||||
if rname := path.Base(importPath) + name; restricted[rname] {
|
||||
pname := p.Name() + "." + name
|
||||
if rname := p.Name() + name; restricted[rname] {
|
||||
// Restricted symbol, locally provided by stdlib wrapper.
|
||||
pname = rname
|
||||
}
|
||||
@@ -203,7 +204,15 @@ func (e *Extractor) genContent(importPath string, p *types.Package) ([]byte, err
|
||||
if args[j] = v.Name(); args[j] == "" {
|
||||
args[j] = fmt.Sprintf("a%d", j)
|
||||
}
|
||||
params[j] = args[j] + " " + types.TypeString(v.Type(), qualify)
|
||||
// process interface method variadic parameter
|
||||
if sign.Variadic() && j == len(args)-1 { // check is last arg
|
||||
// only replace the first "[]" to "..."
|
||||
at := types.TypeString(v.Type(), qualify)[2:]
|
||||
params[j] = args[j] + " ..." + at
|
||||
args[j] += "..."
|
||||
} else {
|
||||
params[j] = args[j] + " " + types.TypeString(v.Type(), qualify)
|
||||
}
|
||||
}
|
||||
arg := "(" + strings.Join(args, ", ") + ")"
|
||||
param := "(" + strings.Join(params, ", ") + ")"
|
||||
@@ -241,7 +250,7 @@ func (e *Extractor) genContent(importPath string, p *types.Package) ([]byte, err
|
||||
base := template.New("extract")
|
||||
parse, err := base.Parse(model)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("template parsing error: %v", err)
|
||||
return nil, fmt.Errorf("template parsing error: %w", err)
|
||||
}
|
||||
|
||||
if importPath == "log/syslog" {
|
||||
@@ -263,24 +272,25 @@ func (e *Extractor) genContent(importPath string, p *types.Package) ([]byte, err
|
||||
|
||||
b := new(bytes.Buffer)
|
||||
data := map[string]interface{}{
|
||||
"Dest": e.Dest,
|
||||
"Imports": imports,
|
||||
"PkgName": importPath,
|
||||
"Val": val,
|
||||
"Typ": typ,
|
||||
"Wrap": wrap,
|
||||
"BuildTags": buildTags,
|
||||
"License": e.License,
|
||||
"Dest": e.Dest,
|
||||
"Imports": imports,
|
||||
"ImportPath": importPath,
|
||||
"PkgName": path.Join(importPath, p.Name()),
|
||||
"Val": val,
|
||||
"Typ": typ,
|
||||
"Wrap": wrap,
|
||||
"BuildTags": buildTags,
|
||||
"License": e.License,
|
||||
}
|
||||
err = parse.Execute(b, data)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("template error: %v", err)
|
||||
return nil, fmt.Errorf("template error: %w", err)
|
||||
}
|
||||
|
||||
// gofmt
|
||||
source, err := format.Source(b.Bytes())
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("failed to format source: %v: %s", err, b.Bytes())
|
||||
return nil, fmt.Errorf("failed to format source: %w: %s", err, b.Bytes())
|
||||
}
|
||||
return source, nil
|
||||
}
|
||||
@@ -436,7 +446,7 @@ func GetMinor(part string) string {
|
||||
return minor
|
||||
}
|
||||
|
||||
const defaultMinorVersion = 15
|
||||
const defaultMinorVersion = 16
|
||||
|
||||
func genBuildTags() (string, error) {
|
||||
version := runtime.Version()
|
||||
@@ -451,7 +461,7 @@ func genBuildTags() (string, error) {
|
||||
|
||||
minor, err := strconv.Atoi(minorRaw)
|
||||
if err != nil {
|
||||
return "", fmt.Errorf("failed to parse version: %v", err)
|
||||
return "", fmt.Errorf("failed to parse version: %w", err)
|
||||
}
|
||||
|
||||
// Only append an upper bound if we are not on the latest go
|
||||
|
||||
@@ -18,7 +18,7 @@ import (
|
||||
)
|
||||
|
||||
func init() {
|
||||
Symbols["guthib.com/baz"] = map[string]reflect.Value{
|
||||
Symbols["guthib.com/baz/baz"] = map[string]reflect.Value{
|
||||
// function, constant and variable definitions
|
||||
"Hello": reflect.ValueOf(baz.Hello),
|
||||
}
|
||||
@@ -78,6 +78,49 @@ func TestPackages(t *testing.T) {
|
||||
importPath: "guthib.com/baz",
|
||||
expected: expectedOutput,
|
||||
},
|
||||
{
|
||||
desc: "using relative path, package name is not same as import path",
|
||||
wd: "./testdata/6/src/guthib.com/bar",
|
||||
arg: "../baz-baz",
|
||||
importPath: "guthib.com/baz",
|
||||
expected: expectedOutput,
|
||||
},
|
||||
{
|
||||
desc: "using relative path, interface method parameter is variadic",
|
||||
wd: "./testdata/7/src/guthib.com/variadic",
|
||||
arg: "../variadic",
|
||||
importPath: "guthib.com/variadic",
|
||||
expected: `
|
||||
// Code generated by 'yaegi extract guthib.com/variadic'. DO NOT EDIT.
|
||||
|
||||
package variadic
|
||||
|
||||
import (
|
||||
"guthib.com/variadic"
|
||||
"reflect"
|
||||
)
|
||||
|
||||
func init() {
|
||||
Symbols["guthib.com/variadic/variadic"] = map[string]reflect.Value{
|
||||
// type definitions
|
||||
"Variadic": reflect.ValueOf((*variadic.Variadic)(nil)),
|
||||
|
||||
// interface wrapper definitions
|
||||
"_Variadic": reflect.ValueOf((*_guthib_com_variadic_Variadic)(nil)),
|
||||
}
|
||||
}
|
||||
|
||||
// _guthib_com_variadic_Variadic is an interface wrapper for Variadic type
|
||||
type _guthib_com_variadic_Variadic struct {
|
||||
IValue interface{}
|
||||
WCall func(method string, args ...[]interface{}) (interface{}, error)
|
||||
}
|
||||
|
||||
func (W _guthib_com_variadic_Variadic) Call(method string, args ...[]interface{}) (interface{}, error) {
|
||||
return W.WCall(method, args...)
|
||||
}
|
||||
`[1:],
|
||||
},
|
||||
}
|
||||
|
||||
for _, test := range testCases {
|
||||
|
||||
9
extract/testdata/6/src/guthib.com/bar/main.go
vendored
Normal file
9
extract/testdata/6/src/guthib.com/bar/main.go
vendored
Normal file
@@ -0,0 +1,9 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"guthib.com/baz"
|
||||
)
|
||||
|
||||
func main() {
|
||||
baz.Hello()
|
||||
}
|
||||
5
extract/testdata/6/src/guthib.com/baz-baz/baz.go
vendored
Normal file
5
extract/testdata/6/src/guthib.com/baz-baz/baz.go
vendored
Normal file
@@ -0,0 +1,5 @@
|
||||
package baz
|
||||
|
||||
func Hello() {
|
||||
println("HELLO")
|
||||
}
|
||||
1
extract/testdata/6/src/guthib.com/baz-baz/go.mod
vendored
Normal file
1
extract/testdata/6/src/guthib.com/baz-baz/go.mod
vendored
Normal file
@@ -0,0 +1 @@
|
||||
module guthib.com/baz-baz
|
||||
1
extract/testdata/7/src/guthib.com/variadic/go.mod
vendored
Normal file
1
extract/testdata/7/src/guthib.com/variadic/go.mod
vendored
Normal file
@@ -0,0 +1 @@
|
||||
module guthib.com/baz-baz/variadic
|
||||
5
extract/testdata/7/src/guthib.com/variadic/variadic.go
vendored
Normal file
5
extract/testdata/7/src/guthib.com/variadic/variadic.go
vendored
Normal file
@@ -0,0 +1,5 @@
|
||||
package variadic
|
||||
|
||||
type Variadic interface {
|
||||
Call(method string, args ...[]interface{}) (interface{}, error)
|
||||
}
|
||||
@@ -11,7 +11,7 @@ import (
|
||||
|
||||
const model = `package interp
|
||||
|
||||
// Code generated by 'go run ../internal/genop/genop.go'. DO NOT EDIT.
|
||||
// Code generated by 'go run ../internal/cmd/genop/genop.go'. DO NOT EDIT.
|
||||
|
||||
import (
|
||||
"go/constant"
|
||||
@@ -24,6 +24,7 @@ import (
|
||||
func {{$name}}(n *node) {
|
||||
next := getExec(n.tnext)
|
||||
typ := n.typ.concrete().TypeOf()
|
||||
isInterface := n.typ.TypeOf().Kind() == reflect.Interface
|
||||
dest := genValueOutput(n, typ)
|
||||
c0, c1 := n.child[0], n.child[1]
|
||||
|
||||
@@ -31,6 +32,13 @@ func {{$name}}(n *node) {
|
||||
{{- if $op.Str}}
|
||||
case reflect.String:
|
||||
switch {
|
||||
case isInterface:
|
||||
v0 := genValue(c0)
|
||||
v1 := genValue(c1)
|
||||
n.exec = func(f *frame) bltn {
|
||||
dest(f).Set(reflect.ValueOf(v0(f).String() {{$op.Name}} v1(f).String()).Convert(typ))
|
||||
return next
|
||||
}
|
||||
case c0.rval.IsValid():
|
||||
s0 := vString(c0.rval)
|
||||
v1 := genValue(c1)
|
||||
@@ -56,6 +64,19 @@ func {{$name}}(n *node) {
|
||||
{{- end}}
|
||||
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
|
||||
switch {
|
||||
case isInterface:
|
||||
v0 := genValueInt(c0)
|
||||
{{- if $op.Shift}}
|
||||
v1 := genValueUint(c1)
|
||||
{{else}}
|
||||
v1 := genValueInt(c1)
|
||||
{{end -}}
|
||||
n.exec = func(f *frame) bltn {
|
||||
_, i := v0(f)
|
||||
_, j := v1(f)
|
||||
dest(f).Set(reflect.ValueOf(i {{$op.Name}} j).Convert(typ))
|
||||
return next
|
||||
}
|
||||
case c0.rval.IsValid():
|
||||
i := vInt(c0.rval)
|
||||
{{- if $op.Shift}}
|
||||
@@ -96,6 +117,15 @@ func {{$name}}(n *node) {
|
||||
}
|
||||
case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr:
|
||||
switch {
|
||||
case isInterface:
|
||||
v0 := genValueUint(c0)
|
||||
v1 := genValueUint(c1)
|
||||
n.exec = func(f *frame) bltn {
|
||||
_, i := v0(f)
|
||||
_, j := v1(f)
|
||||
dest(f).Set(reflect.ValueOf(i {{$op.Name}} j).Convert(typ))
|
||||
return next
|
||||
}
|
||||
case c0.rval.IsValid():
|
||||
i := vUint(c0.rval)
|
||||
v1 := genValueUint(c1)
|
||||
@@ -125,6 +155,15 @@ func {{$name}}(n *node) {
|
||||
{{- if $op.Float}}
|
||||
case reflect.Float32, reflect.Float64:
|
||||
switch {
|
||||
case isInterface:
|
||||
v0 := genValueFloat(c0)
|
||||
v1 := genValueFloat(c1)
|
||||
n.exec = func(f *frame) bltn {
|
||||
_, i := v0(f)
|
||||
_, j := v1(f)
|
||||
dest(f).Set(reflect.ValueOf(i {{$op.Name}} j).Convert(typ))
|
||||
return next
|
||||
}
|
||||
case c0.rval.IsValid():
|
||||
i := vFloat(c0.rval)
|
||||
v1 := genValueFloat(c1)
|
||||
@@ -153,25 +192,32 @@ func {{$name}}(n *node) {
|
||||
}
|
||||
case reflect.Complex64, reflect.Complex128:
|
||||
switch {
|
||||
case isInterface:
|
||||
v0 := genComplex(c0)
|
||||
v1 := genComplex(c1)
|
||||
n.exec = func(f *frame) bltn {
|
||||
dest(f).Set(reflect.ValueOf(v0(f) {{$op.Name}} v1(f)).Convert(typ))
|
||||
return next
|
||||
}
|
||||
case c0.rval.IsValid():
|
||||
r0 := vComplex(c0.rval)
|
||||
v1 := genValue(c1)
|
||||
v1 := genComplex(c1)
|
||||
n.exec = func(f *frame) bltn {
|
||||
dest(f).SetComplex(r0 {{$op.Name}} v1(f).Complex())
|
||||
dest(f).SetComplex(r0 {{$op.Name}} v1(f))
|
||||
return next
|
||||
}
|
||||
case c1.rval.IsValid():
|
||||
r1 := vComplex(c1.rval)
|
||||
v0 := genValue(c0)
|
||||
v0 := genComplex(c0)
|
||||
n.exec = func(f *frame) bltn {
|
||||
dest(f).SetComplex(v0(f).Complex() {{$op.Name}} r1)
|
||||
dest(f).SetComplex(v0(f) {{$op.Name}} r1)
|
||||
return next
|
||||
}
|
||||
default:
|
||||
v0 := genValue(c0)
|
||||
v1 := genValue(c1)
|
||||
v0 := genComplex(c0)
|
||||
v1 := genComplex(c1)
|
||||
n.exec = func(f *frame) bltn {
|
||||
dest(f).SetComplex(v0(f).Complex() {{$op.Name}} v1(f).Complex())
|
||||
dest(f).SetComplex(v0(f) {{$op.Name}} v1(f))
|
||||
return next
|
||||
}
|
||||
}
|
||||
@@ -209,7 +255,11 @@ func {{$name}}Const(n *node) {
|
||||
v := constant.BinaryOp(vConstantValue(v0), operator, vConstantValue(v1))
|
||||
n.rval.Set(reflect.ValueOf(v))
|
||||
{{- else}}
|
||||
{{- if $op.Int}}
|
||||
v := constant.BinaryOp(constant.ToInt(vConstantValue(v0)), token.{{tokenFromName $name}}, constant.ToInt(vConstantValue(v1)))
|
||||
{{- else}}
|
||||
v := constant.BinaryOp(vConstantValue(v0), token.{{tokenFromName $name}}, vConstantValue(v1))
|
||||
{{- end}}
|
||||
n.rval.Set(reflect.ValueOf(v))
|
||||
{{- end}}
|
||||
{{- if $op.Str}}
|
||||
@@ -425,12 +475,24 @@ func {{$name}}Const(n *node) {
|
||||
func {{$name}}(n *node) {
|
||||
tnext := getExec(n.tnext)
|
||||
dest := genValueOutput(n, reflect.TypeOf(true))
|
||||
typ := n.typ.concrete().TypeOf()
|
||||
isInterface := n.typ.TypeOf().Kind() == reflect.Interface
|
||||
c0, c1 := n.child[0], n.child[1]
|
||||
|
||||
{{- if or (eq $op.Name "==") (eq $op.Name "!=") }}
|
||||
|
||||
if c0.typ.cat == aliasT || c1.typ.cat == aliasT {
|
||||
switch {
|
||||
case isInterface:
|
||||
v0 := genValue(c0)
|
||||
v1 := genValue(c1)
|
||||
dest := genValue(n)
|
||||
n.exec = func(f *frame) bltn {
|
||||
i0 := v0(f).Interface()
|
||||
i1 := v1(f).Interface()
|
||||
dest(f).Set(reflect.ValueOf(i0 {{$op.Name}} i1).Convert(typ))
|
||||
return tnext
|
||||
}
|
||||
case c0.rval.IsValid():
|
||||
i0 := c0.rval.Interface()
|
||||
v1 := genValue(c1)
|
||||
@@ -507,9 +569,18 @@ func {{$name}}(n *node) {
|
||||
switch t0, t1 := c0.typ.TypeOf(), c1.typ.TypeOf(); {
|
||||
case isString(t0) || isString(t1):
|
||||
switch {
|
||||
case isInterface:
|
||||
v0 := genValueString(c0)
|
||||
v1 := genValueString(c1)
|
||||
n.exec = func(f *frame) bltn {
|
||||
_, s0 := v0(f)
|
||||
_, s1 := v1(f)
|
||||
dest(f).Set(reflect.ValueOf(s0 {{$op.Name}} s1).Convert(typ))
|
||||
return tnext
|
||||
}
|
||||
case c0.rval.IsValid():
|
||||
s0 := vString(c0.rval)
|
||||
v1 := genValueString(n.child[1])
|
||||
v1 := genValueString(c1)
|
||||
if n.fnext != nil {
|
||||
fnext := getExec(n.fnext)
|
||||
n.exec = func(f *frame) bltn {
|
||||
@@ -530,7 +601,7 @@ func {{$name}}(n *node) {
|
||||
}
|
||||
case c1.rval.IsValid():
|
||||
s1 := vString(c1.rval)
|
||||
v0 := genValueString(n.child[0])
|
||||
v0 := genValueString(c0)
|
||||
if n.fnext != nil {
|
||||
fnext := getExec(n.fnext)
|
||||
n.exec = func(f *frame) bltn {
|
||||
@@ -550,8 +621,8 @@ func {{$name}}(n *node) {
|
||||
}
|
||||
}
|
||||
default:
|
||||
v0 := genValueString(n.child[0])
|
||||
v1 := genValueString(n.child[1])
|
||||
v0 := genValueString(c0)
|
||||
v1 := genValueString(c1)
|
||||
if n.fnext != nil {
|
||||
fnext := getExec(n.fnext)
|
||||
n.exec = func(f *frame) bltn {
|
||||
@@ -575,6 +646,15 @@ func {{$name}}(n *node) {
|
||||
}
|
||||
case isFloat(t0) || isFloat(t1):
|
||||
switch {
|
||||
case isInterface:
|
||||
v0 := genValueFloat(c0)
|
||||
v1 := genValueFloat(c1)
|
||||
n.exec = func(f *frame) bltn {
|
||||
_, s0 := v0(f)
|
||||
_, s1 := v1(f)
|
||||
dest(f).Set(reflect.ValueOf(s0 {{$op.Name}} s1).Convert(typ))
|
||||
return tnext
|
||||
}
|
||||
case c0.rval.IsValid():
|
||||
s0 := vFloat(c0.rval)
|
||||
v1 := genValueFloat(c1)
|
||||
@@ -645,6 +725,15 @@ func {{$name}}(n *node) {
|
||||
}
|
||||
case isUint(t0) || isUint(t1):
|
||||
switch {
|
||||
case isInterface:
|
||||
v0 := genValueUint(c0)
|
||||
v1 := genValueUint(c1)
|
||||
n.exec = func(f *frame) bltn {
|
||||
_, s0 := v0(f)
|
||||
_, s1 := v1(f)
|
||||
dest(f).Set(reflect.ValueOf(s0 {{$op.Name}} s1).Convert(typ))
|
||||
return tnext
|
||||
}
|
||||
case c0.rval.IsValid():
|
||||
s0 := vUint(c0.rval)
|
||||
v1 := genValueUint(c1)
|
||||
@@ -716,6 +805,15 @@ func {{$name}}(n *node) {
|
||||
}
|
||||
case isInt(t0) || isInt(t1):
|
||||
switch {
|
||||
case isInterface:
|
||||
v0 := genValueInt(c0)
|
||||
v1 := genValueInt(c1)
|
||||
n.exec = func(f *frame) bltn {
|
||||
_, s0 := v0(f)
|
||||
_, s1 := v1(f)
|
||||
dest(f).Set(reflect.ValueOf(s0 {{$op.Name}} s1).Convert(typ))
|
||||
return tnext
|
||||
}
|
||||
case c0.rval.IsValid():
|
||||
s0 := vInt(c0.rval)
|
||||
v1 := genValueInt(c1)
|
||||
@@ -788,6 +886,15 @@ func {{$name}}(n *node) {
|
||||
{{- if $op.Complex}}
|
||||
case isComplex(t0) || isComplex(t1):
|
||||
switch {
|
||||
case isInterface:
|
||||
v0 := genComplex(c0)
|
||||
v1 := genComplex(c1)
|
||||
n.exec = func(f *frame) bltn {
|
||||
s0 := v0(f)
|
||||
s1 := v1(f)
|
||||
dest(f).Set(reflect.ValueOf(s0 {{$op.Name}} s1).Convert(typ))
|
||||
return tnext
|
||||
}
|
||||
case c0.rval.IsValid():
|
||||
s0 := vComplex(c0.rval)
|
||||
v1 := genComplex(c1)
|
||||
@@ -832,8 +939,8 @@ func {{$name}}(n *node) {
|
||||
}
|
||||
}
|
||||
default:
|
||||
v0 := genComplex(n.child[0])
|
||||
v1 := genComplex(n.child[1])
|
||||
v0 := genComplex(c0)
|
||||
v1 := genComplex(c1)
|
||||
if n.fnext != nil {
|
||||
fnext := getExec(n.fnext)
|
||||
n.exec = func(f *frame) bltn {
|
||||
@@ -857,6 +964,15 @@ func {{$name}}(n *node) {
|
||||
}
|
||||
default:
|
||||
switch {
|
||||
case isInterface:
|
||||
v0 := genValue(c0)
|
||||
v1 := genValue(c1)
|
||||
n.exec = func(f *frame) bltn {
|
||||
i0 := v0(f).Interface()
|
||||
i1 := v1(f).Interface()
|
||||
dest(f).Set(reflect.ValueOf(i0 {{$op.Name}} i1).Convert(typ))
|
||||
return tnext
|
||||
}
|
||||
case c0.rval.IsValid():
|
||||
i0 := c0.rval.Interface()
|
||||
v1 := genValue(c1)
|
||||
@@ -940,6 +1056,7 @@ type Op struct {
|
||||
Complex bool // true if operator applies to complex
|
||||
Shift bool // true if operator is a shift operation
|
||||
Bool bool // true if operator applies to bool
|
||||
Int bool // true if operator applies to int only
|
||||
}
|
||||
|
||||
func main() {
|
||||
@@ -968,17 +1085,17 @@ func main() {
|
||||
b := &bytes.Buffer{}
|
||||
data := map[string]interface{}{
|
||||
"Arithmetic": map[string]Op{
|
||||
"add": {"+", true, true, true, false, false},
|
||||
"sub": {"-", false, true, true, false, false},
|
||||
"mul": {"*", false, true, true, false, false},
|
||||
"quo": {"/", false, true, true, false, false},
|
||||
"rem": {"%", false, false, false, false, false},
|
||||
"shl": {"<<", false, false, false, true, false},
|
||||
"shr": {">>", false, false, false, true, false},
|
||||
"and": {"&", false, false, false, false, false},
|
||||
"or": {"|", false, false, false, false, false},
|
||||
"xor": {"^", false, false, false, false, false},
|
||||
"andNot": {"&^", false, false, false, false, false},
|
||||
"add": {"+", true, true, true, false, false, false},
|
||||
"sub": {"-", false, true, true, false, false, false},
|
||||
"mul": {"*", false, true, true, false, false, false},
|
||||
"quo": {"/", false, true, true, false, false, false},
|
||||
"rem": {"%", false, false, false, false, false, true},
|
||||
"shl": {"<<", false, false, false, true, false, true},
|
||||
"shr": {">>", false, false, false, true, false, true},
|
||||
"and": {"&", false, false, false, false, false, true},
|
||||
"or": {"|", false, false, false, false, false, true},
|
||||
"xor": {"^", false, false, false, false, false, true},
|
||||
"andNot": {"&^", false, false, false, false, false, true},
|
||||
},
|
||||
"IncDec": map[string]Op{
|
||||
"inc": {Name: "+"},
|
||||
@@ -996,7 +1113,7 @@ func main() {
|
||||
"not": {Name: "!", Float: false, Bool: true},
|
||||
"neg": {Name: "-", Float: true, Bool: false},
|
||||
"pos": {Name: "+", Float: true, Bool: false},
|
||||
"bitNot": {Name: "^", Float: false, Bool: false},
|
||||
"bitNot": {Name: "^", Float: false, Bool: false, Int: true},
|
||||
},
|
||||
}
|
||||
if err = parse.Execute(b, data); err != nil {
|
||||
|
||||
@@ -357,7 +357,7 @@ func ignoreError(err error, src string) bool {
|
||||
}
|
||||
|
||||
func wrapInMain(src string) string {
|
||||
return fmt.Sprintf("package main; func main() {%s}", src)
|
||||
return fmt.Sprintf("package main; func main() {%s\n}", src)
|
||||
}
|
||||
|
||||
// Note: no type analysis is performed at this stage, it is done in pre-order
|
||||
@@ -716,7 +716,7 @@ func (interp *Interpreter) ast(src, name string, inc bool) (string, *node, error
|
||||
n := addChild(&root, anc, pos, identExpr, aNop)
|
||||
n.ident = a.Name
|
||||
st.push(n, nod)
|
||||
if n.anc.kind == defineStmt && n.anc.nright == 0 {
|
||||
if n.anc.kind == defineStmt && n.anc.anc.kind == constDecl && n.anc.nright == 0 {
|
||||
// Implicit assign expression (in a ConstDecl block).
|
||||
// Clone assign source and type from previous
|
||||
a := n.anc
|
||||
@@ -858,7 +858,8 @@ func (interp *Interpreter) ast(src, name string, inc bool) (string, *node, error
|
||||
case *ast.ValueSpec:
|
||||
kind := valueSpec
|
||||
act := aNop
|
||||
if a.Values != nil {
|
||||
switch {
|
||||
case a.Values != nil:
|
||||
if len(a.Names) > 1 && len(a.Values) == 1 {
|
||||
if anc.node.kind == constDecl || anc.node.kind == varDecl {
|
||||
kind = defineXStmt
|
||||
@@ -874,7 +875,9 @@ func (interp *Interpreter) ast(src, name string, inc bool) (string, *node, error
|
||||
}
|
||||
act = aAssign
|
||||
}
|
||||
} else if anc.node.kind == constDecl {
|
||||
case anc.node.kind == constDecl:
|
||||
kind, act = defineStmt, aAssign
|
||||
case anc.node.kind == varDecl && anc.node.anc.kind != fileStmt:
|
||||
kind, act = defineStmt, aAssign
|
||||
}
|
||||
n := addChild(&root, anc, pos, kind, act)
|
||||
|
||||
@@ -163,9 +163,9 @@ var knownOs = map[string]bool{
|
||||
"dragonfly": true,
|
||||
"freebsd": true,
|
||||
"illumos": true,
|
||||
"ios": true,
|
||||
"js": true,
|
||||
"linux": true,
|
||||
"nacl": true,
|
||||
"netbsd": true,
|
||||
"openbsd": true,
|
||||
"plan9": true,
|
||||
@@ -176,7 +176,6 @@ var knownOs = map[string]bool{
|
||||
var knownArch = map[string]bool{
|
||||
"386": true,
|
||||
"amd64": true,
|
||||
"amd64p32": true,
|
||||
"arm": true,
|
||||
"arm64": true,
|
||||
"mips": true,
|
||||
|
||||
521
interp/cfg.go
521
interp/cfg.go
@@ -7,7 +7,6 @@ import (
|
||||
"math"
|
||||
"path/filepath"
|
||||
"reflect"
|
||||
"regexp"
|
||||
"strings"
|
||||
"unicode"
|
||||
)
|
||||
@@ -44,8 +43,6 @@ var constBltn = map[string]func(*node){
|
||||
bltnReal: realConst,
|
||||
}
|
||||
|
||||
var identifier = regexp.MustCompile(`([\pL_][\pL_\d]*)$`)
|
||||
|
||||
const nilIdent = "nil"
|
||||
|
||||
// cfg generates a control flow graph (CFG) from AST (wiring successors in AST)
|
||||
@@ -82,7 +79,14 @@ func (interp *Interpreter) cfg(root *node, importPath string) ([]*node, error) {
|
||||
i--
|
||||
}
|
||||
dest := a.child[i]
|
||||
if dest.typ != nil && !isInterface(dest.typ) {
|
||||
if dest.typ == nil {
|
||||
break
|
||||
}
|
||||
if dest.typ.incomplete {
|
||||
err = n.cfgErrorf("invalid type declaration")
|
||||
return false
|
||||
}
|
||||
if !isInterface(dest.typ) {
|
||||
// Interface type are not propagated, and will be resolved at post-order.
|
||||
n.typ = dest.typ
|
||||
}
|
||||
@@ -147,6 +151,7 @@ func (interp *Interpreter) cfg(root *node, importPath string) ([]*node, error) {
|
||||
vtyp = &itype{cat: valueT, rtype: typ.Elem()}
|
||||
case reflect.String:
|
||||
sc.add(sc.getType("int")) // Add a dummy type to store array shallow copy for range
|
||||
sc.add(sc.getType("int")) // Add a dummy type to store index for range
|
||||
ktyp = sc.getType("int")
|
||||
vtyp = sc.getType("rune")
|
||||
case reflect.Array, reflect.Slice:
|
||||
@@ -170,9 +175,10 @@ func (interp *Interpreter) cfg(root *node, importPath string) ([]*node, error) {
|
||||
}
|
||||
case stringT:
|
||||
sc.add(sc.getType("int")) // Add a dummy type to store array shallow copy for range
|
||||
sc.add(sc.getType("int")) // Add a dummy type to store index for range
|
||||
ktyp = sc.getType("int")
|
||||
vtyp = sc.getType("rune")
|
||||
case arrayT, variadicT:
|
||||
case arrayT, sliceT, variadicT:
|
||||
sc.add(sc.getType("int")) // Add a dummy type to store array shallow copy for range
|
||||
ktyp = sc.getType("int")
|
||||
vtyp = o.typ.val
|
||||
@@ -518,10 +524,6 @@ func (interp *Interpreter) cfg(root *node, importPath string) ([]*node, error) {
|
||||
if dest.typ.incomplete {
|
||||
return
|
||||
}
|
||||
if dest.typ.sizedef {
|
||||
dest.typ.size = arrayTypeLen(src)
|
||||
dest.typ.rtype = nil
|
||||
}
|
||||
if sc.global {
|
||||
// Do not overload existing symbols (defined in GTA) in global scope
|
||||
sym, _, _ = sc.lookup(dest.ident)
|
||||
@@ -555,60 +557,71 @@ func (interp *Interpreter) cfg(root *node, importPath string) ([]*node, error) {
|
||||
n.findex = dest.findex
|
||||
n.level = dest.level
|
||||
|
||||
// Propagate type
|
||||
// TODO: Check that existing destination type matches source type
|
||||
// In the following, we attempt to optimize by skipping the assign
|
||||
// operation and setting the source location directly to the destination
|
||||
// location in the frame.
|
||||
//
|
||||
switch {
|
||||
case n.action == aAssign && isCall(src) && dest.typ.cat != interfaceT && !isMapEntry(dest) && !isRecursiveField(dest):
|
||||
case n.action != aAssign:
|
||||
// Do not skip assign operation if it is combined with another operator.
|
||||
case src.rval.IsValid():
|
||||
// Do not skip assign operation if setting from a constant value.
|
||||
case isMapEntry(dest):
|
||||
// Setting a map entry requires an additional step, do not optimize.
|
||||
// As we only write, skip the default useless getIndexMap dest action.
|
||||
dest.gen = nop
|
||||
case isFuncField(dest):
|
||||
// Setting a struct field of function type requires an extra step. Do not optimize.
|
||||
case isCall(src) && !isInterfaceSrc(dest.typ) && !isRecursiveField(dest) && n.kind != defineStmt:
|
||||
// Call action may perform the assignment directly.
|
||||
if dest.typ.id() != src.typ.id() {
|
||||
// Skip optimitization if returned type doesn't match assigned one.
|
||||
break
|
||||
}
|
||||
n.gen = nop
|
||||
src.level = level
|
||||
src.findex = dest.findex
|
||||
if src.typ.untyped && !dest.typ.untyped {
|
||||
src.typ = dest.typ
|
||||
}
|
||||
case n.action == aAssign && src.action == aRecv:
|
||||
case src.action == aRecv:
|
||||
// Assign by reading from a receiving channel.
|
||||
n.gen = nop
|
||||
src.findex = dest.findex // Set recv address to LHS
|
||||
src.findex = dest.findex // Set recv address to LHS.
|
||||
dest.typ = src.typ
|
||||
case n.action == aAssign && src.action == aCompositeLit && !isMapEntry(dest):
|
||||
case src.action == aCompositeLit:
|
||||
if dest.typ.cat == valueT && dest.typ.rtype.Kind() == reflect.Interface {
|
||||
// Skip optimisation for assigned binary interface or map entry
|
||||
// which require and additional operation to set the value
|
||||
// Skip optimisation for assigned interface.
|
||||
break
|
||||
}
|
||||
if dest.action == aGetIndex {
|
||||
// optimization does not work when assigning to a struct field. Maybe we're not
|
||||
// setting the right frame index or something, and we would end up not writing at
|
||||
// the right place. So disabling it for now.
|
||||
// Skip optimization, as it does not work when assigning to a struct field.
|
||||
break
|
||||
}
|
||||
// Skip the assign operation entirely, the source frame index is set
|
||||
// to destination index, avoiding extra memory alloc and duplication.
|
||||
n.gen = nop
|
||||
src.findex = dest.findex
|
||||
src.level = level
|
||||
case n.action == aAssign && len(n.child) < 4 && !src.rval.IsValid() && isArithmeticAction(src):
|
||||
case len(n.child) < 4 && isArithmeticAction(src):
|
||||
// Optimize single assignments from some arithmetic operations.
|
||||
// Skip the assign operation entirely, the source frame index is set
|
||||
// to destination index, avoiding extra memory alloc and duplication.
|
||||
src.typ = dest.typ
|
||||
src.findex = dest.findex
|
||||
src.level = level
|
||||
n.gen = nop
|
||||
case src.kind == basicLit && !src.rval.IsValid():
|
||||
case src.kind == basicLit:
|
||||
// Assign to nil.
|
||||
src.rval = reflect.New(dest.typ.TypeOf()).Elem()
|
||||
case n.nright == 0:
|
||||
n.gen = reset
|
||||
}
|
||||
|
||||
n.typ = dest.typ
|
||||
if sym != nil {
|
||||
sym.typ = n.typ
|
||||
sym.recv = src.recv
|
||||
}
|
||||
|
||||
n.level = level
|
||||
if isMapEntry(dest) {
|
||||
dest.gen = nop // skip getIndexMap
|
||||
}
|
||||
|
||||
if n.anc.kind == constDecl {
|
||||
n.gen = nop
|
||||
n.findex = notInFrame
|
||||
@@ -708,7 +721,7 @@ func (interp *Interpreter) cfg(root *node, importPath string) ([]*node, error) {
|
||||
break
|
||||
}
|
||||
}
|
||||
if c0.rval.IsValid() && c1.rval.IsValid() && !isInterface(n.typ) && constOp[n.action] != nil {
|
||||
if c0.rval.IsValid() && c1.rval.IsValid() && (!isInterface(n.typ)) && constOp[n.action] != nil {
|
||||
n.typ.TypeOf() // Force compute of reflection type.
|
||||
constOp[n.action](n) // Compute a constant result now rather than during exec.
|
||||
}
|
||||
@@ -718,7 +731,7 @@ func (interp *Interpreter) cfg(root *node, importPath string) ([]*node, error) {
|
||||
// by constOp and available in n.rval. Nothing else to do at execution.
|
||||
n.gen = nop
|
||||
n.findex = notInFrame
|
||||
case n.anc.kind == assignStmt && n.anc.action == aAssign:
|
||||
case n.anc.kind == assignStmt && n.anc.action == aAssign && n.anc.nleft == 1:
|
||||
// To avoid a copy in frame, if the result is to be assigned, store it directly
|
||||
// at the frame location of destination.
|
||||
dest := n.anc.child[childPos(n)-n.anc.nright]
|
||||
@@ -740,7 +753,13 @@ func (interp *Interpreter) cfg(root *node, importPath string) ([]*node, error) {
|
||||
wireChild(n)
|
||||
t := n.child[0].typ
|
||||
switch t.cat {
|
||||
case aliasT, ptrT:
|
||||
case aliasT:
|
||||
if isString(t.val.TypeOf()) {
|
||||
n.typ = sc.getType("byte")
|
||||
break
|
||||
}
|
||||
fallthrough
|
||||
case ptrT:
|
||||
n.typ = t.val
|
||||
if t.val.cat == valueT {
|
||||
n.typ = &itype{cat: valueT, rtype: t.val.rtype.Elem()}
|
||||
@@ -845,14 +864,16 @@ func (interp *Interpreter) cfg(root *node, importPath string) ([]*node, error) {
|
||||
case callExpr:
|
||||
wireChild(n)
|
||||
switch {
|
||||
case interp.isBuiltinCall(n):
|
||||
err = check.builtin(n.child[0].ident, n, n.child[1:], n.action == aCallSlice)
|
||||
case isBuiltinCall(n, sc):
|
||||
c0 := n.child[0]
|
||||
bname := c0.ident
|
||||
err = check.builtin(bname, n, n.child[1:], n.action == aCallSlice)
|
||||
if err != nil {
|
||||
break
|
||||
}
|
||||
|
||||
n.gen = n.child[0].sym.builtin
|
||||
n.child[0].typ = &itype{cat: builtinT}
|
||||
n.gen = c0.sym.builtin
|
||||
c0.typ = &itype{cat: builtinT}
|
||||
if n.typ, err = nodeType(interp, sc, n); err != nil {
|
||||
return
|
||||
}
|
||||
@@ -863,10 +884,28 @@ func (interp *Interpreter) cfg(root *node, importPath string) ([]*node, error) {
|
||||
case n.anc.kind == returnStmt:
|
||||
// Store result directly to frame output location, to avoid a frame copy.
|
||||
n.findex = 0
|
||||
case bname == "cap" && isInConstOrTypeDecl(n):
|
||||
switch n.child[1].typ.TypeOf().Kind() {
|
||||
case reflect.Array, reflect.Chan:
|
||||
capConst(n)
|
||||
default:
|
||||
err = n.cfgErrorf("cap argument is not an array or channel")
|
||||
}
|
||||
n.findex = notInFrame
|
||||
n.gen = nop
|
||||
case bname == "len" && isInConstOrTypeDecl(n):
|
||||
switch n.child[1].typ.TypeOf().Kind() {
|
||||
case reflect.Array, reflect.Chan, reflect.String:
|
||||
lenConst(n)
|
||||
default:
|
||||
err = n.cfgErrorf("len argument is not an array, channel or string")
|
||||
}
|
||||
n.findex = notInFrame
|
||||
n.gen = nop
|
||||
default:
|
||||
n.findex = sc.add(n.typ)
|
||||
}
|
||||
if op, ok := constBltn[n.child[0].ident]; ok && n.anc.action != aAssign {
|
||||
if op, ok := constBltn[bname]; ok && n.anc.action != aAssign {
|
||||
op(n) // pre-compute non-assigned constant :
|
||||
}
|
||||
case n.child[0].isType(sc):
|
||||
@@ -891,9 +930,12 @@ func (interp *Interpreter) cfg(root *node, importPath string) ([]*node, error) {
|
||||
if !c1.typ.implements(c0.typ) {
|
||||
err = n.cfgErrorf("type %v does not implement interface %v", c1.typ.id(), c0.typ.id())
|
||||
}
|
||||
// Pass value as is
|
||||
// Convert type to interface while keeping a reference to the original concrete type.
|
||||
// besides type, the node value remains preserved.
|
||||
n.gen = nop
|
||||
n.typ = c1.typ
|
||||
t := *c0.typ
|
||||
n.typ = &t
|
||||
n.typ.val = c1.typ
|
||||
n.findex = c1.findex
|
||||
n.level = c1.level
|
||||
n.val = c1.val
|
||||
@@ -942,6 +984,20 @@ func (interp *Interpreter) cfg(root *node, importPath string) ([]*node, error) {
|
||||
}
|
||||
}
|
||||
}
|
||||
case isOffsetof(n):
|
||||
if len(n.child) != 2 || n.child[1].kind != selectorExpr || !isStruct(n.child[1].child[0].typ) {
|
||||
err = n.cfgErrorf("Offsetof argument: invalid expression")
|
||||
break
|
||||
}
|
||||
c1 := n.child[1]
|
||||
field, ok := c1.child[0].typ.rtype.FieldByName(c1.child[1].ident)
|
||||
if !ok {
|
||||
err = n.cfgErrorf("struct does not contain field: %s", c1.child[1].ident)
|
||||
break
|
||||
}
|
||||
n.typ = &itype{cat: valueT, rtype: reflect.TypeOf(field.Offset)}
|
||||
n.rval = reflect.ValueOf(field.Offset)
|
||||
n.gen = nop
|
||||
default:
|
||||
err = check.arguments(n, n.child[1:], n.child[0], n.action == aCallSlice)
|
||||
if err != nil {
|
||||
@@ -954,7 +1010,9 @@ func (interp *Interpreter) cfg(root *node, importPath string) ([]*node, error) {
|
||||
}
|
||||
if typ := n.child[0].typ; len(typ.ret) > 0 {
|
||||
n.typ = typ.ret[0]
|
||||
if n.anc.kind == returnStmt {
|
||||
if n.anc.kind == returnStmt && n.typ.id() == sc.def.typ.ret[0].id() {
|
||||
// Store the result directly to the return value area of frame.
|
||||
// It can be done only if no type conversion at return is involved.
|
||||
n.findex = childPos(n)
|
||||
} else {
|
||||
n.findex = sc.add(n.typ)
|
||||
@@ -1011,8 +1069,8 @@ func (interp *Interpreter) cfg(root *node, importPath string) ([]*node, error) {
|
||||
}
|
||||
|
||||
switch n.typ.cat {
|
||||
case arrayT:
|
||||
err = check.arrayLitExpr(child, n.typ.val, n.typ.size)
|
||||
case arrayT, sliceT:
|
||||
err = check.arrayLitExpr(child, n.typ)
|
||||
case mapT:
|
||||
err = check.mapLitExpr(child, n.typ.key, n.typ.val)
|
||||
case structT:
|
||||
@@ -1202,11 +1260,10 @@ func (interp *Interpreter) cfg(root *node, importPath string) ([]*node, error) {
|
||||
}
|
||||
}
|
||||
// Found symbol, populate node info
|
||||
n.typ, n.findex, n.level = sym.typ, sym.index, level
|
||||
n.sym, n.typ, n.findex, n.level = sym, sym.typ, sym.index, level
|
||||
if n.findex < 0 {
|
||||
n.val = sym.node
|
||||
} else {
|
||||
n.sym = sym
|
||||
switch {
|
||||
case sym.kind == constSym && sym.rval.IsValid():
|
||||
n.rval = sym.rval
|
||||
@@ -1394,7 +1451,7 @@ func (interp *Interpreter) cfg(root *node, importPath string) ([]*node, error) {
|
||||
// TODO(mpl): move any of that code to typecheck?
|
||||
c.typ.node = c
|
||||
if !c.typ.assignableTo(typ) {
|
||||
err = fmt.Errorf("cannot use %v (type %v) as type %v in return argument", c.ident, c.typ.cat, typ.cat)
|
||||
err = c.cfgErrorf("cannot use %v (type %v) as type %v in return argument", c.ident, c.typ.cat, typ.cat)
|
||||
return
|
||||
}
|
||||
if c.typ.cat == nilT {
|
||||
@@ -1416,72 +1473,8 @@ func (interp *Interpreter) cfg(root *node, importPath string) ([]*node, error) {
|
||||
err = n.cfgErrorf("undefined type")
|
||||
break
|
||||
}
|
||||
if n.typ.cat == valueT || n.typ.cat == errorT {
|
||||
// Handle object defined in runtime, try to find field or method
|
||||
// Search for method first, as it applies both to types T and *T
|
||||
// Search for field must then be performed on type T only (not *T)
|
||||
switch method, ok := n.typ.rtype.MethodByName(n.child[1].ident); {
|
||||
case ok:
|
||||
hasRecvType := n.typ.rtype.Kind() != reflect.Interface
|
||||
n.val = method.Index
|
||||
n.gen = getIndexBinMethod
|
||||
n.action = aGetMethod
|
||||
n.recv = &receiver{node: n.child[0]}
|
||||
n.typ = &itype{cat: valueT, rtype: method.Type, isBinMethod: true}
|
||||
if hasRecvType {
|
||||
n.typ.recv = n.typ
|
||||
}
|
||||
case n.typ.rtype.Kind() == reflect.Ptr:
|
||||
if field, ok := n.typ.rtype.Elem().FieldByName(n.child[1].ident); ok {
|
||||
n.typ = &itype{cat: valueT, rtype: field.Type}
|
||||
n.val = field.Index
|
||||
n.gen = getPtrIndexSeq
|
||||
} else {
|
||||
err = n.cfgErrorf("undefined field or method: %s", n.child[1].ident)
|
||||
}
|
||||
case n.typ.rtype.Kind() == reflect.Struct:
|
||||
if field, ok := n.typ.rtype.FieldByName(n.child[1].ident); ok {
|
||||
n.typ = &itype{cat: valueT, rtype: field.Type}
|
||||
n.val = field.Index
|
||||
n.gen = getIndexSeq
|
||||
} else {
|
||||
// method lookup failed on type, now lookup on pointer to type
|
||||
pt := reflect.PtrTo(n.typ.rtype)
|
||||
if m2, ok2 := pt.MethodByName(n.child[1].ident); ok2 {
|
||||
n.val = m2.Index
|
||||
n.gen = getIndexBinPtrMethod
|
||||
n.typ = &itype{cat: valueT, rtype: m2.Type, recv: &itype{cat: valueT, rtype: pt}}
|
||||
n.recv = &receiver{node: n.child[0]}
|
||||
n.action = aGetMethod
|
||||
} else {
|
||||
err = n.cfgErrorf("undefined field or method: %s", n.child[1].ident)
|
||||
}
|
||||
}
|
||||
default:
|
||||
err = n.cfgErrorf("undefined field or method: %s", n.child[1].ident)
|
||||
}
|
||||
} else if n.typ.cat == ptrT && (n.typ.val.cat == valueT || n.typ.val.cat == errorT) {
|
||||
// Handle pointer on object defined in runtime
|
||||
if method, ok := n.typ.val.rtype.MethodByName(n.child[1].ident); ok {
|
||||
n.val = method.Index
|
||||
n.typ = &itype{cat: valueT, rtype: method.Type, recv: n.typ}
|
||||
n.recv = &receiver{node: n.child[0]}
|
||||
n.gen = getIndexBinMethod
|
||||
n.action = aGetMethod
|
||||
} else if method, ok := reflect.PtrTo(n.typ.val.rtype).MethodByName(n.child[1].ident); ok {
|
||||
n.val = method.Index
|
||||
n.gen = getIndexBinMethod
|
||||
n.typ = &itype{cat: valueT, rtype: method.Type, recv: &itype{cat: valueT, rtype: reflect.PtrTo(n.typ.val.rtype)}}
|
||||
n.recv = &receiver{node: n.child[0]}
|
||||
n.action = aGetMethod
|
||||
} else if field, ok := n.typ.val.rtype.FieldByName(n.child[1].ident); ok {
|
||||
n.typ = &itype{cat: valueT, rtype: field.Type}
|
||||
n.val = field.Index
|
||||
n.gen = getPtrIndexSeq
|
||||
} else {
|
||||
err = n.cfgErrorf("undefined selector: %s", n.child[1].ident)
|
||||
}
|
||||
} else if n.typ.cat == binPkgT {
|
||||
switch {
|
||||
case n.typ.cat == binPkgT:
|
||||
// Resolve binary package symbol: a type or a value
|
||||
name := n.child[1].ident
|
||||
pkg := n.child[0].sym.typ.path
|
||||
@@ -1489,7 +1482,7 @@ func (interp *Interpreter) cfg(root *node, importPath string) ([]*node, error) {
|
||||
if isBinType(s) {
|
||||
n.typ = &itype{cat: valueT, rtype: s.Type().Elem()}
|
||||
} else {
|
||||
n.typ = &itype{cat: valueT, rtype: s.Type(), untyped: isValueUntyped(s)}
|
||||
n.typ = &itype{cat: valueT, rtype: fixPossibleConstType(s.Type()), untyped: isValueUntyped(s)}
|
||||
n.rval = s
|
||||
}
|
||||
n.action = aGetSym
|
||||
@@ -1497,7 +1490,7 @@ func (interp *Interpreter) cfg(root *node, importPath string) ([]*node, error) {
|
||||
} else {
|
||||
err = n.cfgErrorf("package %s \"%s\" has no symbol %s", n.child[0].ident, pkg, name)
|
||||
}
|
||||
} else if n.typ.cat == srcPkgT {
|
||||
case n.typ.cat == srcPkgT:
|
||||
pkg, name := n.child[0].sym.typ.path, n.child[1].ident
|
||||
// Resolve source package symbol
|
||||
if sym, ok := interp.srcPkg[pkg][name]; ok {
|
||||
@@ -1510,70 +1503,169 @@ func (interp *Interpreter) cfg(root *node, importPath string) ([]*node, error) {
|
||||
n.action = aGetSym
|
||||
n.typ = sym.typ
|
||||
n.sym = sym
|
||||
n.recv = sym.recv
|
||||
n.rval = sym.rval
|
||||
} else {
|
||||
err = n.cfgErrorf("undefined selector: %s.%s", pkg, name)
|
||||
}
|
||||
} else if m, lind := n.typ.lookupMethod(n.child[1].ident); m != nil {
|
||||
n.action = aGetMethod
|
||||
if n.child[0].isType(sc) {
|
||||
// Handle method as a function with receiver in 1st argument
|
||||
n.val = m
|
||||
n.findex = notInFrame
|
||||
n.gen = nop
|
||||
n.typ = &itype{}
|
||||
*n.typ = *m.typ
|
||||
n.typ.arg = append([]*itype{n.child[0].typ}, m.typ.arg...)
|
||||
} else {
|
||||
// Handle method with receiver
|
||||
n.gen = getMethod
|
||||
n.val = m
|
||||
n.typ = m.typ
|
||||
case isStruct(n.typ) || isInterfaceSrc(n.typ):
|
||||
// Find a matching field.
|
||||
if ti := n.typ.lookupField(n.child[1].ident); len(ti) > 0 {
|
||||
if isStruct(n.typ) {
|
||||
// If a method of the same name exists, use it if it is shallower than the struct field.
|
||||
// if method's depth is the same as field's, this is an error.
|
||||
d := n.typ.methodDepth(n.child[1].ident)
|
||||
if d >= 0 && d < len(ti) {
|
||||
goto tryMethods
|
||||
}
|
||||
if d == len(ti) {
|
||||
err = n.cfgErrorf("ambiguous selector: %s", n.child[1].ident)
|
||||
break
|
||||
}
|
||||
}
|
||||
n.val = ti
|
||||
switch {
|
||||
case isInterfaceSrc(n.typ):
|
||||
n.typ = n.typ.fieldSeq(ti)
|
||||
n.gen = getMethodByName
|
||||
n.action = aMethod
|
||||
case n.typ.cat == ptrT:
|
||||
n.typ = n.typ.fieldSeq(ti)
|
||||
n.gen = getPtrIndexSeq
|
||||
if n.typ.cat == funcT {
|
||||
// Function in a struct field is always wrapped in reflect.Value.
|
||||
rtype := n.typ.TypeOf()
|
||||
n.typ = &itype{cat: valueT, rtype: rtype, val: n.typ}
|
||||
}
|
||||
default:
|
||||
n.gen = getIndexSeq
|
||||
n.typ = n.typ.fieldSeq(ti)
|
||||
if n.typ.cat == funcT {
|
||||
// Function in a struct field is always wrapped in reflect.Value.
|
||||
rtype := n.typ.TypeOf()
|
||||
n.typ = &itype{cat: valueT, rtype: rtype, val: n.typ}
|
||||
}
|
||||
}
|
||||
break
|
||||
}
|
||||
if s, lind, ok := n.typ.lookupBinField(n.child[1].ident); ok {
|
||||
// Handle an embedded binary field into a struct field.
|
||||
n.gen = getIndexSeqField
|
||||
lind = append(lind, s.Index...)
|
||||
if isStruct(n.typ) {
|
||||
// If a method of the same name exists, use it if it is shallower than the struct field.
|
||||
// if method's depth is the same as field's, this is an error.
|
||||
d := n.typ.methodDepth(n.child[1].ident)
|
||||
if d >= 0 && d < len(lind) {
|
||||
goto tryMethods
|
||||
}
|
||||
if d == len(lind) {
|
||||
err = n.cfgErrorf("ambiguous selector: %s", n.child[1].ident)
|
||||
break
|
||||
}
|
||||
}
|
||||
n.val = lind
|
||||
n.typ = &itype{cat: valueT, rtype: s.Type}
|
||||
break
|
||||
}
|
||||
// No field (embedded or not) matched. Try to match a method.
|
||||
tryMethods:
|
||||
fallthrough
|
||||
default:
|
||||
// Find a matching method.
|
||||
// TODO (marc): simplify the following if/elseif blocks.
|
||||
if n.typ.cat == valueT || n.typ.cat == errorT {
|
||||
switch method, ok := n.typ.rtype.MethodByName(n.child[1].ident); {
|
||||
case ok:
|
||||
hasRecvType := n.typ.rtype.Kind() != reflect.Interface
|
||||
n.val = method.Index
|
||||
n.gen = getIndexBinMethod
|
||||
n.action = aGetMethod
|
||||
n.recv = &receiver{node: n.child[0]}
|
||||
n.typ = &itype{cat: valueT, rtype: method.Type, isBinMethod: true}
|
||||
if hasRecvType {
|
||||
n.typ.recv = n.typ
|
||||
}
|
||||
case n.typ.rtype.Kind() == reflect.Ptr:
|
||||
if field, ok := n.typ.rtype.Elem().FieldByName(n.child[1].ident); ok {
|
||||
n.typ = &itype{cat: valueT, rtype: field.Type}
|
||||
n.val = field.Index
|
||||
n.gen = getPtrIndexSeq
|
||||
break
|
||||
}
|
||||
err = n.cfgErrorf("undefined field or method: %s", n.child[1].ident)
|
||||
case n.typ.rtype.Kind() == reflect.Struct:
|
||||
if field, ok := n.typ.rtype.FieldByName(n.child[1].ident); ok {
|
||||
n.typ = &itype{cat: valueT, rtype: field.Type}
|
||||
n.val = field.Index
|
||||
n.gen = getIndexSeq
|
||||
break
|
||||
}
|
||||
fallthrough
|
||||
default:
|
||||
// method lookup failed on type, now lookup on pointer to type
|
||||
pt := reflect.PtrTo(n.typ.rtype)
|
||||
if m2, ok2 := pt.MethodByName(n.child[1].ident); ok2 {
|
||||
n.val = m2.Index
|
||||
n.gen = getIndexBinPtrMethod
|
||||
n.typ = &itype{cat: valueT, rtype: m2.Type, recv: &itype{cat: valueT, rtype: pt}, isBinMethod: true}
|
||||
n.recv = &receiver{node: n.child[0]}
|
||||
n.action = aGetMethod
|
||||
break
|
||||
}
|
||||
err = n.cfgErrorf("undefined field or method: %s", n.child[1].ident)
|
||||
}
|
||||
} else if n.typ.cat == ptrT && (n.typ.val.cat == valueT || n.typ.val.cat == errorT) {
|
||||
// Handle pointer on object defined in runtime
|
||||
if method, ok := n.typ.val.rtype.MethodByName(n.child[1].ident); ok {
|
||||
n.val = method.Index
|
||||
n.typ = &itype{cat: valueT, rtype: method.Type, recv: n.typ, isBinMethod: true}
|
||||
n.recv = &receiver{node: n.child[0]}
|
||||
n.gen = getIndexBinElemMethod
|
||||
n.action = aGetMethod
|
||||
} else if method, ok := reflect.PtrTo(n.typ.val.rtype).MethodByName(n.child[1].ident); ok {
|
||||
n.val = method.Index
|
||||
n.gen = getIndexBinMethod
|
||||
n.typ = &itype{cat: valueT, rtype: method.Type, recv: &itype{cat: valueT, rtype: reflect.PtrTo(n.typ.val.rtype)}, isBinMethod: true}
|
||||
n.recv = &receiver{node: n.child[0]}
|
||||
n.action = aGetMethod
|
||||
} else if field, ok := n.typ.val.rtype.FieldByName(n.child[1].ident); ok {
|
||||
n.typ = &itype{cat: valueT, rtype: field.Type}
|
||||
n.val = field.Index
|
||||
n.gen = getPtrIndexSeq
|
||||
} else {
|
||||
err = n.cfgErrorf("undefined selector: %s", n.child[1].ident)
|
||||
}
|
||||
} else if m, lind := n.typ.lookupMethod(n.child[1].ident); m != nil {
|
||||
n.action = aGetMethod
|
||||
if n.child[0].isType(sc) {
|
||||
// Handle method as a function with receiver in 1st argument
|
||||
n.val = m
|
||||
n.findex = notInFrame
|
||||
n.gen = nop
|
||||
n.typ = &itype{}
|
||||
*n.typ = *m.typ
|
||||
n.typ.arg = append([]*itype{n.child[0].typ}, m.typ.arg...)
|
||||
} else {
|
||||
// Handle method with receiver
|
||||
n.gen = getMethod
|
||||
n.val = m
|
||||
n.typ = m.typ
|
||||
n.recv = &receiver{node: n.child[0], index: lind}
|
||||
}
|
||||
} else if m, lind, isPtr, ok := n.typ.lookupBinMethod(n.child[1].ident); ok {
|
||||
n.action = aGetMethod
|
||||
if isPtr && n.typ.fieldSeq(lind).cat != ptrT {
|
||||
n.gen = getIndexSeqPtrMethod
|
||||
} else {
|
||||
n.gen = getIndexSeqMethod
|
||||
}
|
||||
n.recv = &receiver{node: n.child[0], index: lind}
|
||||
}
|
||||
} else if m, lind, isPtr, ok := n.typ.lookupBinMethod(n.child[1].ident); ok {
|
||||
n.action = aGetMethod
|
||||
if isPtr && n.typ.fieldSeq(lind).cat != ptrT {
|
||||
n.gen = getIndexSeqPtrMethod
|
||||
n.val = append([]int{m.Index}, lind...)
|
||||
n.typ = &itype{cat: valueT, rtype: m.Type, recv: n.child[0].typ, isBinMethod: true}
|
||||
} else {
|
||||
n.gen = getIndexSeqMethod
|
||||
err = n.cfgErrorf("undefined selector: %s", n.child[1].ident)
|
||||
}
|
||||
n.recv = &receiver{node: n.child[0], index: lind}
|
||||
n.val = append([]int{m.Index}, lind...)
|
||||
n.typ = &itype{cat: valueT, rtype: m.Type, recv: n.child[0].typ}
|
||||
} else if ti := n.typ.lookupField(n.child[1].ident); len(ti) > 0 {
|
||||
// Handle struct field
|
||||
n.val = ti
|
||||
switch {
|
||||
case isInterfaceSrc(n.typ):
|
||||
n.typ = n.typ.fieldSeq(ti)
|
||||
n.gen = getMethodByName
|
||||
n.action = aMethod
|
||||
case n.typ.cat == ptrT:
|
||||
n.typ = n.typ.fieldSeq(ti)
|
||||
n.gen = getPtrIndexSeq
|
||||
if n.typ.cat == funcT {
|
||||
// function in a struct field is always wrapped in reflect.Value
|
||||
rtype := n.typ.TypeOf()
|
||||
n.typ = &itype{cat: valueT, rtype: rtype, val: n.typ}
|
||||
}
|
||||
default:
|
||||
n.gen = getIndexSeq
|
||||
n.typ = n.typ.fieldSeq(ti)
|
||||
if n.typ.cat == funcT {
|
||||
// function in a struct field is always wrapped in reflect.Value
|
||||
rtype := n.typ.TypeOf()
|
||||
n.typ = &itype{cat: valueT, rtype: rtype, val: n.typ}
|
||||
}
|
||||
}
|
||||
} else if s, lind, ok := n.typ.lookupBinField(n.child[1].ident); ok {
|
||||
// Handle an embedded binary field into a struct field
|
||||
n.gen = getIndexSeqField
|
||||
lind = append(lind, s.Index...)
|
||||
n.val = lind
|
||||
n.typ = &itype{cat: valueT, rtype: s.Type}
|
||||
} else {
|
||||
err = n.cfgErrorf("undefined selector: %s", n.child[1].ident)
|
||||
}
|
||||
if err == nil && n.findex != -1 {
|
||||
n.findex = sc.add(n.typ)
|
||||
@@ -1682,15 +1774,17 @@ func (interp *Interpreter) cfg(root *node, importPath string) ([]*node, error) {
|
||||
break
|
||||
}
|
||||
// Chain case clauses.
|
||||
for i, c := range clauses[:l-1] {
|
||||
// Chain to next clause.
|
||||
setFNext(c, clauses[i+1])
|
||||
for i := l - 1; i >= 0; i-- {
|
||||
c := clauses[i]
|
||||
if len(c.child) == 0 {
|
||||
c.tnext = n // Clause body is empty, exit.
|
||||
} else {
|
||||
body := c.lastChild()
|
||||
c.tnext = body.start
|
||||
if len(body.child) > 0 && body.lastChild().kind == fallthroughtStmt {
|
||||
c.child[0].tnext = c
|
||||
c.start = c.child[0].start
|
||||
|
||||
if i < l-1 && len(body.child) > 0 && body.lastChild().kind == fallthroughtStmt {
|
||||
if n.kind == typeSwitch {
|
||||
err = body.lastChild().cfgErrorf("cannot fallthrough in type switch")
|
||||
}
|
||||
@@ -1703,15 +1797,16 @@ func (interp *Interpreter) cfg(root *node, importPath string) ([]*node, error) {
|
||||
body.tnext = n // Exit switch at end of clause body.
|
||||
}
|
||||
}
|
||||
}
|
||||
c := clauses[l-1] // Last clause.
|
||||
c.fnext = n
|
||||
if len(c.child) == 0 {
|
||||
c.tnext = n // Clause body is empty, exit.
|
||||
} else {
|
||||
body := c.lastChild()
|
||||
c.tnext = body.start
|
||||
body.tnext = n
|
||||
|
||||
if i == l-1 {
|
||||
setFNext(clauses[i], n)
|
||||
continue
|
||||
}
|
||||
if len(clauses[i+1].child) > 1 {
|
||||
setFNext(c, clauses[i+1].start)
|
||||
} else {
|
||||
setFNext(c, clauses[i+1])
|
||||
}
|
||||
}
|
||||
n.start = n.child[0].start
|
||||
n.child[0].tnext = sbn.start
|
||||
@@ -1833,7 +1928,7 @@ func (interp *Interpreter) cfg(root *node, importPath string) ([]*node, error) {
|
||||
case n.rval.IsValid():
|
||||
n.gen = nop
|
||||
n.findex = notInFrame
|
||||
case n.anc.kind == assignStmt && n.anc.action == aAssign:
|
||||
case n.anc.kind == assignStmt && n.anc.action == aAssign && n.anc.nright == 1:
|
||||
dest := n.anc.child[childPos(n)-n.anc.nright]
|
||||
n.typ = dest.typ
|
||||
n.findex = dest.findex
|
||||
@@ -1854,6 +1949,7 @@ func (interp *Interpreter) cfg(root *node, importPath string) ([]*node, error) {
|
||||
return
|
||||
}
|
||||
}
|
||||
|
||||
for _, c := range n.child[:l] {
|
||||
var index int
|
||||
if sc.global {
|
||||
@@ -2127,7 +2223,10 @@ func (n *node) isType(sc *scope) bool {
|
||||
suffixedPkg := filepath.Join(pkg, baseName)
|
||||
sym, _, ok := sc.lookup(suffixedPkg)
|
||||
if !ok {
|
||||
return false
|
||||
sym, _, ok = sc.lookup(pkg)
|
||||
if !ok {
|
||||
return false
|
||||
}
|
||||
}
|
||||
if sym.kind != pkgSym {
|
||||
return false
|
||||
@@ -2317,6 +2416,20 @@ func isRecursiveField(n *node) bool {
|
||||
return false
|
||||
}
|
||||
|
||||
func isInConstOrTypeDecl(n *node) bool {
|
||||
anc := n.anc
|
||||
for anc != nil {
|
||||
switch anc.kind {
|
||||
case constDecl, typeDecl:
|
||||
return true
|
||||
case varDecl, funcDecl:
|
||||
return false
|
||||
}
|
||||
anc = anc.anc
|
||||
}
|
||||
return false
|
||||
}
|
||||
|
||||
// isNewDefine returns true if node refers to a new definition.
|
||||
func isNewDefine(n *node, sc *scope) bool {
|
||||
if n.ident == "_" {
|
||||
@@ -2341,6 +2454,10 @@ func isMethod(n *node) bool {
|
||||
return len(n.child[0].child) > 0 // receiver defined
|
||||
}
|
||||
|
||||
func isFuncField(n *node) bool {
|
||||
return isField(n) && isFunc(n.typ)
|
||||
}
|
||||
|
||||
func isMapEntry(n *node) bool {
|
||||
return n.action == aGetIndex && isMap(n.child[0].typ)
|
||||
}
|
||||
@@ -2353,6 +2470,10 @@ func isBinCall(n *node) bool {
|
||||
return isCall(n) && n.child[0].typ.cat == valueT && n.child[0].typ.rtype.Kind() == reflect.Func
|
||||
}
|
||||
|
||||
func isOffsetof(n *node) bool {
|
||||
return isCall(n) && n.child[0].typ.cat == valueT && n.child[0].rval.String() == "Offsetof"
|
||||
}
|
||||
|
||||
func mustReturnValue(n *node) bool {
|
||||
if len(n.child) < 2 {
|
||||
return false
|
||||
@@ -2443,7 +2564,9 @@ func gotoLabel(s *symbol) {
|
||||
return
|
||||
}
|
||||
for _, c := range s.from {
|
||||
c.tnext = s.node.start
|
||||
if c.tnext == nil {
|
||||
c.tnext = s.node.start
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -2451,7 +2574,7 @@ func compositeGenerator(n *node, typ *itype, rtyp reflect.Type) (gen bltnGenerat
|
||||
switch typ.cat {
|
||||
case aliasT, ptrT:
|
||||
gen = compositeGenerator(n, n.typ.val, rtyp)
|
||||
case arrayT:
|
||||
case arrayT, sliceT:
|
||||
gen = arrayLit
|
||||
case mapT:
|
||||
gen = mapLit
|
||||
@@ -2476,6 +2599,10 @@ func compositeGenerator(n *node, typ *itype, rtyp reflect.Type) (gen bltnGenerat
|
||||
if rtyp == nil {
|
||||
rtyp = n.typ.rtype
|
||||
}
|
||||
// TODO(mpl): I do not understand where this side-effect is coming from, and why it happens. quickfix for now.
|
||||
if rtyp == nil {
|
||||
rtyp = n.typ.val.rtype
|
||||
}
|
||||
switch k := rtyp.Kind(); k {
|
||||
case reflect.Struct:
|
||||
if n.nleft == 1 {
|
||||
@@ -2488,7 +2615,7 @@ func compositeGenerator(n *node, typ *itype, rtyp reflect.Type) (gen bltnGenerat
|
||||
gen = compositeBinMap
|
||||
case reflect.Ptr:
|
||||
gen = compositeGenerator(n, typ, n.typ.val.rtype)
|
||||
case reflect.Slice:
|
||||
case reflect.Slice, reflect.Array:
|
||||
gen = compositeBinSlice
|
||||
default:
|
||||
log.Panic(n.cfgErrorf("compositeGenerator not implemented for type kind: %s", k))
|
||||
@@ -2501,8 +2628,8 @@ func compositeGenerator(n *node, typ *itype, rtyp reflect.Type) (gen bltnGenerat
|
||||
// array variable it is determined from the value's type, otherwise it is
|
||||
// computed from the source definition.
|
||||
func arrayTypeLen(n *node) int {
|
||||
if n.typ != nil && n.typ.sizedef {
|
||||
return n.typ.size
|
||||
if n.typ != nil && n.typ.cat == arrayT {
|
||||
return n.typ.length
|
||||
}
|
||||
max := -1
|
||||
for i, c := range n.child[1:] {
|
||||
@@ -2525,28 +2652,22 @@ func isValueUntyped(v reflect.Value) bool {
|
||||
if v.CanSet() {
|
||||
return false
|
||||
}
|
||||
t := v.Type()
|
||||
if t.Implements(constVal) {
|
||||
return true
|
||||
}
|
||||
return t.String() == t.Kind().String()
|
||||
return v.Type().Implements(constVal)
|
||||
}
|
||||
|
||||
// isArithmeticAction returns true if the node action is an arithmetic operator.
|
||||
func isArithmeticAction(n *node) bool {
|
||||
switch n.action {
|
||||
case aAdd, aAnd, aAndNot, aBitNot, aMul, aQuo, aRem, aShl, aShr, aSub, aXor:
|
||||
case aAdd, aAnd, aAndNot, aBitNot, aMul, aNeg, aOr, aPos, aQuo, aRem, aShl, aShr, aSub, aXor:
|
||||
return true
|
||||
default:
|
||||
return false
|
||||
}
|
||||
return false
|
||||
}
|
||||
|
||||
func isBoolAction(n *node) bool {
|
||||
switch n.action {
|
||||
case aEqual, aGreater, aGreaterEqual, aLand, aLor, aLower, aLowerEqual, aNot, aNotEqual:
|
||||
return true
|
||||
default:
|
||||
return false
|
||||
}
|
||||
return false
|
||||
}
|
||||
|
||||
@@ -27,6 +27,7 @@ func (interp *Interpreter) gta(root *node, rpath, importPath string) ([]*node, e
|
||||
// values which may be used in further declarations.
|
||||
if _, err = interp.cfg(n, importPath); err != nil {
|
||||
// No error processing here, to allow recovery in subtree nodes.
|
||||
// TODO(marc): check for a non recoverable error and return it for better diagnostic.
|
||||
err = nil
|
||||
}
|
||||
|
||||
@@ -79,9 +80,7 @@ func (interp *Interpreter) gta(root *node, rpath, importPath string) ([]*node, e
|
||||
if typ.isBinMethod {
|
||||
typ = &itype{cat: valueT, rtype: typ.methodCallType(), isBinMethod: true, scope: sc}
|
||||
}
|
||||
if sc.sym[dest.ident] == nil || sc.sym[dest.ident].typ.incomplete {
|
||||
sc.sym[dest.ident] = &symbol{kind: varSym, global: true, index: sc.add(typ), typ: typ, rval: val, node: n}
|
||||
}
|
||||
sc.sym[dest.ident] = &symbol{kind: varSym, global: true, index: sc.add(typ), typ: typ, rval: val, node: n}
|
||||
if n.anc.kind == constDecl {
|
||||
sc.sym[dest.ident].kind = constSym
|
||||
if childPos(n) == len(n.anc.child)-1 {
|
||||
@@ -188,11 +187,11 @@ func (interp *Interpreter) gta(root *node, rpath, importPath string) ([]*node, e
|
||||
}
|
||||
// Try to import a binary package first, or a source package
|
||||
var pkgName string
|
||||
if interp.binPkg[ipath] != nil {
|
||||
if pkg := interp.binPkg[ipath]; pkg != nil {
|
||||
switch name {
|
||||
case "_": // no import of symbols
|
||||
case ".": // import symbols in current scope
|
||||
for n, v := range interp.binPkg[ipath] {
|
||||
for n, v := range pkg {
|
||||
typ := v.Type()
|
||||
if isBinType(v) {
|
||||
typ = typ.Elem()
|
||||
@@ -201,16 +200,19 @@ func (interp *Interpreter) gta(root *node, rpath, importPath string) ([]*node, e
|
||||
}
|
||||
default: // import symbols in package namespace
|
||||
if name == "" {
|
||||
name = identifier.FindString(ipath)
|
||||
name = interp.pkgNames[ipath]
|
||||
}
|
||||
// imports of a same package are all mapped in the same scope, so we cannot just
|
||||
// Imports of a same package are all mapped in the same scope, so we cannot just
|
||||
// map them by their names, otherwise we could have collisions from same-name
|
||||
// imports in different source files of the same package. Therefore, we suffix
|
||||
// the key with the basename of the source file.
|
||||
name = filepath.Join(name, baseName)
|
||||
if _, exists := sc.sym[name]; !exists {
|
||||
if sym, exists := sc.sym[name]; !exists {
|
||||
sc.sym[name] = &symbol{kind: pkgSym, typ: &itype{cat: binPkgT, path: ipath, scope: sc}}
|
||||
break
|
||||
} else if sym.kind == pkgSym && sym.typ.cat == srcPkgT && sym.typ.path == ipath {
|
||||
// ignore re-import of identical package
|
||||
break
|
||||
}
|
||||
|
||||
// redeclaration error. Not caught by the parser.
|
||||
@@ -232,9 +234,12 @@ func (interp *Interpreter) gta(root *node, rpath, importPath string) ([]*node, e
|
||||
name = pkgName
|
||||
}
|
||||
name = filepath.Join(name, baseName)
|
||||
if _, exists := sc.sym[name]; !exists {
|
||||
if sym, exists := sc.sym[name]; !exists {
|
||||
sc.sym[name] = &symbol{kind: pkgSym, typ: &itype{cat: srcPkgT, path: ipath, scope: sc}}
|
||||
break
|
||||
} else if sym.kind == pkgSym && sym.typ.cat == srcPkgT && sym.typ.path == ipath {
|
||||
// ignore re-import of identical package
|
||||
break
|
||||
}
|
||||
|
||||
// redeclaration error
|
||||
@@ -319,7 +324,50 @@ func (interp *Interpreter) gtaRetry(nodes []*node, importPath string) error {
|
||||
}
|
||||
|
||||
if len(revisit) > 0 {
|
||||
return revisit[0].cfgErrorf("constant definition loop")
|
||||
n := revisit[0]
|
||||
if n.kind == typeSpec {
|
||||
if err := definedType(n.typ); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
return n.cfgErrorf("constant definition loop")
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func definedType(typ *itype) error {
|
||||
if !typ.incomplete {
|
||||
return nil
|
||||
}
|
||||
switch typ.cat {
|
||||
case interfaceT, structT:
|
||||
for _, f := range typ.field {
|
||||
if err := definedType(f.typ); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
case funcT:
|
||||
for _, t := range typ.arg {
|
||||
if err := definedType(t); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
for _, t := range typ.ret {
|
||||
if err := definedType(t); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
case mapT:
|
||||
if err := definedType(typ.key); err != nil {
|
||||
return err
|
||||
}
|
||||
fallthrough
|
||||
case aliasT, arrayT, chanT, chanSendT, chanRecvT, ptrT, variadicT:
|
||||
if err := definedType(typ.val); err != nil {
|
||||
return err
|
||||
}
|
||||
case nilT:
|
||||
return typ.node.cfgErrorf("undefined: %s", typ.node.ident)
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
146
interp/interp.go
146
interp/interp.go
@@ -14,6 +14,8 @@ import (
|
||||
"log"
|
||||
"os"
|
||||
"os/signal"
|
||||
"path"
|
||||
"path/filepath"
|
||||
"reflect"
|
||||
"runtime"
|
||||
"runtime/debug"
|
||||
@@ -75,10 +77,10 @@ type frame struct {
|
||||
done reflect.SelectCase // for cancellation of channel operations
|
||||
}
|
||||
|
||||
func newFrame(anc *frame, len int, id uint64) *frame {
|
||||
func newFrame(anc *frame, length int, id uint64) *frame {
|
||||
f := &frame{
|
||||
anc: anc,
|
||||
data: make([]reflect.Value, len),
|
||||
data: make([]reflect.Value, length),
|
||||
id: id,
|
||||
}
|
||||
if anc == nil {
|
||||
@@ -92,21 +94,29 @@ func newFrame(anc *frame, len int, id uint64) *frame {
|
||||
|
||||
func (f *frame) runid() uint64 { return atomic.LoadUint64(&f.id) }
|
||||
func (f *frame) setrunid(id uint64) { atomic.StoreUint64(&f.id, id) }
|
||||
func (f *frame) clone() *frame {
|
||||
func (f *frame) clone(fork bool) *frame {
|
||||
f.mutex.RLock()
|
||||
defer f.mutex.RUnlock()
|
||||
return &frame{
|
||||
nf := &frame{
|
||||
anc: f.anc,
|
||||
root: f.root,
|
||||
data: f.data,
|
||||
deferred: f.deferred,
|
||||
recovered: f.recovered,
|
||||
id: f.runid(),
|
||||
done: f.done,
|
||||
}
|
||||
if fork {
|
||||
nf.data = make([]reflect.Value, len(f.data))
|
||||
copy(nf.data, f.data)
|
||||
} else {
|
||||
nf.data = f.data
|
||||
}
|
||||
return nf
|
||||
}
|
||||
|
||||
// Exports stores the map of binary packages per package path.
|
||||
// The package path is the path joined from the import path and the package name
|
||||
// as specified in source files by the "package" statement.
|
||||
type Exports map[string]map[string]reflect.Value
|
||||
|
||||
// imports stores the map of source packages per package path.
|
||||
@@ -118,13 +128,14 @@ type opt struct {
|
||||
cfgDot bool // display CFG graph (debug)
|
||||
// dotCmd is the command to process the dot graph produced when astDot and/or
|
||||
// cfgDot is enabled. It defaults to 'dot -Tdot -o <filename>.dot'.
|
||||
dotCmd string
|
||||
noRun bool // compile, but do not run
|
||||
fastChan bool // disable cancellable chan operations
|
||||
context build.Context // build context: GOPATH, build constraints
|
||||
stdin io.Reader // standard input
|
||||
stdout io.Writer // standard output
|
||||
stderr io.Writer // standard error
|
||||
dotCmd string
|
||||
noRun bool // compile, but do not run
|
||||
fastChan bool // disable cancellable chan operations
|
||||
context build.Context // build context: GOPATH, build constraints
|
||||
specialStdio bool // Allows os.Stdin, os.Stdout, os.Stderr to not be file descriptors
|
||||
stdin io.Reader // standard input
|
||||
stdout io.Writer // standard output
|
||||
stderr io.Writer // standard error
|
||||
}
|
||||
|
||||
// Interpreter contains global resources and state.
|
||||
@@ -162,7 +173,7 @@ type Interpreter struct {
|
||||
const (
|
||||
mainID = "main"
|
||||
selfPrefix = "github.com/traefik/yaegi"
|
||||
selfPath = selfPrefix + "/interp"
|
||||
selfPath = selfPrefix + "/interp/interp"
|
||||
// DefaultSourceName is the name used by default when the name of the input
|
||||
// source file has not been specified for an Eval.
|
||||
// TODO(mpl): something even more special as a name?
|
||||
@@ -174,6 +185,9 @@ const (
|
||||
NoTest = true
|
||||
)
|
||||
|
||||
// Self points to the current interpreter if accessed from within itself, or is nil.
|
||||
var Self *Interpreter
|
||||
|
||||
// Symbols exposes interpreter values.
|
||||
var Symbols = Exports{
|
||||
selfPath: map[string]reflect.Value{
|
||||
@@ -189,6 +203,7 @@ func init() { Symbols[selfPath]["Symbols"] = reflect.ValueOf(Symbols) }
|
||||
|
||||
// _error is a wrapper of error interface type.
|
||||
type _error struct {
|
||||
IValue interface{}
|
||||
WError func() string
|
||||
}
|
||||
|
||||
@@ -208,7 +223,7 @@ type Panic struct {
|
||||
}
|
||||
|
||||
// TODO: Capture interpreter stack frames also and remove
|
||||
// fmt.Println(n.cfgErrorf("panic")) in runCfg.
|
||||
// fmt.Fprintln(n.interp.stderr, oNode.cfgErrorf("panic")) in runCfg.
|
||||
|
||||
func (e Panic) Error() string { return fmt.Sprint(e.Value) }
|
||||
|
||||
@@ -235,7 +250,7 @@ type Options struct {
|
||||
BuildTags []string
|
||||
|
||||
// Standard input, output and error streams.
|
||||
// They default to os.Stding, os.Stdout and os.Stderr respectively.
|
||||
// They default to os.Stdin, os.Stdout and os.Stderr respectively.
|
||||
Stdin io.Reader
|
||||
Stdout, Stderr io.Writer
|
||||
}
|
||||
@@ -288,6 +303,10 @@ func New(options Options) *Interpreter {
|
||||
|
||||
// fastChan disables the cancellable version of channel operations in evalWithContext
|
||||
i.opt.fastChan, _ = strconv.ParseBool(os.Getenv("YAEGI_FAST_CHAN"))
|
||||
|
||||
// specialStdio allows to assign directly io.Writer and io.Reader to os.Stdxxx, even if they are not file descriptors.
|
||||
i.opt.specialStdio, _ = strconv.ParseBool(os.Getenv("YAEGI_SPECIAL_STDIO"))
|
||||
|
||||
return &i
|
||||
}
|
||||
|
||||
@@ -635,33 +654,45 @@ func (interp *Interpreter) getWrapper(t reflect.Type) reflect.Type {
|
||||
|
||||
// Use loads binary runtime symbols in the interpreter context so
|
||||
// they can be used in interpreted code.
|
||||
func (interp *Interpreter) Use(values Exports) {
|
||||
func (interp *Interpreter) Use(values Exports) error {
|
||||
for k, v := range values {
|
||||
if k == selfPrefix {
|
||||
importPath := path.Dir(k)
|
||||
packageName := path.Base(k)
|
||||
|
||||
if importPath == "." {
|
||||
return fmt.Errorf("export path %[1]q is missing a package name; did you mean '%[1]s/%[1]s'?", k)
|
||||
}
|
||||
|
||||
if importPath == selfPrefix {
|
||||
interp.hooks.Parse(v)
|
||||
continue
|
||||
}
|
||||
|
||||
if interp.binPkg[k] == nil {
|
||||
interp.binPkg[k] = make(map[string]reflect.Value)
|
||||
if interp.binPkg[importPath] == nil {
|
||||
interp.binPkg[importPath] = make(map[string]reflect.Value)
|
||||
interp.pkgNames[importPath] = packageName
|
||||
}
|
||||
|
||||
for s, sym := range v {
|
||||
interp.binPkg[k][s] = sym
|
||||
interp.binPkg[importPath][s] = sym
|
||||
}
|
||||
if k == selfPath {
|
||||
interp.binPkg[importPath]["Self"] = reflect.ValueOf(interp)
|
||||
}
|
||||
}
|
||||
|
||||
// Checks if input values correspond to stdlib packages by looking for one
|
||||
// well known stdlib package path.
|
||||
if _, ok := values["fmt"]; ok {
|
||||
if _, ok := values["fmt/fmt"]; ok {
|
||||
fixStdio(interp)
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
// fixStdio redefines interpreter stdlib symbols to use the standard input,
|
||||
// output and errror assigned to the interpreter. The changes are limited to
|
||||
// the interpreter only. Global values os.Stdin, os.Stdout and os.Stderr are
|
||||
// not changed. Note that it is possible to escape the virtualized stdio by
|
||||
// the interpreter only.
|
||||
// Note that it is possible to escape the virtualized stdio by
|
||||
// read/write directly to file descriptors 0, 1, 2.
|
||||
func fixStdio(interp *Interpreter) {
|
||||
p := interp.binPkg["fmt"]
|
||||
@@ -708,9 +739,23 @@ func fixStdio(interp *Interpreter) {
|
||||
}
|
||||
|
||||
if p = interp.binPkg["os"]; p != nil {
|
||||
p["Stdin"] = reflect.ValueOf(&stdin).Elem()
|
||||
p["Stdout"] = reflect.ValueOf(&stdout).Elem()
|
||||
p["Stderr"] = reflect.ValueOf(&stderr).Elem()
|
||||
if interp.specialStdio {
|
||||
// Inherit streams from interpreter even if they do not have a file descriptor.
|
||||
p["Stdin"] = reflect.ValueOf(&stdin).Elem()
|
||||
p["Stdout"] = reflect.ValueOf(&stdout).Elem()
|
||||
p["Stderr"] = reflect.ValueOf(&stderr).Elem()
|
||||
} else {
|
||||
// Inherits streams from interpreter only if they have a file descriptor and preserve original type.
|
||||
if s, ok := stdin.(*os.File); ok {
|
||||
p["Stdin"] = reflect.ValueOf(&s).Elem()
|
||||
}
|
||||
if s, ok := stdout.(*os.File); ok {
|
||||
p["Stdout"] = reflect.ValueOf(&s).Elem()
|
||||
}
|
||||
if s, ok := stderr.(*os.File); ok {
|
||||
p["Stderr"] = reflect.ValueOf(&s).Elem()
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -730,24 +775,47 @@ func ignoreScannerError(e *scanner.Error, s string) bool {
|
||||
return false
|
||||
}
|
||||
|
||||
// ImportUsed automatically imports pre-compiled packages included by Use().
|
||||
// This is mainly useful for REPLs, or single command lines. In case of an ambiguous default
|
||||
// package name, for example "rand" for crypto/rand and math/rand, the package name is
|
||||
// constructed by replacing the last "/" by a "_", producing crypto_rand and math_rand.
|
||||
// ImportUsed should not be called more than once, and not after a first Eval, as it may
|
||||
// rename packages.
|
||||
func (interp *Interpreter) ImportUsed() {
|
||||
sc := interp.universe
|
||||
for k := range interp.binPkg {
|
||||
// By construction, the package name is the last path element of the key.
|
||||
name := path.Base(k)
|
||||
if sym, ok := sc.sym[name]; ok {
|
||||
// Handle collision by renaming old and new entries.
|
||||
name2 := key2name(fixKey(sym.typ.path))
|
||||
sc.sym[name2] = sym
|
||||
if name2 != name {
|
||||
delete(sc.sym, name)
|
||||
}
|
||||
name = key2name(fixKey(k))
|
||||
}
|
||||
sc.sym[name] = &symbol{kind: pkgSym, typ: &itype{cat: binPkgT, path: k, scope: sc}}
|
||||
}
|
||||
}
|
||||
|
||||
func key2name(name string) string {
|
||||
return filepath.Join(name, DefaultSourceName)
|
||||
}
|
||||
|
||||
func fixKey(k string) string {
|
||||
i := strings.LastIndex(k, "/")
|
||||
if i >= 0 {
|
||||
k = k[:i] + "_" + k[i+1:]
|
||||
}
|
||||
return k
|
||||
}
|
||||
|
||||
// REPL performs a Read-Eval-Print-Loop on input reader.
|
||||
// Results are printed to the output writer of the Interpreter, provided as option
|
||||
// at creation time. Errors are printed to the similarly defined errors writer.
|
||||
// The last interpreter result value and error are returned.
|
||||
func (interp *Interpreter) REPL() (reflect.Value, error) {
|
||||
// Preimport used bin packages, to avoid having to import these packages manually
|
||||
// in REPL mode. These packages are already loaded anyway.
|
||||
sc := interp.universe
|
||||
for k := range interp.binPkg {
|
||||
name := identifier.FindString(k)
|
||||
if name == "" || name == "rand" || name == "scanner" || name == "template" || name == "pprof" {
|
||||
// Skip any package with an ambiguous name (i.e crypto/rand vs math/rand).
|
||||
// Those will have to be imported explicitly.
|
||||
continue
|
||||
}
|
||||
sc.sym[name] = &symbol{kind: pkgSym, typ: &itype{cat: binPkgT, path: k, scope: sc}}
|
||||
}
|
||||
|
||||
in, out, errs := interp.stdin, interp.stdout, interp.stderr
|
||||
ctx, cancel := context.WithCancel(context.Background())
|
||||
end := make(chan struct{}) // channel to terminate the REPL
|
||||
|
||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user