Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

V2 merge #1907

Merged
merged 88 commits into from
Oct 19, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
88 commits
Select commit Hold shift + click to select a range
3c5e486
Update README_zh-CN.md (#1545)
tzxdtc Apr 11, 2023
01d20cc
Add option to set template delimiters (#1499)
leosunmo Apr 17, 2023
c14cc8e
fix bug: enums of explicit type conversion (#1556)
sdghchj Apr 17, 2023
21d34e2
add retract to fix proxy cache caused by accidentally pushed tags (#1…
Nerzal Apr 19, 2023
5774b7b
docs: doc to pt Add option to set template delims. (#1563)
Paulo-Lopes-Estevao May 3, 2023
e749ad5
fix: lint error for generated docs.go (#1583)
lowang-bh May 10, 2023
ea35767
fix bug: enums of underscored number (#1581)
sdghchj May 12, 2023
e73a0d0
fix using tab (\t) as separator for custom type names (#1594)
nitram509 Jun 7, 2023
b2f325f
chore(deps): bump github.com/gin-gonic/gin (#1598)
dependabot[bot] Jun 25, 2023
8e5b314
chore(deps): bump github.com/gin-gonic/gin in /example/celler (#1599)
dependabot[bot] Jun 25, 2023
c8372f6
chore(deps): bump github.com/gin-gonic/gin in /example/go-module-supp…
dependabot[bot] Jun 25, 2023
0cee1c5
fix required params parsing for routes with multiple paths and multip…
Phenix66 Jul 7, 2023
fe971d2
parser: if all tags negate return true on no hits (#1624)
rsmarples Jul 17, 2023
4536bf2
fix: enums in body got parse incorrectly (#1625)
hohobilly Jul 18, 2023
575963e
parse binary literal const (#1593)
sdghchj Jul 18, 2023
1bf0078
feat: global security (#1620)
nameoffnv Jul 18, 2023
7534a13
add cli flag --pdl to determine whether parse operations in dependenc…
sdghchj Jul 18, 2023
d0f9dc5
feat: add --packagePrefix=P for only parse packages matched by prefix…
SilverRainZ Jul 19, 2023
27b27bd
enchancement: report which property is triggering a parsing error (#1…
sakishrist Jul 20, 2023
f05ccdc
add byte check before and after file is formatted (#1637)
hohobilly Aug 9, 2023
9f128b4
feat: preserve file permission when write formatted files (#1636)
hohobilly Aug 9, 2023
8ebf32f
docs(readme): fix param brace (#1647)
danielmoncada Aug 15, 2023
23c9b5c
chore(deps): bump gopkg.in/yaml.v3 (#1663)
dependabot[bot] Aug 30, 2023
e9d0aa5
yaml.v3 security patch (#1664)
ubogdan Aug 30, 2023
1460377
test: remove redundant `filepath.Clean` call (#1675)
wholesome-ghoul Sep 25, 2023
2e5d352
chore(deps): bump golang.org/x/net from 0.8.0 to 0.17.0 (#1686)
dependabot[bot] Oct 12, 2023
d23a84a
chore(deps): bump golang.org/x/net in /example/markdown (#1685)
dependabot[bot] Oct 12, 2023
2da9651
When the return value defined by the @Success tag is equal to a null …
Shimizu1111 Oct 12, 2023
d7e961a
chore(deps): bump golang.org/x/net in /example/go-module-support (#1682)
dependabot[bot] Oct 12, 2023
c00103a
chore(deps): bump golang.org/x/net in /example/object-map-example (#1…
dependabot[bot] Oct 12, 2023
2b5852a
chore(deps): bump golang.org/x/net in /example/celler (#1683)
dependabot[bot] Oct 12, 2023
bab7aac
docs: add PT and EN examples for Go generic types (#1697)
rpedrodasilva10 Nov 7, 2023
19b9c00
Update README.md (#1698)
saurabhchatterjee23 Nov 7, 2023
c7c63fc
update gopkg.in/yaml.v3 v3.0.0-20200615113413-eeeca48fe776 to 3.0.0 (…
chncaption Nov 7, 2023
0ade78c
improve docker container usage (#1704)
ngehrsitz Nov 16, 2023
6cdaaf5
fix issue #1662: find definitions from external packages first (#1666)
sdghchj Nov 24, 2023
744a58e
Drop support for go v1.17.x (#1723)
sdghchj Dec 18, 2023
7603121
Add flag state #1628 (#1629)
ivolkoff Dec 18, 2023
9fdba3e
fix deps (#1724)
sdghchj Dec 19, 2023
43fec8e
chore(deps): bump golang.org/x/crypto in /example/celler (#1727)
dependabot[bot] Dec 19, 2023
fb5890c
chore(deps): bump golang.org/x/crypto in /example/go-module-support (…
dependabot[bot] Dec 19, 2023
33da992
chore(deps): bump golang.org/x/crypto in /example/object-map-example …
dependabot[bot] Dec 20, 2023
0fb6820
deprecate some parts of routers in an operation (#1735)
sdghchj Jan 11, 2024
76695ca
bug: array form filed name should not contains bracket which led to i…
Jinof Jan 16, 2024
d4218f2
Struct fields supported for header and path param types (#1740)
tanenbaum Jan 22, 2024
ae7e404
fix #1742 (#1744)
sdghchj Jan 22, 2024
7147984
Feat: Support generic with map params (#1746)
sdghchj Jan 29, 2024
98ed434
Update version.go (#1751)
sdghchj Feb 1, 2024
56fde5c
Update operation.go (#1753)
mathieu-chauvet Feb 6, 2024
928264c
fix: remove dropped tags from general infos (#1764)
sdghchj Feb 20, 2024
87e7d9c
Update docker go build version to 1.21 (#1758)
ngehrsitz Feb 20, 2024
91624ad
add support for "title" tag (#1762)
matteobassan Feb 20, 2024
90aa46f
chore: fix some typos in comments (#1788)
camcui Apr 20, 2024
8a47dcb
bump go version (#1797)
ubogdan Apr 20, 2024
0834357
chore(deps): bump golang.org/x/net from 0.17.0 to 0.23.0 (#1793)
dependabot[bot] Apr 20, 2024
d5af957
chore(deps): bump golang.org/x/net in /example/markdown (#1792)
dependabot[bot] Apr 20, 2024
1bb1445
chore(deps): bump golang.org/x/net in /example/celler (#1794)
dependabot[bot] Apr 20, 2024
6aa6613
chore(deps): bump golang.org/x/net in /example/go-module-support (#1795)
dependabot[bot] Apr 20, 2024
0368d7d
chore(deps): bump golang.org/x/net in /example/object-map-example (#1…
dependabot[bot] Apr 20, 2024
4c2f8dd
Handle case of empty GOROOT (#1798)
evan-goode Apr 21, 2024
85254b4
Added multiline support for @description attribute for securityDefini…
Ponywka Apr 22, 2024
15dae35
Feat: multi-arch docker image (#1756)
tnaroska Apr 23, 2024
b8662de
chore(deps): bump google.golang.org/protobuf (#1773)
dependabot[bot] Apr 23, 2024
4a11e23
chore(deps): bump google.golang.org/protobuf (#1774)
dependabot[bot] Apr 23, 2024
937c239
chore(deps): bump google.golang.org/protobuf in /example/celler (#1775)
dependabot[bot] Apr 23, 2024
fd2fa83
fix issue: #1780: filter $GOROOT path (#1827)
bobsongplus Jun 21, 2024
e55c557
feat: read from stdin, write to stdout (#1831) (#1832)
bfbonatto Jun 21, 2024
7204462
Added suport for parsing comments inside of function bodies (#1824)
j-d-ha Jun 21, 2024
f32d4d3
adds support for complex types with function scope (#1813)
KristofferFJ Jun 28, 2024
807dd1f
[Issue 1812] fix misalignment in expected.json and api.go messing wit…
KristofferFJ Jul 1, 2024
697572a
Fixes Issue 1829 (#1830)
Kafkalasch Jul 1, 2024
ff50cd6
Fix global overrides for any/interface ref types (#1835)
ezequiel Jul 3, 2024
c7f1cd8
adds support for pointer function scoped fields (#1841)
KristofferFJ Jul 11, 2024
10030b0
fix parse nested structs and aliases (#1866)
zdon0 Aug 20, 2024
83fe3ca
Fix generics used with function scoped types (#1883)
berk-karaal Sep 5, 2024
1d730c5
Fix param comment escaping issue (#1890)
yukiomoto Sep 20, 2024
a3c6d12
support markdown description for declaration (#1893)
nicoxb Sep 25, 2024
9069105
update README (#1856)
sdghchj Oct 17, 2024
4fd8a36
Update docs for request and response headers (#1825)
eksrha Oct 17, 2024
a74d34c
fix:parse all field names declared in a row (#1872)
sdghchj Oct 17, 2024
28de14c
Flags to parse internal and dependency package (#1894)
bytesByHarsh Oct 17, 2024
7159b0f
fix: failing assert in enums test on 32bit (#1634)
leso-kn Oct 17, 2024
d323b48
Feat: Add support for parenthesis in router patterns (#1859)
alifemove Oct 17, 2024
103ac42
chore: Update ci.yml (#1902)
ubogdan Oct 17, 2024
0b9e347
new release (#1901)
ubogdan Oct 18, 2024
5b930d4
Merge branch 'v2' into v2-merge
ubogdan Oct 18, 2024
36c14a4
fix some issues
ubogdan Oct 18, 2024
6106783
fix unit tests
ubogdan Oct 19, 2024
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 2 additions & 3 deletions .github/workflows/ci.yml
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@ jobs:
test:
strategy:
matrix:
go: [ '1.18.x', '1.19.x', '1.20.x' ]
go: [ '1.19.x', '1.20.x', '1.21.x', '1.22.x' ]
platform: [ubuntu-latest, macos-latest]
runs-on: ${{ matrix.platform }}
steps:
Expand All @@ -22,8 +22,7 @@ jobs:
- name: deps
run: make deps
- name: static program analysis
run: |
make fmt-check lint vet
run: make fmt-check vet
- name: build
run: make build
- name: test
Expand Down
20 changes: 13 additions & 7 deletions .github/workflows/docker.yml
Original file line number Diff line number Diff line change
Expand Up @@ -5,38 +5,44 @@ on:
tags:
- 'v*'

permissions:
contents: read
packages: write

jobs:
docker-build:
runs-on: ubuntu-latest
steps:
- name: Checkout code
uses: actions/checkout@v3
uses: actions/checkout@v4

- name: Set up Docker Buildx
id: buildx
uses: docker/setup-buildx-action@v2
uses: docker/setup-buildx-action@v3

- name: Login to Github Packages
uses: docker/login-action@v2
uses: docker/login-action@v3
with:
registry: ghcr.io
username: ${{ github.actor }}
password: ${{ secrets.GITHUB_TOKEN }}

- name: Docker meta
id: meta
uses: docker/metadata-action@v4
uses: docker/metadata-action@v5
with:
images: ghcr.io/swaggo/swag

- name: Build image and push to GitHub Container Registry
uses: docker/build-push-action@v2
id: docker_build
uses: docker/build-push-action@v5
with:
context: .
platforms: linux/amd64,linux/arm64
push: true
tags: |
ghcr.io/swaggo/swag:latest
ghcr.io/swaggo/swag:${{github.ref_name}}
ghcr.io/${{ github.repository }}:latest
ghcr.io/${{ github.repository }}:${{github.ref_name}}
labels: ${{ steps.meta.outputs.labels }}

- name: Image digest
Expand Down
16 changes: 12 additions & 4 deletions Dockerfile
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
# Dockerfile References: https://docs.docker.com/engine/reference/builder/

# Start from the latest golang base image
FROM golang:1.18.3-alpine as builder
FROM --platform=$BUILDPLATFORM golang:1.21-alpine as builder

# Set the Current Working Directory inside the container
WORKDIR /app
Expand All @@ -15,14 +15,22 @@ RUN go mod download
# Copy the source from the current directory to the Working Directory inside the container
COPY . .

# Configure go compiler target platform
ARG TARGETOS
ARG TARGETARCH
ENV GOARCH=$TARGETARCH \
GOOS=$TARGETOS

# Build the Go app
RUN CGO_ENABLED=0 GOOS=linux go build -v -a -installsuffix cgo -o swag cmd/swag/main.go


######## Start a new stage from scratch #######
FROM scratch
FROM --platform=$TARGETPLATFORM scratch

WORKDIR /root/
WORKDIR /code/

# Copy the Pre-built binary file from the previous stage
COPY --from=builder /app/swag .
COPY --from=builder /app/swag /bin/swag

ENTRYPOINT ["/bin/swag"]
22 changes: 3 additions & 19 deletions Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@ GOBUILD:=$(GOCMD) build
GOINSTALL:=$(GOCMD) install
GOCLEAN:=$(GOCMD) clean
GOTEST:=$(GOCMD) test
GOMODTIDY:=$(GOCMD) mod tidy
GOGET:=$(GOCMD) get
GOLIST:=$(GOCMD) list
GOVET:=$(GOCMD) vet
Expand All @@ -16,8 +17,6 @@ BINARY_NAME:=swag
PACKAGES:=$(shell $(GOLIST) github.com/swaggo/swag/v2 github.com/swaggo/swag/v2/cmd/swag github.com/swaggo/swag/v2/gen github.com/swaggo/swag/v2/format)
GOFILES:=$(shell find . -name "*.go" -type f)

export GO111MODULE := on

all: test build

.PHONY: build
Expand Down Expand Up @@ -54,25 +53,10 @@ clean:

.PHONY: deps
deps:
$(GOGET) github.com/swaggo/cli
$(GOGET) sigs.k8s.io/yaml
$(GOGET) github.com/KyleBanks/depth
$(GOGET) github.com/go-openapi/jsonreference
$(GOGET) github.com/go-openapi/spec
$(GOGET) github.com/stretchr/testify/assert
$(GOGET) golang.org/x/tools/go/loader

.PHONY: devel-deps
devel-deps:
GO111MODULE=off $(GOGET) -v -u \
golang.org/x/lint/golint

.PHONY: lint
lint: devel-deps
for PKG in $(PACKAGES); do golint -set_exit_status $$PKG || exit 1; done;
$(GOMODTIDY)

.PHONY: vet
vet: deps devel-deps
vet: deps
$(GOVET) $(PACKAGES)

.PHONY: fmt
Expand Down
93 changes: 70 additions & 23 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -42,7 +42,8 @@ Swag converts Go annotations to Swagger Documentation 2.0. We've created a varie
- [User defined structure with an array type](#user-defined-structure-with-an-array-type)
- [Function scoped struct declaration](#function-scoped-struct-declaration)
- [Model composition in response](#model-composition-in-response)
- [Add a headers in response](#add-a-headers-in-response)
- [Add headers in request](#add-request-headers)
- [Add headers in response](#add-a-headers-in-response)
- [Use multiple path params](#use-multiple-path-params)
- [Add multiple paths](#add-multiple-paths)
- [Example value of struct](#example-value-of-struct)
Expand All @@ -56,6 +57,7 @@ Swag converts Go annotations to Swagger Documentation 2.0. We've created a varie
- [How to use security annotations](#how-to-use-security-annotations)
- [Add a description for enum items](#add-a-description-for-enum-items)
- [Generate only specific docs file types](#generate-only-specific-docs-file-types)
- [How to use Go generic types](#how-to-use-generics)
- [Change the default Go Template action delimiters](#change-the-default-go-template-action-delimiters)
- [About the Project](#about-the-project)
- [Contributors](#contributors)
Expand All @@ -67,11 +69,16 @@ Swag converts Go annotations to Swagger Documentation 2.0. We've created a varie

1. Add comments to your API source code, See [Declarative Comments Format](#declarative-comments-format).

2. Download swag by using:
2. Install swag by using:
```sh
go install github.com/swaggo/swag/v2/cmd/swag@latest
```
To build from source you need [Go](https://golang.org/dl/) (1.18 or newer).
To build from source you need [Go](https://golang.org/dl/) (1.19 or newer).

Alternatively you can run the docker image:
```sh
docker run --rm -v $(pwd):/code ghcr.io/swaggo/swag:latest
```

Or download a pre-compiled binary from the [release page](https://github.com/swaggo/swag/releases).

Expand All @@ -81,6 +88,9 @@ swag init
```

Make sure to import the generated `docs/docs.go` so that your specific configuration gets `init`'ed. If your General API annotations do not live in `main.go`, you can let swag know with `-g` flag.
```go
import _ "example-module-name/docs"
```
```sh
swag init -g http/api.go
```
Expand Down Expand Up @@ -112,6 +122,7 @@ OPTIONS:
--outputTypes value, --ot value Output types of generated files (docs.go, swagger.json, swagger.yaml) like go,json,yaml (default: "go,json,yaml")
--parseVendor Parse go files in 'vendor' folder, disabled by default (default: false)
--parseDependency, --pd Parse go files inside dependency folder, disabled by default (default: false)
--parseDependencyLevel, --pdl Enhancement of '--parseDependency', parse go files inside dependency folder, 0 disabled, 1 only parse models, 2 only parse operations, 3 parse all (default: 0)
--markdownFiles value, --md value Parse folder containing markdown files to use as description, disabled by default
--codeExampleFiles value, --cef value Parse folder containing code example files to use for the x-codeSamples extension, disabled by default
--parseInternal Parse go files in internal packages, disabled by default (default: false)
Expand All @@ -125,6 +136,9 @@ OPTIONS:
--tags value, -t value A comma-separated list of tags to filter the APIs for which the documentation is generated.Special case if the tag is prefixed with the '!' character then the APIs with that tag will be excluded
--v3.1 Generate OpenAPI V3.1 spec (default: false)
--templateDelims value, --td value Provide custom delimeters for Go template generation. The format is leftDelim,rightDelim. For example: "[[,]]"
--collectionFormat value, --cf value Set default collection format (default: "csv")
--state value Initial state for the state machine (default: ""), @HostState in root file, @State in other files
--parseFuncBody Parse API info within body of functions in go files, disabled by default (default: false)
--packageName --output A package name of docs.go, using output directory name by default (check --output option)
--collectionFormat value, --cf value Set default collection format (default: "csv")
--help, -h show help
Expand Down Expand Up @@ -163,6 +177,7 @@ OPTIONS:

Find the example source code [here](https://github.com/swaggo/swag/tree/master/example/celler).

Finish the steps in [Getting started](#getting-started)
1. After using `swag init` to generate Swagger 2.0 docs, import the following packages:
```go
import "github.com/swaggo/gin-swagger" // gin-swagger middleware
Expand Down Expand Up @@ -443,25 +458,26 @@ The following annotations are only available if you set the -v3.1 flag in the CL
[celler/controller](https://github.com/swaggo/swag/tree/master/example/celler/controller)


| annotation | description |
|-------------|----------------------------------------------------------------------------------------------------------------------------|
| description | A verbose explanation of the operation behavior. |
| description.markdown | A short description of the application. The description will be read from a file. E.g. `@description.markdown details` will load `details.md`| // @description.file endpoint.description.markdown |
| id | A unique string used to identify the operation. Must be unique among all API operations. |
| tags | A list of tags to each API operation that separated by commas. |
| summary | A short summary of what the operation does. |
| accept | A list of MIME types the APIs can consume. Note that Accept only affects operations with a request body, such as POST, PUT and PATCH. Value MUST be as described under [Mime Types](#mime-types). |
| produce | A list of MIME types the APIs can produce. Value MUST be as described under [Mime Types](#mime-types). |
| param | Parameters that separated by spaces. `param name`,`param type`,`data type`,`is mandatory?`,`comment` `attribute(optional)` |
| security | [Security](#security) to each API operation. |
| success | Success response that separated by spaces. `return code or default`,`{param type}`,`data type`,`comment` |
| failure | Failure response that separated by spaces. `return code or default`,`{param type}`,`data type`,`comment` |
| response | As same as `success` and `failure` |
| header | Header in response that separated by spaces. `return code`,`{param type}`,`data type`,`comment` |
| router | Path definition that separated by spaces. `path`,`[httpMethod]` |
| x-name | The extension key, must be start by x- and take only json value. |
| x-codeSample | Optional Markdown usage. take `file` as parameter. This will then search for a file named like the summary in the given folder. |
| deprecated | Mark endpoint as deprecated. |
| annotation | description |
|----------------------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| description | A verbose explanation of the operation behavior. |
| description.markdown | A short description of the application. The description will be read from a file. E.g. `@description.markdown details` will load `details.md` | // @description.file endpoint.description.markdown |
| id | A unique string used to identify the operation. Must be unique among all API operations. |
| tags | A list of tags to each API operation that separated by commas. |
| summary | A short summary of what the operation does. |
| accept | A list of MIME types the APIs can consume. Note that Accept only affects operations with a request body, such as POST, PUT and PATCH. Value MUST be as described under [Mime Types](#mime-types). |
| produce | A list of MIME types the APIs can produce. Value MUST be as described under [Mime Types](#mime-types). |
| param | Parameters that separated by spaces. `param name`,`param type`,`data type`,`is mandatory?`,`comment` `attribute(optional)` |
| security | [Security](#security) to each API operation. |
| success | Success response that separated by spaces. `return code or default`,`{param type}`,`data type`,`comment` |
| failure | Failure response that separated by spaces. `return code or default`,`{param type}`,`data type`,`comment` |
| response | As same as `success` and `failure` |
| header | Header in response that separated by spaces. `return code`,`{param type}`,`data type`,`comment` |
| router | Path definition that separated by spaces. `path`,`[httpMethod]` |
| deprecatedrouter | As same as router, but deprecated. |
| x-name | The extension key, must be start by x- and take only json value. |
| x-codeSample | Optional Markdown usage. take `file` as parameter. This will then search for a file named like the summary in the given folder. |
| deprecated | Mark endpoint as deprecated. |



Expand Down Expand Up @@ -663,7 +679,14 @@ type DeepObject struct { //in `proto` package
}
@success 200 {object} jsonresult.JSONResult{data1=proto.Order{data=proto.DeepObject},data2=[]proto.Order{data=[]proto.DeepObject}} "desc"
```
### Add a headers in response
### Add response request

```go
// @Param X-MyHeader header string true "MyHeader must be set for valid response"
// @Param X-API-VERSION header string true "API version eg.: 1.0"
```

### Add response headers

```go
// @Success 200 {string} string "ok"
Expand Down Expand Up @@ -939,6 +962,19 @@ By default `swag` command generates Swagger specification in three different fil

If you would like to limit a set of file types which should be generated you can use `--outputTypes` (short `-ot`) flag. Default value is `go,json,yaml` - output types separated with comma. To limit output only to `go` and `yaml` files, you would write `go,yaml`. With complete command that would be `swag init --outputTypes go,yaml`.

### How to use Generics

```go
// @Success 200 {object} web.GenericNestedResponse[types.Post]
// @Success 204 {object} web.GenericNestedResponse[types.Post, Types.AnotherOne]
// @Success 201 {object} web.GenericNestedResponse[web.GenericInnerType[types.Post]]
func GetPosts(w http.ResponseWriter, r *http.Request) {
_ = web.GenericNestedResponse[types.Post]{}
}
```
See [this file](https://github.com/swaggo/swag/blob/master/testdata/generics_nested/api/api.go) for more details
and other examples.

### Change the default Go Template action delimiters
[#980](https://github.com/swaggo/swag/issues/980)
[#1177](https://github.com/swaggo/swag/issues/1177)
Expand All @@ -951,6 +987,17 @@ swag init -g http/api.go -td "[[,]]"
```
The new delimiter is a string with the format "`<left delimiter>`,`<right delimiter>`".

### Parse Internal and Dependency Packages

If the struct is defined in a dependency package, use `--parseDependency`.

If the struct is defined in your main project, use `--parseInternal`.

if you want to include both internal and from dependencies use both flags
```
swag init --parseDependency --parseInternal
```

## About the Project
This project was inspired by [yvasiyarov/swagger](https://github.com/yvasiyarov/swagger) but we simplified the usage and added support a variety of [web frameworks](#supported-web-frameworks). Gopher image source is [tenntenn/gopher-stickers](https://github.com/tenntenn/gopher-stickers). It has licenses [creative commons licensing](http://creativecommons.org/licenses/by/3.0/deed.en).
## Contributors
Expand Down
Loading
Loading