Compare commits
115 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 | ||
|
|
24b5375636 | ||
|
|
a83f492309 | ||
|
|
02c30482cc | ||
|
|
9e1da978b0 | ||
|
|
662838fd80 | ||
|
|
92d65c22f0 | ||
|
|
2db4579b6f | ||
|
|
101633c380 | ||
|
|
1e0f6ece6e | ||
|
|
662d2a6afe | ||
|
|
b25ee3f809 | ||
|
|
81e1e5f206 | ||
|
|
81d8339132 | ||
|
|
d494f9e420 | ||
|
|
6da1107c39 | ||
|
|
38a7331bf9 | ||
|
|
13783889cb | ||
|
|
ed626f3fb9 | ||
|
|
d0a34d467b | ||
|
|
83676577ac | ||
|
|
f0fc907269 | ||
|
|
61f4704925 | ||
|
|
b1ccfbf47f | ||
|
|
0ed4b362dc | ||
|
|
98807387a4 | ||
|
|
c817823ba1 | ||
|
|
3cb8bca81a | ||
|
|
a38d19288f | ||
|
|
7f8ffa6719 |
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]
|
[linters]
|
||||||
enable-all = true
|
enable-all = true
|
||||||
disable = [
|
disable = [
|
||||||
"maligned",
|
"golint", # deprecated
|
||||||
|
"scopelint", # deprecated
|
||||||
|
"interfacer", # deprecated
|
||||||
|
"maligned", # deprecated
|
||||||
"lll",
|
"lll",
|
||||||
"gas",
|
"gas",
|
||||||
"dupl",
|
"dupl",
|
||||||
"prealloc",
|
"prealloc",
|
||||||
"scopelint",
|
|
||||||
"gocyclo",
|
"gocyclo",
|
||||||
|
"cyclop",
|
||||||
"gochecknoinits",
|
"gochecknoinits",
|
||||||
"gochecknoglobals",
|
"gochecknoglobals",
|
||||||
"wsl",
|
"wsl",
|
||||||
|
"nlreturn",
|
||||||
"godox",
|
"godox",
|
||||||
"funlen",
|
"funlen",
|
||||||
"gocognit",
|
"gocognit",
|
||||||
"stylecheck",
|
"stylecheck",
|
||||||
"gomnd",
|
"gomnd",
|
||||||
"testpackage",
|
"testpackage",
|
||||||
|
"paralleltest",
|
||||||
|
"tparallel",
|
||||||
"goerr113",
|
"goerr113",
|
||||||
|
"wrapcheck",
|
||||||
"nestif",
|
"nestif",
|
||||||
"exhaustive",
|
"exhaustive",
|
||||||
"nlreturn",
|
"exhaustivestruct",
|
||||||
|
"forbidigo",
|
||||||
|
"ifshort",
|
||||||
|
"forcetypeassert",
|
||||||
|
"errorlint", # TODO: must be reactivate before fixes
|
||||||
]
|
]
|
||||||
|
|
||||||
[issues]
|
[issues]
|
||||||
@@ -54,6 +65,9 @@
|
|||||||
[[issues.exclude-rules]]
|
[[issues.exclude-rules]]
|
||||||
path = "interp/.+_test\\.go"
|
path = "interp/.+_test\\.go"
|
||||||
linters = ["goconst"]
|
linters = ["goconst"]
|
||||||
|
[[issues.exclude-rules]]
|
||||||
|
path = "interp/.+_test\\.go"
|
||||||
|
text = "var-declaration:"
|
||||||
|
|
||||||
[[issues.exclude-rules]]
|
[[issues.exclude-rules]]
|
||||||
path = "interp/interp.go"
|
path = "interp/interp.go"
|
||||||
@@ -61,3 +75,6 @@
|
|||||||
[[issues.exclude-rules]]
|
[[issues.exclude-rules]]
|
||||||
path = "interp/interp.go"
|
path = "interp/interp.go"
|
||||||
text = "`out` can be `io.Writer`"
|
text = "`out` can be `io.Writer`"
|
||||||
|
[[issues.exclude-rules]]
|
||||||
|
path = "interp/interp_eval_test.go"
|
||||||
|
linters = ["thelper"]
|
||||||
|
|||||||
@@ -47,7 +47,7 @@ archives:
|
|||||||
- LICENSE
|
- LICENSE
|
||||||
|
|
||||||
brews:
|
brews:
|
||||||
- github:
|
- tap:
|
||||||
owner: traefik
|
owner: traefik
|
||||||
name: homebrew-tap
|
name: homebrew-tap
|
||||||
commit_author:
|
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>
|
</p>
|
||||||
|
|
||||||
[](https://github.com/traefik/yaegi/releases)
|
[](https://github.com/traefik/yaegi/releases)
|
||||||
[](https://travis-ci.com/traefik/yaegi)
|
[](https://github.com/traefik/yaegi/actions/workflows/main.yml)
|
||||||
[](https://godoc.org/github.com/traefik/yaegi)
|
[](https://pkg.go.dev/mod/github.com/traefik/yaegi)
|
||||||
[](https://community.containo.us/c/yaegi)
|
[](https://community.traefik.io/c/yaegi)
|
||||||
|
|
||||||
Yaegi is Another Elegant Go Interpreter.
|
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.
|
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
|
* Works everywhere Go works
|
||||||
* All Go & runtime resources accessible from script (with control)
|
* All Go & runtime resources accessible from script (with control)
|
||||||
* Security: `unsafe` and `syscall` packages neither used nor exported by default
|
* 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
|
## Install
|
||||||
|
|
||||||
@@ -186,7 +186,7 @@ Beside the known [bugs] which are supposed to be fixed in the short term, there
|
|||||||
[Apache 2.0][License].
|
[Apache 2.0][License].
|
||||||
|
|
||||||
[specs]: https://golang.org/ref/spec
|
[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
|
[license]: https://github.com/traefik/yaegi/blob/master/LICENSE
|
||||||
[github]: https://github.com/traefik/yaegi
|
[github]: https://github.com/traefik/yaegi
|
||||||
[bugs]: https://github.com/traefik/yaegi/issues?q=is%3Aissue+is%3Aopen+label%3Abug
|
[bugs]: https://github.com/traefik/yaegi/issues?q=is%3Aissue+is%3Aopen+label%3Abug
|
||||||
|
|||||||
@@ -1,10 +1,41 @@
|
|||||||
package main
|
package main
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
b := 2
|
b := 2 // int
|
||||||
var a interface{} = 5 + b
|
|
||||||
|
var c int = 5 + b
|
||||||
|
println(c)
|
||||||
|
|
||||||
|
var d int32 = 6 + int32(b)
|
||||||
|
println(d)
|
||||||
|
|
||||||
|
var a interface{} = 7 + b
|
||||||
println(a.(int))
|
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:
|
// Output:
|
||||||
// 7
|
// 7
|
||||||
|
// 8
|
||||||
|
// 9
|
||||||
|
// 10
|
||||||
|
// 11
|
||||||
|
// 12
|
||||||
|
// 13
|
||||||
|
// hello
|
||||||
|
|||||||
63
_test/addr2.go
Normal file
63
_test/addr2.go
Normal file
@@ -0,0 +1,63 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/xml"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Email struct {
|
||||||
|
Where string `xml:"where,attr"`
|
||||||
|
Addr string
|
||||||
|
}
|
||||||
|
|
||||||
|
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)
|
||||||
|
}
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
data := `
|
||||||
|
<Email where='work'>
|
||||||
|
<Addr>bob@work.com</Addr>
|
||||||
|
</Email>
|
||||||
|
`
|
||||||
|
v := Email{}
|
||||||
|
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)
|
||||||
|
}
|
||||||
|
|
||||||
|
// 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>]
|
||||||
128
_test/assert0.go
Normal file
128
_test/assert0.go
Normal file
@@ -0,0 +1,128 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
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) {
|
||||||
|
return len(p), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func usesWriter(w MyWriter) {
|
||||||
|
n, _ := w.Write([]byte("hello world"))
|
||||||
|
fmt.Println(n)
|
||||||
|
}
|
||||||
|
|
||||||
|
type MyStringer interface {
|
||||||
|
String() string
|
||||||
|
}
|
||||||
|
|
||||||
|
type DummyStringer interface {
|
||||||
|
String() string
|
||||||
|
}
|
||||||
|
|
||||||
|
func usesStringer(s MyStringer) {
|
||||||
|
fmt.Println(s.String())
|
||||||
|
}
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
// TODO(mpl): restore when we can deal with empty interface.
|
||||||
|
// var t interface{}
|
||||||
|
var t DummyWriter
|
||||||
|
t = TestStruct{}
|
||||||
|
var tw MyWriter
|
||||||
|
var ok bool
|
||||||
|
tw, ok = t.(MyWriter)
|
||||||
|
if !ok {
|
||||||
|
fmt.Println("TestStruct does not implement MyWriter")
|
||||||
|
} else {
|
||||||
|
fmt.Println("TestStruct implements MyWriter")
|
||||||
|
usesWriter(tw)
|
||||||
|
}
|
||||||
|
n, _ := t.(MyWriter).Write([]byte("hello world"))
|
||||||
|
fmt.Println(n)
|
||||||
|
|
||||||
|
// not redundant with the above, because it goes through a slightly different code path.
|
||||||
|
if _, ok := t.(MyWriter); !ok {
|
||||||
|
fmt.Println("TestStruct does not implement MyWriter")
|
||||||
|
return
|
||||||
|
} else {
|
||||||
|
fmt.Println("TestStruct implements MyWriter")
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO(mpl): restore
|
||||||
|
/*
|
||||||
|
t = 42
|
||||||
|
foo, ok := t.(MyWriter)
|
||||||
|
if !ok {
|
||||||
|
fmt.Println("42 does not implement MyWriter")
|
||||||
|
} else {
|
||||||
|
fmt.Println("42 implements MyWriter")
|
||||||
|
}
|
||||||
|
_ = foo
|
||||||
|
|
||||||
|
if _, ok := t.(MyWriter); !ok {
|
||||||
|
fmt.Println("42 does not implement MyWriter")
|
||||||
|
} else {
|
||||||
|
fmt.Println("42 implements MyWriter")
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
|
||||||
|
// var tt interface{}
|
||||||
|
var tt DummyStringer
|
||||||
|
tt = time.Nanosecond
|
||||||
|
var myD MyStringer
|
||||||
|
myD, ok = tt.(MyStringer)
|
||||||
|
if !ok {
|
||||||
|
fmt.Println("time.Nanosecond does not implement MyStringer")
|
||||||
|
} else {
|
||||||
|
fmt.Println("time.Nanosecond implements MyStringer")
|
||||||
|
usesStringer(myD)
|
||||||
|
}
|
||||||
|
fmt.Println(tt.(MyStringer).String())
|
||||||
|
|
||||||
|
if _, ok := tt.(MyStringer); !ok {
|
||||||
|
fmt.Println("time.Nanosecond does not implement MyStringer")
|
||||||
|
} else {
|
||||||
|
fmt.Println("time.Nanosecond implements MyStringer")
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO(mpl): restore
|
||||||
|
/*
|
||||||
|
tt = 42
|
||||||
|
bar, ok := tt.(MyStringer)
|
||||||
|
if !ok {
|
||||||
|
fmt.Println("42 does not implement MyStringer")
|
||||||
|
} else {
|
||||||
|
fmt.Println("42 implements MyStringer")
|
||||||
|
}
|
||||||
|
_ = bar
|
||||||
|
|
||||||
|
if _, ok := tt.(MyStringer); !ok {
|
||||||
|
fmt.Println("42 does not implement MyStringer")
|
||||||
|
} else {
|
||||||
|
fmt.Println("42 implements MyStringer")
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
}
|
||||||
|
|
||||||
|
// Output:
|
||||||
|
// TestStruct implements MyWriter
|
||||||
|
// 11
|
||||||
|
// 11
|
||||||
|
// TestStruct implements MyWriter
|
||||||
|
// time.Nanosecond implements MyStringer
|
||||||
|
// 1ns
|
||||||
|
// 1ns
|
||||||
|
// time.Nanosecond implements MyStringer
|
||||||
88
_test/assert1.go
Normal file
88
_test/assert1.go
Normal file
@@ -0,0 +1,88 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"reflect"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
type TestStruct struct{}
|
||||||
|
|
||||||
|
func (t TestStruct) String() string {
|
||||||
|
return "hello world"
|
||||||
|
}
|
||||||
|
|
||||||
|
type DummyStringer interface{
|
||||||
|
String() string
|
||||||
|
}
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
aType := reflect.TypeOf((*fmt.Stringer)(nil)).Elem()
|
||||||
|
|
||||||
|
var t interface{}
|
||||||
|
t = time.Nanosecond
|
||||||
|
s, ok := t.(fmt.Stringer)
|
||||||
|
if !ok {
|
||||||
|
fmt.Println("time.Nanosecond does not implement fmt.Stringer")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
fmt.Println(s.String())
|
||||||
|
fmt.Println(t.(fmt.Stringer).String())
|
||||||
|
bType := reflect.TypeOf(time.Nanosecond)
|
||||||
|
fmt.Println(bType.Implements(aType))
|
||||||
|
|
||||||
|
// not redundant with the above, because it goes through a slightly different code path.
|
||||||
|
if _, ok := t.(fmt.Stringer); !ok {
|
||||||
|
fmt.Println("time.Nanosecond does not implement fmt.Stringer")
|
||||||
|
return
|
||||||
|
} else {
|
||||||
|
fmt.Println("time.Nanosecond implements fmt.Stringer")
|
||||||
|
}
|
||||||
|
|
||||||
|
t = 42
|
||||||
|
foo, ok := t.(fmt.Stringer)
|
||||||
|
if !ok {
|
||||||
|
fmt.Println("42 does not implement fmt.Stringer")
|
||||||
|
} else {
|
||||||
|
fmt.Println("42 implements fmt.Stringer")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
_ = foo
|
||||||
|
|
||||||
|
if _, ok := t.(fmt.Stringer); !ok {
|
||||||
|
fmt.Println("42 does not implement fmt.Stringer")
|
||||||
|
} else {
|
||||||
|
fmt.Println("42 implements fmt.Stringer")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO(mpl): restore when fixed
|
||||||
|
// var tt interface{}
|
||||||
|
var tt DummyStringer
|
||||||
|
tt = TestStruct{}
|
||||||
|
ss, ok := tt.(fmt.Stringer)
|
||||||
|
if !ok {
|
||||||
|
fmt.Println("TestStuct does not implement fmt.Stringer")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
fmt.Println(ss.String())
|
||||||
|
fmt.Println(tt.(fmt.Stringer).String())
|
||||||
|
|
||||||
|
if _, ok := tt.(fmt.Stringer); !ok {
|
||||||
|
fmt.Println("TestStuct does not implement fmt.Stringer")
|
||||||
|
return
|
||||||
|
} else {
|
||||||
|
fmt.Println("TestStuct implements fmt.Stringer")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Output:
|
||||||
|
// 1ns
|
||||||
|
// 1ns
|
||||||
|
// true
|
||||||
|
// time.Nanosecond implements fmt.Stringer
|
||||||
|
// 42 does not implement fmt.Stringer
|
||||||
|
// 42 does not implement fmt.Stringer
|
||||||
|
// hello world
|
||||||
|
// hello world
|
||||||
|
// TestStuct implements fmt.Stringer
|
||||||
@@ -4,10 +4,16 @@ func f(a []int) interface{} {
|
|||||||
return cap(a)
|
return cap(a)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func g(a []int) int {
|
||||||
|
return cap(a)
|
||||||
|
}
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
a := []int{1, 2}
|
a := []int{1, 2}
|
||||||
|
println(g(a))
|
||||||
println(f(a).(int))
|
println(f(a).(int))
|
||||||
}
|
}
|
||||||
|
|
||||||
// Output:
|
// Output:
|
||||||
// 2
|
// 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
|
||||||
15
_test/composite14.go
Normal file
15
_test/composite14.go
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import "fmt"
|
||||||
|
|
||||||
|
type T struct {
|
||||||
|
b []byte
|
||||||
|
}
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
t := T{nil}
|
||||||
|
fmt.Println(t)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Output:
|
||||||
|
// {[]}
|
||||||
52
_test/composite15.go
Normal file
52
_test/composite15.go
Normal file
@@ -0,0 +1,52 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
)
|
||||||
|
|
||||||
|
func interfaceAsInts() {
|
||||||
|
var a interface{}
|
||||||
|
b := 2
|
||||||
|
c := 3
|
||||||
|
a = []int{b, c}
|
||||||
|
|
||||||
|
d, ok := a.([]int)
|
||||||
|
if !ok {
|
||||||
|
println("nope")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, v := range d {
|
||||||
|
fmt.Println(v)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func interfaceAsInterfaces() {
|
||||||
|
var a, b, c interface{}
|
||||||
|
b = 2
|
||||||
|
c = 3
|
||||||
|
a = []interface{}{b, c}
|
||||||
|
|
||||||
|
d, ok := a.([]interface{})
|
||||||
|
if !ok {
|
||||||
|
println("nope")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
fmt.Println(d)
|
||||||
|
|
||||||
|
for _, v := range d {
|
||||||
|
fmt.Println(v)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
interfaceAsInts()
|
||||||
|
interfaceAsInterfaces()
|
||||||
|
}
|
||||||
|
|
||||||
|
// Output:
|
||||||
|
// 2
|
||||||
|
// 3
|
||||||
|
// [2 3]
|
||||||
|
// 2
|
||||||
|
// 3
|
||||||
16
_test/composite16.go
Normal file
16
_test/composite16.go
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"net/url"
|
||||||
|
)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
body := url.Values{
|
||||||
|
"Action": {"none"},
|
||||||
|
}
|
||||||
|
fmt.Println(body)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Output:
|
||||||
|
// map[Action:[none]]
|
||||||
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)
|
||||||
|
}
|
||||||
21
_test/convert0.go
Normal file
21
_test/convert0.go
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
type T struct {
|
||||||
|
v int
|
||||||
|
}
|
||||||
|
|
||||||
|
type comparator func(T, T) bool
|
||||||
|
|
||||||
|
func sort(items []T, comp comparator) {
|
||||||
|
println("in sort")
|
||||||
|
}
|
||||||
|
|
||||||
|
func compT(t0, t1 T) bool { return t0.v < t1.v }
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
a := []T{}
|
||||||
|
sort(a, comparator(compT))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Output:
|
||||||
|
// in sort
|
||||||
17
_test/convert1.go
Normal file
17
_test/convert1.go
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import "strconv"
|
||||||
|
|
||||||
|
type atoidef func(s string) (int, error)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
stdatoi := atoidef(strconv.Atoi)
|
||||||
|
n, err := stdatoi("7")
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
println(n)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Output:
|
||||||
|
// 7
|
||||||
19
_test/convert2.go
Normal file
19
_test/convert2.go
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import "bufio"
|
||||||
|
|
||||||
|
func fakeSplitFunc(data []byte, atEOF bool) (advance int, token []byte, err error) {
|
||||||
|
return 7, nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
splitfunc := bufio.SplitFunc(fakeSplitFunc)
|
||||||
|
n, _, err := splitfunc(nil, true)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
println(n)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Output:
|
||||||
|
// 7
|
||||||
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
|
||||||
16
_test/for15.go
Normal file
16
_test/for15.go
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
func f() int { println("in f"); return 1 }
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
for i := f(); ; {
|
||||||
|
println("in loop")
|
||||||
|
if i > 0 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Output:
|
||||||
|
// in f
|
||||||
|
// in loop
|
||||||
16
_test/for16.go
Normal file
16
_test/for16.go
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
max := 1
|
||||||
|
for ; ; max-- {
|
||||||
|
if max == 0 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
println("in for")
|
||||||
|
}
|
||||||
|
println("bye")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Output:
|
||||||
|
// in for
|
||||||
|
// bye
|
||||||
@@ -2,10 +2,15 @@ package main
|
|||||||
|
|
||||||
func f1(a int) interface{} { return a + 1 }
|
func f1(a int) interface{} { return a + 1 }
|
||||||
|
|
||||||
|
func f2(a int64) interface{} { return a + 1 }
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
c := f1(3)
|
c := f1(3)
|
||||||
println(c.(int))
|
println(c.(int))
|
||||||
|
b := f2(3)
|
||||||
|
println(b.(int64))
|
||||||
}
|
}
|
||||||
|
|
||||||
// Output:
|
// Output:
|
||||||
// 4
|
// 4
|
||||||
|
// 4
|
||||||
|
|||||||
@@ -2,7 +2,13 @@ package main
|
|||||||
|
|
||||||
func f1(a int) int { return a + 1 }
|
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() {
|
func main() {
|
||||||
c := f2(3)
|
c := f2(3)
|
||||||
|
|||||||
17
_test/fun26.go
Normal file
17
_test/fun26.go
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
type F func() (int, error)
|
||||||
|
|
||||||
|
func f1() (int, error) { return 3, nil }
|
||||||
|
|
||||||
|
func f2(a string, f F) {
|
||||||
|
c, _ := f()
|
||||||
|
println(a, c)
|
||||||
|
}
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
f2("hello", F(f1))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Output:
|
||||||
|
// hello 3
|
||||||
12
_test/goto1.go
Normal file
12
_test/goto1.go
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
if true {
|
||||||
|
goto here
|
||||||
|
}
|
||||||
|
here:
|
||||||
|
println("ok")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Output:
|
||||||
|
// ok
|
||||||
32
_test/interface47.go
Normal file
32
_test/interface47.go
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
type Doer interface {
|
||||||
|
Do() error
|
||||||
|
}
|
||||||
|
|
||||||
|
type T struct {
|
||||||
|
Name string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *T) Do() error { println("in do"); return nil }
|
||||||
|
|
||||||
|
func f() (Doer, error) { return &T{"truc"}, nil }
|
||||||
|
|
||||||
|
type Ev struct {
|
||||||
|
doer func() (Doer, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (e *Ev) do() {
|
||||||
|
d, _ := e.doer()
|
||||||
|
d.Do()
|
||||||
|
}
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
e := &Ev{f}
|
||||||
|
println(e != nil)
|
||||||
|
e.do()
|
||||||
|
}
|
||||||
|
|
||||||
|
// Output:
|
||||||
|
// true
|
||||||
|
// in do
|
||||||
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]
|
||||||
15
_test/issue-993.go
Normal file
15
_test/issue-993.go
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
var m map[string]int64
|
||||||
|
|
||||||
|
func initVar() {
|
||||||
|
m = make(map[string]int64)
|
||||||
|
}
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
initVar()
|
||||||
|
println(len(m))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Output:
|
||||||
|
// 0
|
||||||
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
|
Hello() string
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type Hey interface {
|
||||||
|
Hello() string
|
||||||
|
}
|
||||||
|
|
||||||
func (r *Root) Hello() string { return "Hello " + r.Name }
|
func (r *Root) Hello() string { return "Hello " + r.Name }
|
||||||
|
|
||||||
func main() {
|
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())
|
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/nil3.go
Normal file
28
_test/nil3.go
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
type I interface {
|
||||||
|
Hello()
|
||||||
|
}
|
||||||
|
|
||||||
|
type T struct {
|
||||||
|
h I
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *T) Hello() { println("Hello") }
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
t := &T{}
|
||||||
|
println(t.h != nil)
|
||||||
|
println(t.h == nil)
|
||||||
|
t.h = t
|
||||||
|
println(t.h != nil)
|
||||||
|
println(t.h == nil)
|
||||||
|
t.h.Hello()
|
||||||
|
}
|
||||||
|
|
||||||
|
// Output:
|
||||||
|
// false
|
||||||
|
// true
|
||||||
|
// true
|
||||||
|
// false
|
||||||
|
// 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 (
|
const (
|
||||||
period = 100 * time.Millisecond
|
period = 100 * time.Millisecond
|
||||||
precision = 5 * time.Millisecond
|
precision = 7 * time.Millisecond
|
||||||
)
|
)
|
||||||
|
|
||||||
func main() {
|
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() {
|
func main() {
|
||||||
var a interface{}
|
var a interface{}
|
||||||
|
a = []int{3}
|
||||||
switch a.(type) {
|
switch a.(type) {
|
||||||
case []int:
|
case []int:
|
||||||
|
println("a is []int")
|
||||||
case []string:
|
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")
|
println("bye")
|
||||||
}
|
}
|
||||||
|
|
||||||
// Output:
|
// Output:
|
||||||
|
// a is []int
|
||||||
|
// b is []string
|
||||||
// bye
|
// bye
|
||||||
|
|||||||
20
_test/time14.go
Normal file
20
_test/time14.go
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
var t time.Time
|
||||||
|
|
||||||
|
func f() time.Time {
|
||||||
|
time := t
|
||||||
|
return time
|
||||||
|
}
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
fmt.Println(f())
|
||||||
|
}
|
||||||
|
|
||||||
|
// Output:
|
||||||
|
// 0001-01-01 00:00:00 +0000 UTC
|
||||||
15
_test/time15.go
Normal file
15
_test/time15.go
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import "time"
|
||||||
|
|
||||||
|
type TimeValue time.Time
|
||||||
|
|
||||||
|
func (v *TimeValue) decode() { println("in decode") }
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
var tv TimeValue
|
||||||
|
tv.decode()
|
||||||
|
}
|
||||||
|
|
||||||
|
// Output:
|
||||||
|
// in decode
|
||||||
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() {
|
func main() {
|
||||||
size := unsafe.Sizeof(S{})
|
x := S{}
|
||||||
align := unsafe.Alignof(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:
|
// Output:
|
||||||
// 24 8
|
// 24 8 16
|
||||||
|
|||||||
15
_test/var15.go
Normal file
15
_test/var15.go
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
var a int = 2
|
||||||
|
|
||||||
|
func inca() {
|
||||||
|
a = a + 1
|
||||||
|
}
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
inca()
|
||||||
|
println(a)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Output:
|
||||||
|
// 3
|
||||||
@@ -7,7 +7,7 @@ import (
|
|||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"os"
|
"os"
|
||||||
"path"
|
"path/filepath"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"github.com/traefik/yaegi/extract"
|
"github.com/traefik/yaegi/extract"
|
||||||
@@ -50,7 +50,7 @@ func extractCmd(arg []string) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if name == "" {
|
if name == "" {
|
||||||
name = path.Base(wd)
|
name = filepath.Base(wd)
|
||||||
}
|
}
|
||||||
ext := extract.Extractor{
|
ext := extract.Extractor{
|
||||||
Dest: name,
|
Dest: name,
|
||||||
@@ -102,7 +102,7 @@ func genLicense(fname string) (string, error) {
|
|||||||
|
|
||||||
f, err := os.Open(fname)
|
f, err := os.Open(fname)
|
||||||
if err != nil {
|
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() }()
|
defer func() { _ = f.Close() }()
|
||||||
|
|
||||||
@@ -116,7 +116,7 @@ func genLicense(fname string) (string, error) {
|
|||||||
license.WriteString("//" + txt + "\n")
|
license.WriteString("//" + txt + "\n")
|
||||||
}
|
}
|
||||||
if sc.Err() != nil {
|
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
|
return license.String(), nil
|
||||||
|
|||||||
@@ -6,6 +6,7 @@ import (
|
|||||||
"go/build"
|
"go/build"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"os"
|
"os"
|
||||||
|
"reflect"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
@@ -18,6 +19,7 @@ import (
|
|||||||
|
|
||||||
func run(arg []string) error {
|
func run(arg []string) error {
|
||||||
var interactive bool
|
var interactive bool
|
||||||
|
var noAutoImport bool
|
||||||
var tags string
|
var tags string
|
||||||
var cmd string
|
var cmd string
|
||||||
var err error
|
var err error
|
||||||
@@ -33,6 +35,7 @@ func run(arg []string) error {
|
|||||||
rflag.BoolVar(&useUnrestricted, "unrestricted", useUnrestricted, "include unrestricted symbols")
|
rflag.BoolVar(&useUnrestricted, "unrestricted", useUnrestricted, "include unrestricted symbols")
|
||||||
rflag.StringVar(&tags, "tags", "", "set a list of build tags")
|
rflag.StringVar(&tags, "tags", "", "set a list of build tags")
|
||||||
rflag.BoolVar(&useUnsafe, "unsafe", useUnsafe, "include unsafe symbols")
|
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.StringVar(&cmd, "e", "", "set the command to be executed (instead of script or/and shell)")
|
||||||
rflag.Usage = func() {
|
rflag.Usage = func() {
|
||||||
fmt.Println("Usage: yaegi run [options] [path] [args]")
|
fmt.Println("Usage: yaegi run [options] [path] [args]")
|
||||||
@@ -45,48 +48,68 @@ func run(arg []string) error {
|
|||||||
args := rflag.Args()
|
args := rflag.Args()
|
||||||
|
|
||||||
i := interp.New(interp.Options{GoPath: build.Default.GOPATH, BuildTags: strings.Split(tags, ",")})
|
i := interp.New(interp.Options{GoPath: build.Default.GOPATH, BuildTags: strings.Split(tags, ",")})
|
||||||
i.Use(stdlib.Symbols)
|
if err := i.Use(stdlib.Symbols); err != nil {
|
||||||
i.Use(interp.Symbols)
|
return err
|
||||||
|
}
|
||||||
|
if err := i.Use(interp.Symbols); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
if useSyscall {
|
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.
|
// Using a environment var allows a nested interpreter to import the syscall package.
|
||||||
if err := os.Setenv("YAEGI_SYSCALL", "1"); err != nil {
|
if err := os.Setenv("YAEGI_SYSCALL", "1"); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if useUnsafe {
|
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 {
|
if err := os.Setenv("YAEGI_UNSAFE", "1"); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if useUnrestricted {
|
if useUnrestricted {
|
||||||
// Use of unrestricted symbols should always follow stdlib and syscall symbols, to update them.
|
// 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 {
|
if err := os.Setenv("YAEGI_UNRESTRICTED", "1"); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if cmd != "" {
|
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 len(args) == 0 {
|
||||||
if interactive || cmd == "" {
|
if cmd == "" || interactive {
|
||||||
showError(err)
|
showError(err)
|
||||||
|
if !noAutoImport {
|
||||||
|
i.ImportUsed()
|
||||||
|
}
|
||||||
_, err = i.REPL()
|
_, err = i.REPL()
|
||||||
}
|
}
|
||||||
return err
|
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]
|
path := args[0]
|
||||||
os.Args = arg
|
os.Args = arg
|
||||||
flag.CommandLine = flag.NewFlagSet(path, flag.ExitOnError)
|
flag.CommandLine = flag.NewFlagSet(path, flag.ExitOnError)
|
||||||
|
|
||||||
if isFile(path) {
|
if isFile(path) {
|
||||||
err = runFile(i, path)
|
err = runFile(i, path, noAutoImport)
|
||||||
} else {
|
} else {
|
||||||
_, err = i.EvalPath(path)
|
_, err = i.EvalPath(path)
|
||||||
}
|
}
|
||||||
@@ -106,7 +129,7 @@ func isFile(path string) bool {
|
|||||||
return err == nil && fi.Mode().IsRegular()
|
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)
|
b, err := ioutil.ReadFile(path)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
@@ -115,6 +138,9 @@ func runFile(i *interp.Interpreter, path string) error {
|
|||||||
if s := string(b); strings.HasPrefix(s, "#!") {
|
if s := string(b); strings.HasPrefix(s, "#!") {
|
||||||
// Allow executable go scripts, Have the same behavior as in interactive mode.
|
// Allow executable go scripts, Have the same behavior as in interactive mode.
|
||||||
s = strings.Replace(s, "#!", "//", 1)
|
s = strings.Replace(s, "#!", "//", 1)
|
||||||
|
if !noAutoImport {
|
||||||
|
i.ImportUsed()
|
||||||
|
}
|
||||||
_, err = i.Eval(s)
|
_, err = i.Eval(s)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,6 +6,7 @@ import (
|
|||||||
"fmt"
|
"fmt"
|
||||||
"go/build"
|
"go/build"
|
||||||
"os"
|
"os"
|
||||||
|
"path/filepath"
|
||||||
"regexp"
|
"regexp"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
@@ -102,25 +103,47 @@ func test(arg []string) (err error) {
|
|||||||
testing.Init()
|
testing.Init()
|
||||||
os.Args = tf
|
os.Args = tf
|
||||||
flag.Parse()
|
flag.Parse()
|
||||||
|
path += string(filepath.Separator)
|
||||||
|
var dir string
|
||||||
|
|
||||||
|
switch strings.Split(path, string(filepath.Separator))[0] {
|
||||||
|
case ".", "..", string(filepath.Separator):
|
||||||
|
dir = path
|
||||||
|
default:
|
||||||
|
dir = filepath.Join(build.Default.GOPATH, "src", path)
|
||||||
|
}
|
||||||
|
if err = os.Chdir(dir); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
i := interp.New(interp.Options{GoPath: build.Default.GOPATH, BuildTags: strings.Split(tags, ",")})
|
i := interp.New(interp.Options{GoPath: build.Default.GOPATH, BuildTags: strings.Split(tags, ",")})
|
||||||
i.Use(stdlib.Symbols)
|
if err := i.Use(stdlib.Symbols); err != nil {
|
||||||
i.Use(interp.Symbols)
|
return err
|
||||||
|
}
|
||||||
|
if err := i.Use(interp.Symbols); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
if useSyscall {
|
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.
|
// Using a environment var allows a nested interpreter to import the syscall package.
|
||||||
if err := os.Setenv("YAEGI_SYSCALL", "1"); err != nil {
|
if err := os.Setenv("YAEGI_SYSCALL", "1"); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if useUnrestricted {
|
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 {
|
if err := os.Setenv("YAEGI_UNRESTRICTED", "1"); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if useUnsafe {
|
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 {
|
if err := os.Setenv("YAEGI_UNSAFE", "1"); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -50,9 +50,9 @@ func TestYaegiCmdCancel(t *testing.T) {
|
|||||||
|
|
||||||
yaegi := filepath.Join(tmp, "yaegi")
|
yaegi := filepath.Join(tmp, "yaegi")
|
||||||
build := exec.Command("go", "build", "-race", "-o", yaegi, ".")
|
build := exec.Command("go", "build", "-race", "-o", yaegi, ".")
|
||||||
err = build.Run()
|
out, err := build.CombinedOutput()
|
||||||
if err != nil {
|
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.
|
// Test src must be terminated by a single newline.
|
||||||
@@ -82,7 +82,7 @@ func TestYaegiCmdCancel(t *testing.T) {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
t.Errorf("failed pipe test source to yaegi command: %v", err)
|
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)
|
err = cmd.Process.Signal(os.Interrupt)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Errorf("failed to send os.Interrupt to yaegi command: %v", err)
|
t.Errorf("failed to send os.Interrupt to yaegi command: %v", err)
|
||||||
|
|||||||
@@ -9,7 +9,9 @@ import (
|
|||||||
|
|
||||||
func TestFunctionCall(t *testing.T) {
|
func TestFunctionCall(t *testing.T) {
|
||||||
i := interp.New(interp.Options{GoPath: "./_pkg"})
|
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"`)
|
_, err := i.Eval(`import "foo/bar"`)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user