├── LICENSE
├── README.md
├── common
└── redis.go
├── config.json
├── go.mod
├── go.sum
├── handler
├── auth.go
└── todo.go
├── main.go
├── models
├── autogenerated_todo.go
├── autogenerated_users.go
├── gen_queryset.sh
├── setup.go
├── todo.go
└── users.go
├── requests
└── todo.go
└── routers
├── routers.go
└── v1
└── router.go
/LICENSE:
--------------------------------------------------------------------------------
1 | MIT License
2 |
3 | Copyright (c) 2020 Eonyak Ko
4 |
5 | Permission is hereby granted, free of charge, to any person obtaining a copy
6 | of this software and associated documentation files (the "Software"), to deal
7 | in the Software without restriction, including without limitation the rights
8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9 | copies of the Software, and to permit persons to whom the Software is
10 | furnished to do so, subject to the following conditions:
11 |
12 | The above copyright notice and this permission notice shall be included in all
13 | copies or substantial portions of the Software.
14 |
15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21 | SOFTWARE.
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 |
2 |
3 | # Boilerplate JWT Auth Server with Go
4 |
5 | [](https://covenant.tistory.com/)
6 | [](https://github.com/brave-people/Dev-Event/pulls)
7 |
8 |
9 |
10 |
11 |
12 |
13 | ## 📚 Tech Stack
14 |
15 | - Go
16 | - Gingonic
17 | - MySQL
18 | - Redis
19 | - Docker
20 | - Travis CI
21 | - Deepsource Go
22 |
23 |
24 | ## 📄 Reference
25 |
26 | - [Github. golang gin realworld example app](https://github.com/gothinkster/golang-gin-realworld-example-app)
27 | - [Using JWT for Authentication in a Golang Application](https://www.nexmo.com/blog/2020/03/13/using-jwt-for-authentication-in-a-golang-application-dr)
28 |
29 |
30 | ## License
31 |
32 |
--------------------------------------------------------------------------------
/common/redis.go:
--------------------------------------------------------------------------------
1 | package common
2 |
3 | import (
4 | "os"
5 |
6 | "github.com/go-redis/redis"
7 | )
8 |
9 | var client *redis.Client
10 |
11 | func RedisInit() {
12 | dsn := os.Getenv("REDIS_DSN")
13 | if len(dsn) == 0 {
14 | dsn = "localhost:6379"
15 | }
16 |
17 | client = redis.NewClient(&redis.Options{
18 | Addr: dsn,
19 | })
20 |
21 | _, err := client.Ping().Result()
22 | if err != nil {
23 | panic(err)
24 | }
25 | }
26 |
27 | func GetClient() *redis.Client {
28 | return client
29 | }
30 |
--------------------------------------------------------------------------------
/config.json:
--------------------------------------------------------------------------------
1 | {
2 | "debug": true,
3 | "server": {
4 | "port": ":8080"
5 | },
6 | "database": {
7 | "host": "docker.for.mac.localhost",
8 | "port": "3306",
9 | "user": "test",
10 | "pass": "1234",
11 | "name": "users"
12 | },
13 | "token": {
14 | "ACCESS_SECRET": "your_access_key_here",
15 | "REFRESH_SECRET": "your_refrech_key_here"
16 | }
17 | }
--------------------------------------------------------------------------------
/go.mod:
--------------------------------------------------------------------------------
1 | module github.com/golang-crew/Bolierplate-JWT-Auth
2 |
3 | go 1.14
4 |
5 | require (
6 | github.com/dgrijalva/jwt-go v3.2.0+incompatible
7 | github.com/gin-gonic/gin v1.6.3
8 | github.com/go-redis/redis v6.15.9+incompatible
9 | github.com/go-sql-driver/mysql v1.5.0
10 | github.com/jinzhu/gorm v1.9.16
11 | github.com/satori/go.uuid v1.2.0
12 | github.com/spf13/viper v1.7.1
13 | github.com/swaggo/gin-swagger v1.2.0
14 | )
15 |
--------------------------------------------------------------------------------
/go.sum:
--------------------------------------------------------------------------------
1 | cloud.google.com/go v0.26.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw=
2 | cloud.google.com/go v0.34.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw=
3 | cloud.google.com/go v0.38.0/go.mod h1:990N+gfupTy94rShfmMCWGDn0LpTmnzTp2qbd1dvSRU=
4 | cloud.google.com/go v0.44.1/go.mod h1:iSa0KzasP4Uvy3f1mN/7PiObzGgflwredwwASm/v6AU=
5 | cloud.google.com/go v0.44.2/go.mod h1:60680Gw3Yr4ikxnPRS/oxxkBccT6SA1yMk63TGekxKY=
6 | cloud.google.com/go v0.45.1/go.mod h1:RpBamKRgapWJb87xiFSdk4g1CME7QZg3uwTez+TSTjc=
7 | cloud.google.com/go v0.46.3/go.mod h1:a6bKKbmY7er1mI7TEI4lsAkts/mkhTSZK8w33B4RAg0=
8 | cloud.google.com/go/bigquery v1.0.1/go.mod h1:i/xbL2UlR5RvWAURpBYZTtm/cXjCha9lbfbpx4poX+o=
9 | cloud.google.com/go/datastore v1.0.0/go.mod h1:LXYbyblFSglQ5pkeyhO+Qmw7ukd3C+pD7TKLgZqpHYE=
10 | cloud.google.com/go/firestore v1.1.0/go.mod h1:ulACoGHTpvq5r8rxGJ4ddJZBZqakUQqClKRT5SZwBmk=
11 | cloud.google.com/go/pubsub v1.0.1/go.mod h1:R0Gpsv3s54REJCy4fxDixWD93lHJMoZTyQ2kNxGRt3I=
12 | cloud.google.com/go/storage v1.0.0/go.mod h1:IhtSnM/ZTZV8YYJWCY8RULGVqBDmpoyjwiyrjsg+URw=
13 | dmitri.shuralyov.com/gpu/mtl v0.0.0-20190408044501-666a987793e9/go.mod h1:H6x//7gZCb22OMCxBHrMx7a5I7Hp++hsVxbQ4BYO7hU=
14 | github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU=
15 | github.com/BurntSushi/xgb v0.0.0-20160522181843-27f122750802/go.mod h1:IVnqGOEym/WlBOVXweHU+Q+/VP0lqqI8lqeDx9IjBqo=
16 | github.com/OneOfOne/xxhash v1.2.2/go.mod h1:HSdplMjZKSmBqAxg5vPj2TmRDmfkzw+cTzAElWljhcU=
17 | github.com/PuerkitoBio/goquery v1.5.1/go.mod h1:GsLWisAFVj4WgDibEWF4pvYnkVQBpKBKeU+7zCJoLcc=
18 | github.com/PuerkitoBio/purell v1.1.0 h1:rmGxhojJlM0tuKtfdvliR84CFHljx9ag64t2xmVkjK4=
19 | github.com/PuerkitoBio/purell v1.1.0/go.mod h1:c11w/QuzBsJSee3cPx9rAFu61PvFxuPbtSwDGJws/X0=
20 | github.com/PuerkitoBio/urlesc v0.0.0-20170810143723-de5bf2ad4578 h1:d+Bc7a5rLufV/sSk/8dngufqelfh6jnri85riMAaF/M=
21 | github.com/PuerkitoBio/urlesc v0.0.0-20170810143723-de5bf2ad4578/go.mod h1:uGdkoq3SwY9Y+13GIhn11/XLaGBb4BfwItxLd5jeuXE=
22 | github.com/alecthomas/template v0.0.0-20160405071501-a0175ee3bccc/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc=
23 | github.com/alecthomas/units v0.0.0-20151022065526-2efee857e7cf/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0=
24 | github.com/andybalholm/cascadia v1.1.0/go.mod h1:GsXiBklL0woXo1j/WYWtSYYC4ouU9PqHO0sqidkEA4Y=
25 | github.com/armon/circbuf v0.0.0-20150827004946-bbbad097214e/go.mod h1:3U/XgcO3hCbHZ8TKRvWD2dDTCfh9M9ya+I9JpbB7O8o=
26 | github.com/armon/go-metrics v0.0.0-20180917152333-f0300d1749da/go.mod h1:Q73ZrmVTwzkszR9V5SSuryQ31EELlFMUz1kKyl939pY=
27 | github.com/armon/go-radix v0.0.0-20180808171621-7fddfc383310/go.mod h1:ufUuZ+zHj4x4TnLV4JWEpy2hxWSpsRywHrMgIH9cCH8=
28 | github.com/beorn7/perks v0.0.0-20180321164747-3a771d992973/go.mod h1:Dwedo/Wpr24TaqPxmxbtue+5NUziq4I4S80YR8gNf3Q=
29 | github.com/beorn7/perks v1.0.0/go.mod h1:KWe93zE9D1o94FZ5RNwFwVgaQK1VOXiVxmqh+CedLV8=
30 | github.com/bgentry/speakeasy v0.1.0/go.mod h1:+zsyZBPWlz7T6j88CTgSN5bM796AkVf0kBD4zp0CCIs=
31 | github.com/bketelsen/crypt v0.0.3-0.20200106085610-5cbc8cc4026c/go.mod h1:MKsuJmJgSg28kpZDP6UIiPt0e0Oz0kqKNGyRaWEPv84=
32 | github.com/cespare/xxhash v1.1.0/go.mod h1:XrSqR1VqqWfGrhpAt58auRo0WTKS1nRRg3ghfAqPWnc=
33 | github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDkc90ppPyw=
34 | github.com/coreos/bbolt v1.3.2/go.mod h1:iRUV2dpdMOn7Bo10OQBFzIJO9kkE559Wcmn+qkEiiKk=
35 | github.com/coreos/etcd v3.3.13+incompatible/go.mod h1:uF7uidLiAD3TWHmW31ZFd/JWoc32PjwdhPthX9715RE=
36 | github.com/coreos/go-semver v0.3.0/go.mod h1:nnelYz7RCh+5ahJtPPxZlU+153eP4D4r3EedlOD2RNk=
37 | github.com/coreos/go-systemd v0.0.0-20190321100706-95778dfbb74e/go.mod h1:F5haX7vjVVG0kc13fIWeqUViNPyEJxv/OmvnBo0Yme4=
38 | github.com/coreos/pkg v0.0.0-20180928190104-399ea9e2e55f/go.mod h1:E3G3o1h8I7cfcXa63jLwjI0eiQQMgzzUDFVpN/nH/eA=
39 | github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
40 | github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
41 | github.com/denisenkom/go-mssqldb v0.0.0-20191124224453-732737034ffd/go.mod h1:xbL0rPBG9cCiLr28tMa8zpbdarY27NDyej4t/EjAShU=
42 | github.com/dgrijalva/jwt-go v3.2.0+incompatible h1:7qlOGliEKZXTDg6OTjfoBKDXWrumCAMpl/TFQ4/5kLM=
43 | github.com/dgrijalva/jwt-go v3.2.0+incompatible/go.mod h1:E3ru+11k8xSBh+hMPgOLZmtrrCbhqsmaPHjLKYnJCaQ=
44 | github.com/dgryski/go-sip13 v0.0.0-20181026042036-e10d5fee7954/go.mod h1:vAd38F8PWV+bWy6jNmig1y/TA+kYO4g3RSRF0IAv0no=
45 | github.com/erikstmartin/go-testdb v0.0.0-20160219214506-8d10e4a1bae5/go.mod h1:a2zkGnVExMxdzMo3M0Hi/3sEU+cWnZpSni0O6/Yb/P0=
46 | github.com/fatih/color v1.7.0/go.mod h1:Zm6kSWBoL9eyXnKyktHP6abPY2pDugNf5KwzbycvMj4=
47 | github.com/fsnotify/fsnotify v1.4.7 h1:IXs+QLmnXW2CcXuY+8Mzv/fWEsPGWxqefPtCP5CnV9I=
48 | github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo=
49 | github.com/ghodss/yaml v1.0.0/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeMEF04=
50 | github.com/gin-contrib/gzip v0.0.1/go.mod h1:fGBJBCdt6qCZuCAOwWuFhBB4OOq9EFqlo5dEaFhhu5w=
51 | github.com/gin-contrib/sse v0.0.0-20170109093832-22d885f9ecc7/go.mod h1:VJ0WA2NBN22VlZ2dKZQPAPnyWw5XTlK1KymzLKsr59s=
52 | github.com/gin-contrib/sse v0.0.0-20190301062529-5545eab6dad3/go.mod h1:VJ0WA2NBN22VlZ2dKZQPAPnyWw5XTlK1KymzLKsr59s=
53 | github.com/gin-contrib/sse v0.1.0 h1:Y/yl/+YNO8GZSjAhjMsSuLt29uWRFHdHYUb5lYOV9qE=
54 | github.com/gin-contrib/sse v0.1.0/go.mod h1:RHrZQHXnP2xjPF+u1gW/2HnVO7nvIa9PG3Gm+fLHvGI=
55 | github.com/gin-gonic/gin v1.3.0/go.mod h1:7cKuhb5qV2ggCFctp2fJQ+ErvciLZrIeoOSOm6mUr7Y=
56 | github.com/gin-gonic/gin v1.4.0/go.mod h1:OW2EZn3DO8Ln9oIKOvM++LBO+5UPHJJDH72/q/3rZdM=
57 | github.com/gin-gonic/gin v1.6.3 h1:ahKqKTFpO5KTPHxWZjEdPScmYaGtLo8Y4DMHoEsnp14=
58 | github.com/gin-gonic/gin v1.6.3/go.mod h1:75u5sXoLsGZoRN5Sgbi1eraJ4GU3++wFwWzhwvtwp4M=
59 | github.com/go-gl/glfw v0.0.0-20190409004039-e6da0acd62b1/go.mod h1:vR7hzQXu2zJy9AVAgeJqvqgH9Q5CA+iKCZ2gyEVpxRU=
60 | github.com/go-kit/kit v0.8.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as=
61 | github.com/go-logfmt/logfmt v0.3.0/go.mod h1:Qt1PoO58o5twSAckw1HlFXLmHsOX5/0LbT9GBnD5lWE=
62 | github.com/go-logfmt/logfmt v0.4.0/go.mod h1:3RMwSq7FuexP4Kalkev3ejPJsZTpXXBr9+V4qmtdjCk=
63 | github.com/go-openapi/jsonpointer v0.17.0 h1:nH6xp8XdXHx8dqveo0ZuJBluCO2qGrPbDNZ0dwoRHP0=
64 | github.com/go-openapi/jsonpointer v0.17.0/go.mod h1:cOnomiV+CVVwFLk0A/MExoFMjwdsUdVpsRhURCKh+3M=
65 | github.com/go-openapi/jsonreference v0.17.0/go.mod h1:g4xxGn04lDIRh0GJb5QlpE3HfopLOL6uZrK/VgnsK9I=
66 | github.com/go-openapi/jsonreference v0.19.0 h1:BqWKpV1dFd+AuiKlgtddwVIFQsuMpxfBDBHGfM2yNpk=
67 | github.com/go-openapi/jsonreference v0.19.0/go.mod h1:g4xxGn04lDIRh0GJb5QlpE3HfopLOL6uZrK/VgnsK9I=
68 | github.com/go-openapi/spec v0.19.0 h1:A4SZ6IWh3lnjH0rG0Z5lkxazMGBECtrZcbyYQi+64k4=
69 | github.com/go-openapi/spec v0.19.0/go.mod h1:XkF/MOi14NmjsfZ8VtAKf8pIlbZzyoTvZsdfssdxcBI=
70 | github.com/go-openapi/swag v0.17.0 h1:iqrgMg7Q7SvtbWLlltPrkMs0UBJI6oTSs79JFRUi880=
71 | github.com/go-openapi/swag v0.17.0/go.mod h1:AByQ+nYG6gQg71GINrmuDXCPWdL640yX49/kXLo40Tg=
72 | github.com/go-playground/assert/v2 v2.0.1/go.mod h1:VDjEfimB/XKnb+ZQfWdccd7VUvScMdVu0Titje2rxJ4=
73 | github.com/go-playground/locales v0.13.0 h1:HyWk6mgj5qFqCT5fjGBuRArbVDfE4hi8+e8ceBS/t7Q=
74 | github.com/go-playground/locales v0.13.0/go.mod h1:taPMhCMXrRLJO55olJkUXHZBHCxTMfnGwq/HNwmWNS8=
75 | github.com/go-playground/universal-translator v0.17.0 h1:icxd5fm+REJzpZx7ZfpaD876Lmtgy7VtROAbHHXk8no=
76 | github.com/go-playground/universal-translator v0.17.0/go.mod h1:UkSxE5sNxxRwHyU+Scu5vgOQjsIJAF8j9muTVoKLVtA=
77 | github.com/go-playground/validator/v10 v10.2.0 h1:KgJ0snyC2R9VXYN2rneOtQcw5aHQB1Vv0sFl1UcHBOY=
78 | github.com/go-playground/validator/v10 v10.2.0/go.mod h1:uOYAAleCW8F/7oMFd6aG0GOhaH6EGOAJShg8Id5JGkI=
79 | github.com/go-redis/redis v6.15.9+incompatible h1:K0pv1D7EQUjfyoMql+r/jZqCLizCGKFlFgcHWWmHQjg=
80 | github.com/go-redis/redis v6.15.9+incompatible/go.mod h1:NAIEuMOZ/fxfXJIrKDQDz8wamY7mA7PouImQ2Jvg6kA=
81 | github.com/go-sql-driver/mysql v1.5.0 h1:ozyZYNQW3x3HtqT1jira07DN2PArx2v7/mN66gGcHOs=
82 | github.com/go-sql-driver/mysql v1.5.0/go.mod h1:DCzpHaOWr8IXmIStZouvnhqoel9Qv2LBy8hT2VhHyBg=
83 | github.com/go-stack/stack v1.8.0/go.mod h1:v0f6uXyyMGvRgIKkXu+yp6POWl0qKG85gN/melR3HDY=
84 | github.com/gogo/protobuf v1.1.1/go.mod h1:r8qH/GZQm5c6nD/R0oafs1akxWv10x8SbQlK7atdtwQ=
85 | github.com/gogo/protobuf v1.2.1/go.mod h1:hp+jE20tsWTFYpLwKvXlhS1hjn+gTNwPg2I6zVXpSg4=
86 | github.com/golang-sql/civil v0.0.0-20190719163853-cb61b32ac6fe/go.mod h1:8vg3r2VgvsThLBIFL93Qb5yWzgyZWhEmBwUJWevAkK0=
87 | github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q=
88 | github.com/golang/groupcache v0.0.0-20190129154638-5b532d6fd5ef/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc=
89 | github.com/golang/mock v1.1.1/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A=
90 | github.com/golang/mock v1.2.0/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A=
91 | github.com/golang/mock v1.3.1/go.mod h1:sBzyDLLjw3U8JLTeZvSv8jJB+tU5PVekmnlKIyFUx0Y=
92 | github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
93 | github.com/golang/protobuf v1.3.1/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
94 | github.com/golang/protobuf v1.3.2/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
95 | github.com/golang/protobuf v1.3.3 h1:gyjaxf+svBWX08ZjK86iN9geUJF0H6gp2IRKX6Nf6/I=
96 | github.com/golang/protobuf v1.3.3/go.mod h1:vzj43D7+SQXF/4pzW/hwtAqwc6iTitCiVSaWz5lYuqw=
97 | github.com/google/btree v0.0.0-20180813153112-4030bb1f1f0c/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ=
98 | github.com/google/btree v1.0.0/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ=
99 | github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M=
100 | github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU=
101 | github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg=
102 | github.com/google/martian v2.1.0+incompatible/go.mod h1:9I4somxYTbIHy5NJKHRl3wXiIaQGbYVAs8BPL6v8lEs=
103 | github.com/google/pprof v0.0.0-20181206194817-3ea8567a2e57/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc=
104 | github.com/google/pprof v0.0.0-20190515194954-54271f7e092f/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc=
105 | github.com/google/renameio v0.1.0/go.mod h1:KWCgfxg9yswjAJkECMjeO8J8rahYeXnNhOm40UhjYkI=
106 | github.com/googleapis/gax-go/v2 v2.0.4/go.mod h1:0Wqv26UfaUD9n4G6kQubkQ+KchISgw+vpHVxEJEs9eg=
107 | github.com/googleapis/gax-go/v2 v2.0.5/go.mod h1:DWXyrwAJ9X0FpwwEdw+IPEYBICEFu5mhpdKc/us6bOk=
108 | github.com/gopherjs/gopherjs v0.0.0-20181017120253-0766667cb4d1/go.mod h1:wJfORRmW1u3UXTncJ5qlYoELFm8eSnnEO6hX4iZ3EWY=
109 | github.com/gorilla/websocket v1.4.2/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE=
110 | github.com/grpc-ecosystem/go-grpc-middleware v1.0.0/go.mod h1:FiyG127CGDf3tlThmgyCl78X/SZQqEOJBCDaAfeWzPs=
111 | github.com/grpc-ecosystem/go-grpc-prometheus v1.2.0/go.mod h1:8NvIoxWQoOIhqOTXgfV/d3M/q6VIi02HzZEHgUlZvzk=
112 | github.com/grpc-ecosystem/grpc-gateway v1.9.0/go.mod h1:vNeuVxBJEsws4ogUvrchl83t/GYV9WGTSLVdBhOQFDY=
113 | github.com/hashicorp/consul/api v1.1.0/go.mod h1:VmuI/Lkw1nC05EYQWNKwWGbkg+FbDBtguAZLlVdkD9Q=
114 | github.com/hashicorp/consul/sdk v0.1.1/go.mod h1:VKf9jXwCTEY1QZP2MOLRhb5i/I/ssyNV1vwHyQBF0x8=
115 | github.com/hashicorp/errwrap v1.0.0/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4=
116 | github.com/hashicorp/go-cleanhttp v0.5.1/go.mod h1:JpRdi6/HCYpAwUzNwuwqhbovhLtngrth3wmdIIUrZ80=
117 | github.com/hashicorp/go-immutable-radix v1.0.0/go.mod h1:0y9vanUI8NX6FsYoO3zeMjhV/C5i9g4Q3DwcSNZ4P60=
118 | github.com/hashicorp/go-msgpack v0.5.3/go.mod h1:ahLV/dePpqEmjfWmKiqvPkv/twdG7iPBM1vqhUKIvfM=
119 | github.com/hashicorp/go-multierror v1.0.0/go.mod h1:dHtQlpGsu+cZNNAkkCN/P3hoUDHhCYQXV3UM06sGGrk=
120 | github.com/hashicorp/go-rootcerts v1.0.0/go.mod h1:K6zTfqpRlCUIjkwsN4Z+hiSfzSTQa6eBIzfwKfwNnHU=
121 | github.com/hashicorp/go-sockaddr v1.0.0/go.mod h1:7Xibr9yA9JjQq1JpNB2Vw7kxv8xerXegt+ozgdvDeDU=
122 | github.com/hashicorp/go-syslog v1.0.0/go.mod h1:qPfqrKkXGihmCqbJM2mZgkZGvKG1dFdvsLplgctolz4=
123 | github.com/hashicorp/go-uuid v1.0.0/go.mod h1:6SBZvOh/SIDV7/2o3Jml5SYk/TvGqwFJ/bN7x4byOro=
124 | github.com/hashicorp/go-uuid v1.0.1/go.mod h1:6SBZvOh/SIDV7/2o3Jml5SYk/TvGqwFJ/bN7x4byOro=
125 | github.com/hashicorp/go.net v0.0.1/go.mod h1:hjKkEWcCURg++eb33jQU7oqQcI9XDCnUzHA0oac0k90=
126 | github.com/hashicorp/golang-lru v0.5.0/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8=
127 | github.com/hashicorp/golang-lru v0.5.1/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8=
128 | github.com/hashicorp/hcl v1.0.0 h1:0Anlzjpi4vEasTeNFn2mLJgTSwt0+6sfsiTG8qcWGx4=
129 | github.com/hashicorp/hcl v1.0.0/go.mod h1:E5yfLk+7swimpb2L/Alb/PJmXilQ/rhwaUYs4T20WEQ=
130 | github.com/hashicorp/logutils v1.0.0/go.mod h1:QIAnNjmIWmVIIkWDTG1z5v++HQmx9WQRO+LraFDTW64=
131 | github.com/hashicorp/mdns v1.0.0/go.mod h1:tL+uN++7HEJ6SQLQ2/p+z2pH24WQKWjBPkE0mNTz8vQ=
132 | github.com/hashicorp/memberlist v0.1.3/go.mod h1:ajVTdAv/9Im8oMAAj5G31PhhMCZJV2pPBoIllUwCN7I=
133 | github.com/hashicorp/serf v0.8.2/go.mod h1:6hOLApaqBFA1NXqRQAsxw9QxuDEvNxSQRwA/JwenrHc=
134 | github.com/jinzhu/gorm v1.9.16 h1:+IyIjPEABKRpsu/F8OvDPy9fyQlgsg2luMV2ZIH5i5o=
135 | github.com/jinzhu/gorm v1.9.16/go.mod h1:G3LB3wezTOWM2ITLzPxEXgSkOXAntiLHS7UdBefADcs=
136 | github.com/jinzhu/inflection v1.0.0 h1:K317FqzuhWc8YvSVlFMCCUb36O/S9MCKRDI7QkRKD/E=
137 | github.com/jinzhu/inflection v1.0.0/go.mod h1:h+uFLlag+Qp1Va5pdKtLDYj+kHp5pxUVkryuEj+Srlc=
138 | github.com/jinzhu/now v1.0.1/go.mod h1:d3SSVoowX0Lcu0IBviAWJpolVfI5UJVZZ7cO71lE/z8=
139 | github.com/jonboulle/clockwork v0.1.0/go.mod h1:Ii8DK3G1RaLaWxj9trq07+26W01tbo22gdxWY5EU2bo=
140 | github.com/json-iterator/go v1.1.5/go.mod h1:+SdeFBvtyEkXs7REEP0seUULqWtbJapLOCVDaaPEHmU=
141 | github.com/json-iterator/go v1.1.6/go.mod h1:+SdeFBvtyEkXs7REEP0seUULqWtbJapLOCVDaaPEHmU=
142 | github.com/json-iterator/go v1.1.9/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4=
143 | github.com/jstemmer/go-junit-report v0.0.0-20190106144839-af01ea7f8024/go.mod h1:6v2b51hI/fHJwM22ozAgKL4VKDeJcHhJFhtBdhmNjmU=
144 | github.com/jtolds/gls v4.20.0+incompatible/go.mod h1:QJZ7F/aHp+rZTRtaJ1ow/lLfFfVYBRgL+9YlvaHOwJU=
145 | github.com/julienschmidt/httprouter v1.2.0/go.mod h1:SYymIcj16QtmaHHD7aYtjjsJG7VTCxuUUipMqKk8s4w=
146 | github.com/kisielk/errcheck v1.1.0/go.mod h1:EZBBE59ingxPouuu3KfxchcWSUPOHkagtvWXihfKN4Q=
147 | github.com/kisielk/gotool v1.0.0/go.mod h1:XhKaO+MFFWcvkIS/tQcRk01m1F5IRFswLeQ+oQHNcck=
148 | github.com/konsorten/go-windows-terminal-sequences v1.0.1/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ=
149 | github.com/kr/logfmt v0.0.0-20140226030751-b84e30acd515/go.mod h1:+0opPa2QZZtGFBFZlji/RkVcI2GknAs/DXo4wKdlNEc=
150 | github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo=
151 | github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ=
152 | github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI=
153 | github.com/leodido/go-urn v1.2.0 h1:hpXL4XnriNwQ/ABnpepYM/1vCLWNDfUNts8dX3xTG6Y=
154 | github.com/leodido/go-urn v1.2.0/go.mod h1:+8+nEpDfqqsY+g338gtMEUOtuK+4dEMhiQEgxpxOKII=
155 | github.com/lib/pq v1.1.1/go.mod h1:5WUZQaWbwv1U+lTReE5YruASi9Al49XbQIvNi/34Woo=
156 | github.com/magiconair/properties v1.8.1 h1:ZC2Vc7/ZFkGmsVC9KvOjumD+G5lXy2RtTKyzRKO2BQ4=
157 | github.com/magiconair/properties v1.8.1/go.mod h1:PppfXfuXeibc/6YijjN8zIbojt8czPbwD3XqdrwzmxQ=
158 | github.com/mailru/easyjson v0.0.0-20180823135443-60711f1a8329 h1:2gxZ0XQIU/5z3Z3bUBu+FXuk2pFbkN6tcwi/pjyaDic=
159 | github.com/mailru/easyjson v0.0.0-20180823135443-60711f1a8329/go.mod h1:C1wdFJiN94OJF2b5HbByQZoLdCWB1Yqtg26g4irojpc=
160 | github.com/mattn/go-colorable v0.0.9/go.mod h1:9vuHe8Xs5qXnSaW/c/ABM9alt+Vo+STaOChaDxuIBZU=
161 | github.com/mattn/go-isatty v0.0.3/go.mod h1:M+lRXTBqGeGNdLjl/ufCoiOlB5xdOkqRJdNxMWT7Zi4=
162 | github.com/mattn/go-isatty v0.0.4/go.mod h1:M+lRXTBqGeGNdLjl/ufCoiOlB5xdOkqRJdNxMWT7Zi4=
163 | github.com/mattn/go-isatty v0.0.7/go.mod h1:Iq45c/XA43vh69/j3iqttzPXn0bhXyGjM0Hdxcsrc5s=
164 | github.com/mattn/go-isatty v0.0.8/go.mod h1:Iq45c/XA43vh69/j3iqttzPXn0bhXyGjM0Hdxcsrc5s=
165 | github.com/mattn/go-isatty v0.0.12 h1:wuysRhFDzyxgEmMf5xjvJ2M9dZoWAXNNr5LSBS7uHXY=
166 | github.com/mattn/go-isatty v0.0.12/go.mod h1:cbi8OIDigv2wuxKPP5vlRcQ1OAZbq2CE4Kysco4FUpU=
167 | github.com/mattn/go-sqlite3 v1.14.0/go.mod h1:JIl7NbARA7phWnGvh0LKTyg7S9BA+6gx71ShQilpsus=
168 | github.com/matttproud/golang_protobuf_extensions v1.0.1/go.mod h1:D8He9yQNgCq6Z5Ld7szi9bcBfOoFv/3dc6xSMkL2PC0=
169 | github.com/miekg/dns v1.0.14/go.mod h1:W1PPwlIAgtquWBMBEV9nkV9Cazfe8ScdGz/Lj7v3Nrg=
170 | github.com/mitchellh/cli v1.0.0/go.mod h1:hNIlj7HEI86fIcpObd7a0FcrxTWetlwJDGcceTlRvqc=
171 | github.com/mitchellh/go-homedir v1.0.0/go.mod h1:SfyaCUpYCn1Vlf4IUYiD9fPX4A5wJrkLzIz1N1q0pr0=
172 | github.com/mitchellh/go-testing-interface v1.0.0/go.mod h1:kRemZodwjscx+RGhAo8eIhFbs2+BFgRtFPeD/KE+zxI=
173 | github.com/mitchellh/gox v0.4.0/go.mod h1:Sd9lOJ0+aimLBi73mGofS1ycjY8lL3uZM3JPS42BGNg=
174 | github.com/mitchellh/iochan v1.0.0/go.mod h1:JwYml1nuB7xOzsp52dPpHFffvOCDupsG0QubkSMEySY=
175 | github.com/mitchellh/mapstructure v0.0.0-20160808181253-ca63d7c062ee/go.mod h1:FVVH3fgwuzCH5S8UJGiWEs2h04kUh9fWfEaFds41c1Y=
176 | github.com/mitchellh/mapstructure v1.1.2 h1:fmNYVwqnSfB9mZU6OS2O6GsXM+wcskZDuKQzvN1EDeE=
177 | github.com/mitchellh/mapstructure v1.1.2/go.mod h1:FVVH3fgwuzCH5S8UJGiWEs2h04kUh9fWfEaFds41c1Y=
178 | github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
179 | github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
180 | github.com/modern-go/reflect2 v0.0.0-20180701023420-4b7aa43c6742/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0=
181 | github.com/modern-go/reflect2 v1.0.1/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0=
182 | github.com/mwitkow/go-conntrack v0.0.0-20161129095857-cc309e4a2223/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U=
183 | github.com/oklog/ulid v1.3.1/go.mod h1:CirwcVhetQ6Lv90oh/F+FBtV6XMibvdAFo93nm5qn4U=
184 | github.com/pascaldekloe/goe v0.0.0-20180627143212-57f6aae5913c/go.mod h1:lzWF7FIEvWOWxwDKqyGYQf6ZUaNfKdP144TG7ZOy1lc=
185 | github.com/pelletier/go-toml v1.2.0 h1:T5zMGML61Wp+FlcbWjRDT7yAxhJNAiPPLOFECq181zc=
186 | github.com/pelletier/go-toml v1.2.0/go.mod h1:5z9KED0ma1S8pY6P1sdut58dfprrGBbd/94hg7ilaic=
187 | github.com/pkg/errors v0.8.0/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
188 | github.com/pkg/errors v0.8.1 h1:iURUrRGxPUNPdy5/HRSm+Yj6okJ6UtLINN0Q9M4+h3I=
189 | github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
190 | github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
191 | github.com/posener/complete v1.1.1/go.mod h1:em0nMJCgc9GFtwrmVmEMR/ZL6WyhyjMBndrE9hABlRI=
192 | github.com/prometheus/client_golang v0.9.1/go.mod h1:7SWBe2y4D6OKWSNQJUaRYU/AaXPKyh/dDVn+NZz0KFw=
193 | github.com/prometheus/client_golang v0.9.3/go.mod h1:/TN21ttK/J9q6uSwhBd54HahCDft0ttaMvbicHlPoso=
194 | github.com/prometheus/client_model v0.0.0-20180712105110-5c3871d89910/go.mod h1:MbSGuTsp3dbXC40dX6PRTWyKYBIrTGTE9sqQNg2J8bo=
195 | github.com/prometheus/client_model v0.0.0-20190129233127-fd36f4220a90/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA=
196 | github.com/prometheus/common v0.0.0-20181113130724-41aa239b4cce/go.mod h1:daVV7qP5qjZbuso7PdcryaAu0sAZbrN9i7WWcTMWvro=
197 | github.com/prometheus/common v0.4.0/go.mod h1:TNfzLD0ON7rHzMJeJkieUDPYmFC7Snx/y86RQel1bk4=
198 | github.com/prometheus/procfs v0.0.0-20181005140218-185b4288413d/go.mod h1:c3At6R/oaqEKCNdg8wHV1ftS6bRYblBhIjjI8uT2IGk=
199 | github.com/prometheus/procfs v0.0.0-20190507164030-5867b95ac084/go.mod h1:TjEm7ze935MbeOT/UhFTIMYKhuLP4wbCsTZCD3I8kEA=
200 | github.com/prometheus/tsdb v0.7.1/go.mod h1:qhTCs0VvXwvX/y3TZrWD7rabWM+ijKTux40TwIPHuXU=
201 | github.com/rogpeppe/fastuuid v0.0.0-20150106093220-6724a57986af/go.mod h1:XWv6SoW27p1b0cqNHllgS5HIMJraePCO15w5zCzIWYg=
202 | github.com/rogpeppe/go-internal v1.3.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4=
203 | github.com/ryanuber/columnize v0.0.0-20160712163229-9b3edd62028f/go.mod h1:sm1tb6uqfes/u+d4ooFouqFdy9/2g9QGwK3SQygK0Ts=
204 | github.com/satori/go.uuid v1.2.0 h1:0uYX9dsZ2yD7q2RtLRtPSdGDWzjeM3TbMJP9utgA0ww=
205 | github.com/satori/go.uuid v1.2.0/go.mod h1:dA0hQrYB0VpLJoorglMZABFdXlWrHn1NEOzdhQKdks0=
206 | github.com/sean-/seed v0.0.0-20170313163322-e2103e2c3529/go.mod h1:DxrIzT+xaE7yg65j358z/aeFdxmN0P9QXhEzd20vsDc=
207 | github.com/sirupsen/logrus v1.2.0/go.mod h1:LxeOpSwHxABJmUn/MG1IvRgCAasNZTLOkJPxbbu5VWo=
208 | github.com/smartystreets/assertions v0.0.0-20180927180507-b2de0cb4f26d/go.mod h1:OnSkiWE9lh6wB0YB77sQom3nweQdgAjqCqsofrRNTgc=
209 | github.com/smartystreets/goconvey v1.6.4/go.mod h1:syvi0/a8iFYH4r/RixwvyeAJjdLS9QV7WQ/tjFTllLA=
210 | github.com/soheilhy/cmux v0.1.4/go.mod h1:IM3LyeVVIOuxMH7sFAkER9+bJ4dT7Ms6E4xg4kGIyLM=
211 | github.com/spaolacci/murmur3 v0.0.0-20180118202830-f09979ecbc72/go.mod h1:JwIasOWyU6f++ZhiEuf87xNszmSA2myDM2Kzu9HwQUA=
212 | github.com/spf13/afero v1.1.2 h1:m8/z1t7/fwjysjQRYbP0RD+bUIF/8tJwPdEZsI83ACI=
213 | github.com/spf13/afero v1.1.2/go.mod h1:j4pytiNVoe2o6bmDsKpLACNPDBIoEAkihy7loJ1B0CQ=
214 | github.com/spf13/cast v1.3.0 h1:oget//CVOEoFewqQxwr0Ej5yjygnqGkvggSE/gB35Q8=
215 | github.com/spf13/cast v1.3.0/go.mod h1:Qx5cxh0v+4UWYiBimWS+eyWzqEqokIECu5etghLkUJE=
216 | github.com/spf13/jwalterweatherman v1.0.0 h1:XHEdyB+EcvlqZamSM4ZOMGlc93t6AcsBEu9Gc1vn7yk=
217 | github.com/spf13/jwalterweatherman v1.0.0/go.mod h1:cQK4TGJAtQXfYWX+Ddv3mKDzgVb68N+wFjFa4jdeBTo=
218 | github.com/spf13/pflag v1.0.3 h1:zPAT6CGy6wXeQ7NtTnaTerfKOsV6V6F8agHXFiazDkg=
219 | github.com/spf13/pflag v1.0.3/go.mod h1:DYY7MBk1bdzusC3SYhjObp+wFpr4gzcvqqNjLnInEg4=
220 | github.com/spf13/viper v1.7.1 h1:pM5oEahlgWv/WnHXpgbKz7iLIxRf65tye2Ci+XFK5sk=
221 | github.com/spf13/viper v1.7.1/go.mod h1:8WkrPz2fc9jxqZNCJI/76HCieCp4Q8HaLFoCha5qpdg=
222 | github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
223 | github.com/stretchr/objx v0.1.1/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
224 | github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs=
225 | github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
226 | github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4=
227 | github.com/subosito/gotenv v1.2.0 h1:Slr1R9HxAlEKefgq5jn9U+DnETlIUa6HfgEzj0g5d7s=
228 | github.com/subosito/gotenv v1.2.0/go.mod h1:N0PQaV/YGNqwC0u51sEeR/aUtSLEXKX9iv69rRypqCw=
229 | github.com/swaggo/gin-swagger v1.2.0 h1:YskZXEiv51fjOMTsXrOetAjrMDfFaXD79PEoQBOe2W0=
230 | github.com/swaggo/gin-swagger v1.2.0/go.mod h1:qlH2+W7zXGZkczuL+r2nEBR2JTT+/lX05Nn6vPhc7OI=
231 | github.com/swaggo/swag v1.5.1 h1:2Agm8I4K5qb00620mHq0VJ05/KT4FtmALPIcQR9lEZM=
232 | github.com/swaggo/swag v1.5.1/go.mod h1:1Bl9F/ZBpVWh22nY0zmYyASPO1lI/zIwRDrpZU+tv8Y=
233 | github.com/tmc/grpc-websocket-proxy v0.0.0-20190109142713-0ad062ec5ee5/go.mod h1:ncp9v5uamzpCO7NfCPTXjqaC+bZgJeR0sMTm6dMHP7U=
234 | github.com/ugorji/go v1.1.4/go.mod h1:uQMGLiO92mf5W77hV/PUCpI3pbzQx3CRekS0kk+RGrc=
235 | github.com/ugorji/go v1.1.5-pre/go.mod h1:FwP/aQVg39TXzItUBMwnWp9T9gPQnXw4Poh4/oBQZ/0=
236 | github.com/ugorji/go v1.1.7 h1:/68gy2h+1mWMrwZFeD1kQialdSzAb432dtpeJ42ovdo=
237 | github.com/ugorji/go v1.1.7/go.mod h1:kZn38zHttfInRq0xu/PH0az30d+z6vm202qpg1oXVMw=
238 | github.com/ugorji/go/codec v0.0.0-20181022190402-e5e69e061d4f/go.mod h1:VFNgLljTbGfSG7qAOspJ7OScBnGdDN/yBr0sguwnwf0=
239 | github.com/ugorji/go/codec v1.1.5-pre/go.mod h1:tULtS6Gy1AE1yCENaw4Vb//HLH5njI2tfCQDUqRd8fI=
240 | github.com/ugorji/go/codec v1.1.7 h1:2SvQaVZ1ouYrrKKwoSk2pzd4A9evlKJb9oTL+OaLUSs=
241 | github.com/ugorji/go/codec v1.1.7/go.mod h1:Ax+UKWsSmolVDwsd+7N3ZtXu+yMGCf907BLYF3GoBXY=
242 | github.com/urfave/cli v1.20.0/go.mod h1:70zkFmudgCuE/ngEzBv17Jvp/497gISqfk5gWijbERA=
243 | github.com/xiang90/probing v0.0.0-20190116061207-43a291ad63a2/go.mod h1:UETIi67q53MR2AWcXfiuqkDkRtnGDLqkBTpCHuJHxtU=
244 | go.etcd.io/bbolt v1.3.2/go.mod h1:IbVyRI1SCnLcuJnV2u8VeU0CEYM7e686BmAb1XKL+uU=
245 | go.opencensus.io v0.21.0/go.mod h1:mSImk1erAIZhrmZN+AvHh14ztQfjbGwt4TtuofqLduU=
246 | go.opencensus.io v0.22.0/go.mod h1:+kGneAE2xo2IficOXnaByMWTGM9T73dGwxeWcUqIpI8=
247 | go.uber.org/atomic v1.4.0/go.mod h1:gD2HeocX3+yG+ygLZcrzQJaqmWj9AIm7n08wl/qW/PE=
248 | go.uber.org/multierr v1.1.0/go.mod h1:wR5kodmAFQ0UK8QlbwjlSNy0Z68gJhDJUG5sjR94q/0=
249 | go.uber.org/zap v1.10.0/go.mod h1:vwi/ZaCAaUcBkycHslxD9B2zi4UTXhF60s6SWpuDF0Q=
250 | golang.org/x/crypto v0.0.0-20180904163835-0709b304e793/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4=
251 | golang.org/x/crypto v0.0.0-20181029021203-45a5f77698d3/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4=
252 | golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
253 | golang.org/x/crypto v0.0.0-20190325154230-a5d413f7728c/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
254 | golang.org/x/crypto v0.0.0-20190510104115-cbcb75029529/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
255 | golang.org/x/crypto v0.0.0-20190605123033-f99c8df09eb5/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
256 | golang.org/x/crypto v0.0.0-20191205180655-e7c4368fe9dd/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto=
257 | golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA=
258 | golang.org/x/exp v0.0.0-20190306152737-a1d7652674e8/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA=
259 | golang.org/x/exp v0.0.0-20190510132918-efd6b22b2522/go.mod h1:ZjyILWgesfNpC6sMxTJOJm9Kp84zZh5NQWvqDGG3Qr8=
260 | golang.org/x/exp v0.0.0-20190829153037-c13cbed26979/go.mod h1:86+5VVa7VpoJ4kLfm080zCjGlMRFzhUhsZKEZO7MGek=
261 | golang.org/x/exp v0.0.0-20191030013958-a1ab85dbe136/go.mod h1:JXzH8nQsPlswgeRAPE3MuO9GYsAcnJvJ4vnMwN/5qkY=
262 | golang.org/x/image v0.0.0-20190227222117-0694c2d4d067/go.mod h1:kZ7UVZpmo3dzQBMxlp+ypCbDeSB+sBbTgSJuh5dn5js=
263 | golang.org/x/image v0.0.0-20190802002840-cff245a6509b/go.mod h1:FeLwcggjj3mMvU+oOTbSwawSJRM1uh48EjtB4UJZlP0=
264 | golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE=
265 | golang.org/x/lint v0.0.0-20190227174305-5b3e6a55c961/go.mod h1:wehouNa3lNwaWXcvxsM5YxQ5yQlVC4a0KAMCusXpPoU=
266 | golang.org/x/lint v0.0.0-20190301231843-5614ed5bae6f/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE=
267 | golang.org/x/lint v0.0.0-20190313153728-d0100b6bd8b3/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc=
268 | golang.org/x/lint v0.0.0-20190409202823-959b441ac422/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc=
269 | golang.org/x/lint v0.0.0-20190909230951-414d861bb4ac/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc=
270 | golang.org/x/lint v0.0.0-20190930215403-16217165b5de/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc=
271 | golang.org/x/mobile v0.0.0-20190312151609-d3739f865fa6/go.mod h1:z+o9i4GpDbdi3rU15maQ/Ox0txvL9dWGYEHz965HBQE=
272 | golang.org/x/mobile v0.0.0-20190719004257-d2bd2a29d028/go.mod h1:E/iHnbuqvinMTCcRqshq8CkpyQDoeVncDDYHnLhea+o=
273 | golang.org/x/mod v0.0.0-20190513183733-4bf6d317e70e/go.mod h1:mXi4GBBbnImb6dmsKGUJ2LatrhH/nqhxcFungHvyanc=
274 | golang.org/x/mod v0.1.0/go.mod h1:0QHyrYULN0/3qlju5TqG8bIK38QM8yzMo5ekMj3DlcY=
275 | golang.org/x/net v0.0.0-20180218175443-cbe0f9307d01/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
276 | golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
277 | golang.org/x/net v0.0.0-20180826012351-8a410e7b638d/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
278 | golang.org/x/net v0.0.0-20181005035420-146acd28ed58/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
279 | golang.org/x/net v0.0.0-20181023162649-9b4f9f5ad519/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
280 | golang.org/x/net v0.0.0-20181114220301-adae6a3d119a/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
281 | golang.org/x/net v0.0.0-20181201002055-351d144fa1fc/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
282 | golang.org/x/net v0.0.0-20181220203305-927f97764cc3/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
283 | golang.org/x/net v0.0.0-20190108225652-1e06a53dbb7e/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
284 | golang.org/x/net v0.0.0-20190213061140-3a22650c66bd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
285 | golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
286 | golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
287 | golang.org/x/net v0.0.0-20190501004415-9ce7a6920f09/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
288 | golang.org/x/net v0.0.0-20190503192946-f4e77d36d62c/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
289 | golang.org/x/net v0.0.0-20190603091049-60506f45cf65/go.mod h1:HSz+uSET+XFnRR8LxR5pz3Of3rY3CfYBVs4xY44aLks=
290 | golang.org/x/net v0.0.0-20190611141213-3f473d35a33a/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
291 | golang.org/x/net v0.0.0-20190620200207-3b0461eec859 h1:R/3boaszxrf1GEUWTVDzSKVwLmSJpwZ1yqXm8j0v2QI=
292 | golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
293 | golang.org/x/net v0.0.0-20200202094626-16171245cfb2/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
294 | golang.org/x/net v0.0.0-20200324143707-d3edc9973b7e h1:3G+cUijn7XD+S4eJFddp53Pv7+slrESplyjG25HgL+k=
295 | golang.org/x/net v0.0.0-20200324143707-d3edc9973b7e/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A=
296 | golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U=
297 | golang.org/x/oauth2 v0.0.0-20190226205417-e64efc72b421/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw=
298 | golang.org/x/oauth2 v0.0.0-20190604053449-0f29369cfe45/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw=
299 | golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
300 | golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
301 | golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
302 | golang.org/x/sync v0.0.0-20190227155943-e225da77a7e6/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
303 | golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
304 | golang.org/x/sys v0.0.0-20180823144017-11551d06cbcc/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
305 | golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
306 | golang.org/x/sys v0.0.0-20180905080454-ebe1bf3edb33/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
307 | golang.org/x/sys v0.0.0-20181026203630-95b1ffbd15a5/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
308 | golang.org/x/sys v0.0.0-20181107165924-66b7b1311ac8/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
309 | golang.org/x/sys v0.0.0-20181116152217-5ac8a444bdc5/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
310 | golang.org/x/sys v0.0.0-20181228144115-9a3f9b0469bb/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
311 | golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
312 | golang.org/x/sys v0.0.0-20190222072716-a9d3bda3a223/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
313 | golang.org/x/sys v0.0.0-20190312061237-fead79001313/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
314 | golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
315 | golang.org/x/sys v0.0.0-20190502145724-3ef323f4f1fd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
316 | golang.org/x/sys v0.0.0-20190507160741-ecd444e8653b/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
317 | golang.org/x/sys v0.0.0-20190606165138-5da285871e9c/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
318 | golang.org/x/sys v0.0.0-20190610200419-93c9922d18ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
319 | golang.org/x/sys v0.0.0-20190624142023-c5567b49c5d0/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
320 | golang.org/x/sys v0.0.0-20200116001909-b77594299b42 h1:vEOn+mP2zCOVzKckCZy6YsCtDblrpj/w7B9nxGNELpg=
321 | golang.org/x/sys v0.0.0-20200116001909-b77594299b42/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
322 | golang.org/x/sys v0.0.0-20200323222414-85ca7c5b95cd h1:xhmwyvizuTgC2qz7ZlMluP20uW+C3Rm0FD/WLDX8884=
323 | golang.org/x/sys v0.0.0-20200323222414-85ca7c5b95cd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
324 | golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
325 | golang.org/x/text v0.3.1-0.20180807135948-17ff2d5776d2/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
326 | golang.org/x/text v0.3.2 h1:tW2bmiBqwgJj/UpqtC8EpXEZVYOwU0yG4iWbprSVAcs=
327 | golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk=
328 | golang.org/x/time v0.0.0-20181108054448-85acf8d2951c/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
329 | golang.org/x/time v0.0.0-20190308202827-9d24e82272b4/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
330 | golang.org/x/tools v0.0.0-20180221164845-07fd8470d635/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
331 | golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
332 | golang.org/x/tools v0.0.0-20190114222345-bf090417da8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
333 | golang.org/x/tools v0.0.0-20190226205152-f727befe758c/go.mod h1:9Yl7xja0Znq3iFh3HoIrodX9oNMXvdceNzlUR8zjMvY=
334 | golang.org/x/tools v0.0.0-20190311212946-11955173bddd/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
335 | golang.org/x/tools v0.0.0-20190312151545-0bb0c0a6e846/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
336 | golang.org/x/tools v0.0.0-20190312170243-e65039ee4138/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
337 | golang.org/x/tools v0.0.0-20190328211700-ab21143f2384/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
338 | golang.org/x/tools v0.0.0-20190425150028-36563e24a262/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q=
339 | golang.org/x/tools v0.0.0-20190506145303-2d16b83fe98c/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q=
340 | golang.org/x/tools v0.0.0-20190606050223-4d9ae51c2468/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc=
341 | golang.org/x/tools v0.0.0-20190606124116-d0a3d012864b/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc=
342 | golang.org/x/tools v0.0.0-20190611222205-d73e1c7e250b/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc=
343 | golang.org/x/tools v0.0.0-20190621195816-6e04913cbbac/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc=
344 | golang.org/x/tools v0.0.0-20190628153133-6cdbf07be9d0/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc=
345 | golang.org/x/tools v0.0.0-20190816200558-6889da9d5479/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
346 | golang.org/x/tools v0.0.0-20190911174233-4f2ddba30aff/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
347 | golang.org/x/tools v0.0.0-20191012152004-8de300cfc20a/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
348 | golang.org/x/tools v0.0.0-20191112195655-aa38f8e97acc h1:NCy3Ohtk6Iny5V/reW2Ktypo4zIpWBdRJ1uFMjBxdg8=
349 | golang.org/x/tools v0.0.0-20191112195655-aa38f8e97acc/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
350 | golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
351 | google.golang.org/api v0.4.0/go.mod h1:8k5glujaEP+g9n7WNsDg8QP6cUVNI86fCNMcbazEtwE=
352 | google.golang.org/api v0.7.0/go.mod h1:WtwebWUNSVBH/HAw79HIFXZNqEvBhG+Ra+ax0hx3E3M=
353 | google.golang.org/api v0.8.0/go.mod h1:o4eAsZoiT+ibD93RtjEohWalFOjRDx6CVaqeizhEnKg=
354 | google.golang.org/api v0.9.0/go.mod h1:o4eAsZoiT+ibD93RtjEohWalFOjRDx6CVaqeizhEnKg=
355 | google.golang.org/api v0.13.0/go.mod h1:iLdEw5Ide6rF15KTC1Kkl0iskquN2gFfn9o9XIsbkAI=
356 | google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM=
357 | google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4=
358 | google.golang.org/appengine v1.5.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4=
359 | google.golang.org/appengine v1.6.1/go.mod h1:i06prIuMbXzDqacNJfV5OdTW448YApPu5ww/cMBSeb0=
360 | google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc=
361 | google.golang.org/genproto v0.0.0-20190307195333-5fe7a883aa19/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE=
362 | google.golang.org/genproto v0.0.0-20190418145605-e7d98fc518a7/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE=
363 | google.golang.org/genproto v0.0.0-20190425155659-357c62f0e4bb/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE=
364 | google.golang.org/genproto v0.0.0-20190502173448-54afdca5d873/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE=
365 | google.golang.org/genproto v0.0.0-20190801165951-fa694d86fc64/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc=
366 | google.golang.org/genproto v0.0.0-20190819201941-24fa4b261c55/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc=
367 | google.golang.org/genproto v0.0.0-20190911173649-1774047e7e51/go.mod h1:IbNlFCBrqXvoKpeg0TB2l7cyZUmoaFKYIwrEpbDKLA8=
368 | google.golang.org/genproto v0.0.0-20191108220845-16a3f7862a1a/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc=
369 | google.golang.org/grpc v1.19.0/go.mod h1:mqu4LbDTu4XGKhr4mRzUsmM4RtVoemTSY81AxZiDr8c=
370 | google.golang.org/grpc v1.20.1/go.mod h1:10oTOabMzJvdu6/UiuZezV6QK5dSlG84ov/aaiqXj38=
371 | google.golang.org/grpc v1.21.1/go.mod h1:oYelfM1adQP15Ek0mdvEgi9Df8B9CZIaU1084ijfRaM=
372 | gopkg.in/alecthomas/kingpin.v2 v2.2.6/go.mod h1:FMv+mEhP44yOT+4EoQTLFTRgOQ1FBLkstjWtayDeSgw=
373 | gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
374 | gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
375 | gopkg.in/errgo.v2 v2.1.0/go.mod h1:hNsd1EY+bozCKY1Ytp96fpM3vjJbqLJn88ws8XvfDNI=
376 | gopkg.in/go-playground/assert.v1 v1.2.1/go.mod h1:9RXL0bg/zibRAgZUYszZSwO/z8Y/a8bDuhia5mkpMnE=
377 | gopkg.in/go-playground/validator.v8 v8.18.2/go.mod h1:RX2a/7Ha8BgOhfk7j780h4/u/RRjR0eouCJSH80/M2Y=
378 | gopkg.in/ini.v1 v1.51.0 h1:AQvPpx3LzTDM0AjnIRlVFwFFGC+npRopjZxLJj6gdno=
379 | gopkg.in/ini.v1 v1.51.0/go.mod h1:pNLf8WUiyNEtQjuu5G5vTm06TEv9tsIgeAvK8hOrP4k=
380 | gopkg.in/resty.v1 v1.12.0/go.mod h1:mDo4pnntr5jdWRML875a/NmxYqAlA73dVijT2AXvQQo=
381 | gopkg.in/yaml.v2 v2.0.0-20170812160011-eb3733d160e7/go.mod h1:JAlM8MvJe8wmxCU4Bli9HhUf9+ttbYbLASfIpnQbh74=
382 | gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
383 | gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
384 | gopkg.in/yaml.v2 v2.2.4/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
385 | gopkg.in/yaml.v2 v2.2.8 h1:obN1ZagJSUGI0Ek/LBmuj4SNLPfIny3KsKFopxRdj10=
386 | gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
387 | honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
388 | honnef.co/go/tools v0.0.0-20190106161140-3f1c8253044a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
389 | honnef.co/go/tools v0.0.0-20190418001031-e561f6794a2a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
390 | honnef.co/go/tools v0.0.1-2019.2.3/go.mod h1:a3bituU0lyd329TUQxRnasdCoJDkEUEAqEt0JzvZhAg=
391 | rsc.io/binaryregexp v0.2.0/go.mod h1:qTv7/COck+e2FymRvadv62gMdZztPaShugOCi3I+8D8=
392 |
--------------------------------------------------------------------------------
/handler/auth.go:
--------------------------------------------------------------------------------
1 | package handler
2 |
3 | import (
4 | "errors"
5 | "fmt"
6 | "net/http"
7 | "strconv"
8 | "strings"
9 | "time"
10 |
11 | "github.com/dgrijalva/jwt-go"
12 | "github.com/gin-gonic/gin"
13 | "github.com/golang-crew/Bolierplate-JWT-Auth/common"
14 | "github.com/golang-crew/Bolierplate-JWT-Auth/models"
15 | uuid "github.com/satori/go.uuid"
16 | "github.com/spf13/viper"
17 | )
18 |
19 | var user = models.Users{
20 | ID: 1,
21 | Username: "username",
22 | Password: "password",
23 | }
24 |
25 | type AccessDetails struct {
26 | AccessUuid string
27 | UserId uint64
28 | }
29 |
30 | type TokenDetails struct {
31 | AccessToken string
32 | RefreshToken string
33 | AccessUuid string
34 | RefreshUuid string
35 | AtExpires int64
36 | RtExpires int64
37 | }
38 |
39 | var ACCESS_SECRET = viper.GetString(`token.ACCESS_SECRET`)
40 | var REFRESH_SECRET = viper.GetString(`token.REFRESH_SECRET`)
41 |
42 | func Login(c *gin.Context) {
43 | var u models.Users
44 | if err := c.ShouldBindJSON(&u); err != nil {
45 | c.JSON(http.StatusUnprocessableEntity, "Invalid json provided")
46 | return
47 | }
48 |
49 | // compare the user from the request, with the one we defined:
50 | // TODO. DB에 값이 있는지 확인
51 | if user.Username != u.Username || user.Password != u.Password {
52 | c.JSON(http.StatusUnauthorized, "Please provide valid login details")
53 | return
54 | }
55 |
56 | // 토큰 발급
57 | ts, err := CreateToken(user.ID)
58 | if err != nil {
59 | c.JSON(http.StatusUnprocessableEntity, err.Error())
60 | return
61 | }
62 |
63 | saveErr := CreateAuth(user.ID, ts)
64 | if saveErr != nil {
65 | c.JSON(http.StatusUnprocessableEntity, saveErr.Error())
66 | }
67 |
68 | tokens := map[string]string{
69 | "access_token": ts.AccessToken,
70 | "refresh_token": ts.RefreshToken,
71 | }
72 | c.JSON(http.StatusOK, tokens)
73 | }
74 |
75 | func CreateToken(userid uint64) (td TokenDetails, err error) {
76 | td.AtExpires = time.Now().Add(time.Minute * 15).Unix()
77 | td.AccessUuid = uuid.NewV4().String()
78 | td.RtExpires = time.Now().Add(time.Hour * 24 * 7).Unix()
79 | td.RefreshUuid = td.AccessUuid + "++" + strconv.Itoa(int(userid))
80 |
81 | //Creating Access Token
82 | atClaims := jwt.MapClaims{}
83 | atClaims["authorized"] = true
84 | atClaims["access_uuid"] = td.AccessUuid
85 | atClaims["user_id"] = userid
86 | atClaims["exp"] = td.AtExpires
87 | at := jwt.NewWithClaims(jwt.SigningMethodHS256, atClaims)
88 | td.AccessToken, err = at.SignedString([]byte(ACCESS_SECRET))
89 | if err != nil {
90 | return
91 | }
92 |
93 | //Creating Refresh Token
94 | rtClaims := jwt.MapClaims{}
95 | rtClaims["refresh_uuid"] = td.RefreshUuid
96 | rtClaims["user_id"] = userid
97 | rtClaims["exp"] = td.RtExpires
98 | rt := jwt.NewWithClaims(jwt.SigningMethodHS256, rtClaims)
99 | td.RefreshToken, err = rt.SignedString([]byte(REFRESH_SECRET))
100 | if err != nil {
101 | return
102 | }
103 |
104 | return td, nil
105 | }
106 |
107 | func CreateAuth(userid uint64, td TokenDetails) (err error) {
108 | client := common.GetClient()
109 |
110 | at := time.Unix(td.AtExpires, 0) //converting Unix to UTC
111 | rt := time.Unix(td.RtExpires, 0)
112 | now := time.Now()
113 |
114 | if err = client.Set(td.AccessUuid, strconv.Itoa(int(userid)), at.Sub(now)).Err(); err != nil {
115 | return
116 | }
117 | if err = client.Set(td.RefreshUuid, strconv.Itoa(int(userid)), rt.Sub(now)).Err(); err != nil {
118 | return
119 | }
120 |
121 | return
122 | }
123 |
124 | func ExtractToken(r *http.Request) string {
125 | bearToken := r.Header.Get("Authorization")
126 | strArr := strings.Split(bearToken, " ")
127 | if len(strArr) == 2 {
128 | return strArr[1]
129 | }
130 | return ""
131 | }
132 |
133 | // Parse, validate, and return a token.
134 | // keyFunc will receive the parsed token and should return the key for validating.
135 | func VerifyToken(r *http.Request) (token *jwt.Token, err error) {
136 | tokenString := ExtractToken(r)
137 | token, err = jwt.Parse(tokenString, func(token *jwt.Token) (interface{}, error) {
138 | if _, ok := token.Method.(*jwt.SigningMethodHMAC); !ok {
139 | return nil, fmt.Errorf("unexpected signing method: %v", token.Header["alg"])
140 | }
141 | return []byte(ACCESS_SECRET), nil
142 | })
143 |
144 | // if err != nil {
145 | // return nil, err
146 | // }
147 | // return token, nil
148 | return
149 | }
150 |
151 | func TokenValid(r *http.Request) error {
152 | token, err := VerifyToken(r)
153 | if err != nil {
154 | return err
155 | }
156 |
157 | if _, ok := token.Claims.(jwt.Claims); !ok || !token.Valid {
158 | return err
159 | }
160 |
161 | return nil
162 | }
163 |
164 | func ExtractTokenMetadata(r *http.Request) (*AccessDetails, error) {
165 | token, err := VerifyToken(r)
166 | if err != nil {
167 | return nil, err
168 | }
169 | claims, ok := token.Claims.(jwt.MapClaims)
170 | if ok && token.Valid {
171 | accessUuid, ok := claims["access_uuid"].(string)
172 | if !ok {
173 | return nil, err
174 | }
175 | userId, err := strconv.ParseUint(fmt.Sprintf("%.f", claims["user_id"]), 10, 64)
176 | if err != nil {
177 | return nil, err
178 | }
179 | return &AccessDetails{
180 | AccessUuid: accessUuid,
181 | UserId: userId,
182 | }, nil
183 | }
184 | return nil, err
185 | }
186 |
187 | func FetchAuth(authD *AccessDetails) (uint64, error) {
188 | client := common.GetClient()
189 | userid, err := client.Get(authD.AccessUuid).Result()
190 | if err != nil {
191 | return 0, err
192 | }
193 |
194 | userID, _ := strconv.ParseUint(userid, 10, 64)
195 | if authD.UserId != userID {
196 | return 0, errors.New("unauthorized")
197 | }
198 | return userID, nil
199 | }
200 |
201 | func DeleteAuth(givenUuid string) (uint64, error) {
202 | client := common.GetClient()
203 | deleted, err := client.Del(givenUuid).Result()
204 | if err != nil {
205 | return 0, err
206 | }
207 |
208 | return uint64(deleted), nil
209 | }
210 |
211 | func Logout(c *gin.Context) {
212 | metadata, err := ExtractTokenMetadata(c.Request)
213 | if err != nil {
214 | c.JSON(http.StatusUnauthorized, "unauthorized")
215 | return
216 | }
217 |
218 | delErr := DeleteTokens(metadata)
219 | if delErr != nil {
220 | c.JSON(http.StatusUnauthorized, delErr.Error())
221 | return
222 | }
223 | c.JSON(http.StatusOK, "Successfully logged out")
224 | }
225 |
226 | func Refresh(c *gin.Context) {
227 | mapToken := map[string]string{}
228 | if err := c.ShouldBindJSON(&mapToken); err != nil {
229 | c.JSON(http.StatusUnprocessableEntity, err.Error())
230 | return
231 | }
232 | refreshToken := mapToken["refresh_token"]
233 |
234 | //verify the token
235 | // os.Setenv("REFRESH_SECRET", "mcmvmkmsdnfsdmfdsjf") //this should be in an env file
236 | token, err := jwt.Parse(refreshToken, func(token *jwt.Token) (interface{}, error) {
237 | //Make sure that the token method conform to "SigningMethodHMAC"
238 | if _, ok := token.Method.(*jwt.SigningMethodHMAC); !ok {
239 | return nil, fmt.Errorf("unexpected signing method: %v", token.Header["alg"])
240 | }
241 | return []byte(REFRESH_SECRET), nil
242 | })
243 |
244 | //if there is an error, the token must have expired
245 | if err != nil {
246 | c.JSON(http.StatusUnauthorized, "Refresh token expired")
247 | return
248 | }
249 |
250 | //is token valid?
251 | if _, ok := token.Claims.(jwt.Claims); !ok && !token.Valid {
252 | c.JSON(http.StatusUnauthorized, err)
253 | return
254 | }
255 |
256 | //Since token is valid, get the uuid:
257 | claims, ok := token.Claims.(jwt.MapClaims) //the token claims should conform to MapClaims
258 | if ok && token.Valid {
259 | refreshUuid, ok := claims["refresh_uuid"].(string) //convert the interface to string
260 | if !ok {
261 | c.JSON(http.StatusUnprocessableEntity, err)
262 | return
263 | }
264 | userId, err := strconv.ParseUint(fmt.Sprintf("%.f", claims["user_id"]), 10, 64)
265 | if err != nil {
266 | c.JSON(http.StatusUnprocessableEntity, "Error occurred")
267 | return
268 | }
269 | //Delete the previous Refresh Token
270 | deleted, delErr := DeleteAuth(refreshUuid)
271 | if delErr != nil || deleted == 0 { //if any goes wrong
272 | c.JSON(http.StatusUnauthorized, "unauthorized")
273 | return
274 | }
275 | //Create new pairs of refresh and access tokens
276 | ts, createErr := CreateToken(userId)
277 | if createErr != nil {
278 | c.JSON(http.StatusForbidden, createErr.Error())
279 | return
280 | }
281 | //save the tokens metadata to redis
282 | saveErr := CreateAuth(userId, ts)
283 | if saveErr != nil {
284 | c.JSON(http.StatusForbidden, saveErr.Error())
285 | return
286 | }
287 | tokens := map[string]string{
288 | "access_token": ts.AccessToken,
289 | "refresh_token": ts.RefreshToken,
290 | }
291 | c.JSON(http.StatusCreated, tokens)
292 | } else {
293 | c.JSON(http.StatusUnauthorized, "refresh expired")
294 | }
295 | }
296 |
297 | func DeleteTokens(authD *AccessDetails) error {
298 | client := common.GetClient()
299 |
300 | //get the refresh uuid
301 | refreshUuid := fmt.Sprintf("%s++%d", authD.AccessUuid, authD.UserId)
302 |
303 | //delete access token
304 | deletedAt, err := client.Del(authD.AccessUuid).Result()
305 | if err != nil {
306 | return err
307 | }
308 |
309 | //delete refresh token
310 | deletedRt, err := client.Del(refreshUuid).Result()
311 | if err != nil {
312 | return err
313 | }
314 |
315 | //When the record is deleted, the return value is 1
316 | if deletedAt != 1 || deletedRt != 1 {
317 | return errors.New("something went wrong")
318 | }
319 |
320 | return nil
321 | }
322 |
--------------------------------------------------------------------------------
/handler/todo.go:
--------------------------------------------------------------------------------
1 | package handler
2 |
3 | import (
4 | "fmt"
5 | "net/http"
6 |
7 | "github.com/gin-gonic/gin"
8 | "github.com/golang-crew/Bolierplate-JWT-Auth/requests"
9 | )
10 |
11 | func CreateTodo(c *gin.Context) {
12 | var td requests.Todo
13 | if err := c.ShouldBindJSON(&td); err != nil {
14 | c.JSON(http.StatusUnprocessableEntity, "invalid json")
15 | return
16 | }
17 | fmt.Println(td)
18 |
19 | //Extract the access token metadata
20 | metadata, err := ExtractTokenMetadata(c.Request)
21 | if err != nil {
22 | c.JSON(http.StatusUnauthorized, "unauthorized")
23 | return
24 | }
25 |
26 | userid, err := FetchAuth(metadata)
27 | if err != nil {
28 | c.JSON(http.StatusUnauthorized, err.Error())
29 | return
30 | }
31 | td.UserID = userid
32 | //you can proceed to save the Todo to a database
33 | //but we will just return it to the caller:
34 |
35 | c.JSON(http.StatusCreated, td)
36 | }
37 |
--------------------------------------------------------------------------------
/main.go:
--------------------------------------------------------------------------------
1 | package main
2 |
3 | import (
4 | "log"
5 |
6 | "github.com/gin-gonic/gin"
7 | "github.com/golang-crew/Bolierplate-JWT-Auth/common"
8 | "github.com/golang-crew/Bolierplate-JWT-Auth/models"
9 | "github.com/golang-crew/Bolierplate-JWT-Auth/routers"
10 | "github.com/spf13/viper"
11 | )
12 |
13 | func viperInit() {
14 | viper.SetConfigFile(`./config.json`)
15 | err := viper.ReadInConfig()
16 | if err != nil {
17 | panic(err)
18 | }
19 | }
20 |
21 | func main() {
22 | r := gin.Default()
23 |
24 | viperInit()
25 | common.RedisInit()
26 | models.Init()
27 | defer models.CloseDB()
28 | routers.Init(r)
29 |
30 | log.Fatal(r.Run(":8080"))
31 | }
32 |
--------------------------------------------------------------------------------
/models/autogenerated_todo.go:
--------------------------------------------------------------------------------
1 | // Code generated by go-queryset. DO NOT EDIT.
2 | package models
3 |
4 | import (
5 | "errors"
6 | "fmt"
7 | "strings"
8 |
9 | "github.com/jinzhu/gorm"
10 | )
11 |
12 | // ===== BEGIN of all query sets
13 |
14 | // ===== BEGIN of query set TodoQuerySet
15 |
16 | // TodoQuerySet is an queryset type for Todo
17 | type TodoQuerySet struct {
18 | db *gorm.DB
19 | }
20 |
21 | // NewTodoQuerySet constructs new TodoQuerySet
22 | func NewTodoQuerySet(db *gorm.DB) TodoQuerySet {
23 | return TodoQuerySet{
24 | db: db.Model(&Todo{}),
25 | }
26 | }
27 |
28 | func (qs TodoQuerySet) w(db *gorm.DB) TodoQuerySet {
29 | return NewTodoQuerySet(db)
30 | }
31 |
32 | func (qs TodoQuerySet) Select(fields ...TodoDBSchemaField) TodoQuerySet {
33 | names := []string{}
34 | for _, f := range fields {
35 | names = append(names, f.String())
36 | }
37 |
38 | return qs.w(qs.db.Select(strings.Join(names, ",")))
39 | }
40 |
41 | // Create is an autogenerated method
42 | // nolint: dupl
43 | func (o *Todo) Create(db *gorm.DB) error {
44 | return db.Create(o).Error
45 | }
46 |
47 | // Delete is an autogenerated method
48 | // nolint: dupl
49 | func (o *Todo) Delete(db *gorm.DB) error {
50 | return db.Delete(o).Error
51 | }
52 |
53 | // All is an autogenerated method
54 | // nolint: dupl
55 | func (qs TodoQuerySet) All(ret *[]Todo) error {
56 | return qs.db.Find(ret).Error
57 | }
58 |
59 | // Count is an autogenerated method
60 | // nolint: dupl
61 | func (qs TodoQuerySet) Count() (int, error) {
62 | var count int
63 | err := qs.db.Count(&count).Error
64 | return count, err
65 | }
66 |
67 | // Delete is an autogenerated method
68 | // nolint: dupl
69 | func (qs TodoQuerySet) Delete() error {
70 | return qs.db.Delete(Todo{}).Error
71 | }
72 |
73 | // DeleteNum is an autogenerated method
74 | // nolint: dupl
75 | func (qs TodoQuerySet) DeleteNum() (int64, error) {
76 | db := qs.db.Delete(Todo{})
77 | return db.RowsAffected, db.Error
78 | }
79 |
80 | // DeleteNumUnscoped is an autogenerated method
81 | // nolint: dupl
82 | func (qs TodoQuerySet) DeleteNumUnscoped() (int64, error) {
83 | db := qs.db.Unscoped().Delete(Todo{})
84 | return db.RowsAffected, db.Error
85 | }
86 |
87 | // GetDB is an autogenerated method
88 | // nolint: dupl
89 | func (qs TodoQuerySet) GetDB() *gorm.DB {
90 | return qs.db
91 | }
92 |
93 | // GetUpdater is an autogenerated method
94 | // nolint: dupl
95 | func (qs TodoQuerySet) GetUpdater() TodoUpdater {
96 | return NewTodoUpdater(qs.db)
97 | }
98 |
99 | // Limit is an autogenerated method
100 | // nolint: dupl
101 | func (qs TodoQuerySet) Limit(limit int) TodoQuerySet {
102 | return qs.w(qs.db.Limit(limit))
103 | }
104 |
105 | // Offset is an autogenerated method
106 | // nolint: dupl
107 | func (qs TodoQuerySet) Offset(offset int) TodoQuerySet {
108 | return qs.w(qs.db.Offset(offset))
109 | }
110 |
111 | // One is used to retrieve one result. It returns gorm.ErrRecordNotFound
112 | // if nothing was fetched
113 | func (qs TodoQuerySet) One(ret *Todo) error {
114 | return qs.db.First(ret).Error
115 | }
116 |
117 | // OrderAscByTitle is an autogenerated method
118 | // nolint: dupl
119 | func (qs TodoQuerySet) OrderAscByTitle() TodoQuerySet {
120 | return qs.w(qs.db.Order("title ASC"))
121 | }
122 |
123 | // OrderAscByUserID is an autogenerated method
124 | // nolint: dupl
125 | func (qs TodoQuerySet) OrderAscByUserID() TodoQuerySet {
126 | return qs.w(qs.db.Order("user_id ASC"))
127 | }
128 |
129 | // OrderDescByTitle is an autogenerated method
130 | // nolint: dupl
131 | func (qs TodoQuerySet) OrderDescByTitle() TodoQuerySet {
132 | return qs.w(qs.db.Order("title DESC"))
133 | }
134 |
135 | // OrderDescByUserID is an autogenerated method
136 | // nolint: dupl
137 | func (qs TodoQuerySet) OrderDescByUserID() TodoQuerySet {
138 | return qs.w(qs.db.Order("user_id DESC"))
139 | }
140 |
141 | // TitleEq is an autogenerated method
142 | // nolint: dupl
143 | func (qs TodoQuerySet) TitleEq(title string) TodoQuerySet {
144 | return qs.w(qs.db.Where("title = ?", title))
145 | }
146 |
147 | // TitleGt is an autogenerated method
148 | // nolint: dupl
149 | func (qs TodoQuerySet) TitleGt(title string) TodoQuerySet {
150 | return qs.w(qs.db.Where("title > ?", title))
151 | }
152 |
153 | // TitleGte is an autogenerated method
154 | // nolint: dupl
155 | func (qs TodoQuerySet) TitleGte(title string) TodoQuerySet {
156 | return qs.w(qs.db.Where("title >= ?", title))
157 | }
158 |
159 | // TitleIn is an autogenerated method
160 | // nolint: dupl
161 | func (qs TodoQuerySet) TitleIn(title ...string) TodoQuerySet {
162 | if len(title) == 0 {
163 | qs.db.AddError(errors.New("must at least pass one title in TitleIn"))
164 | return qs.w(qs.db)
165 | }
166 | return qs.w(qs.db.Where("title IN (?)", title))
167 | }
168 |
169 | // TitleLike is an autogenerated method
170 | // nolint: dupl
171 | func (qs TodoQuerySet) TitleLike(title string) TodoQuerySet {
172 | return qs.w(qs.db.Where("title LIKE ?", title))
173 | }
174 |
175 | // TitleLt is an autogenerated method
176 | // nolint: dupl
177 | func (qs TodoQuerySet) TitleLt(title string) TodoQuerySet {
178 | return qs.w(qs.db.Where("title < ?", title))
179 | }
180 |
181 | // TitleLte is an autogenerated method
182 | // nolint: dupl
183 | func (qs TodoQuerySet) TitleLte(title string) TodoQuerySet {
184 | return qs.w(qs.db.Where("title <= ?", title))
185 | }
186 |
187 | // TitleNe is an autogenerated method
188 | // nolint: dupl
189 | func (qs TodoQuerySet) TitleNe(title string) TodoQuerySet {
190 | return qs.w(qs.db.Where("title != ?", title))
191 | }
192 |
193 | // TitleNotIn is an autogenerated method
194 | // nolint: dupl
195 | func (qs TodoQuerySet) TitleNotIn(title ...string) TodoQuerySet {
196 | if len(title) == 0 {
197 | qs.db.AddError(errors.New("must at least pass one title in TitleNotIn"))
198 | return qs.w(qs.db)
199 | }
200 | return qs.w(qs.db.Where("title NOT IN (?)", title))
201 | }
202 |
203 | // TitleNotlike is an autogenerated method
204 | // nolint: dupl
205 | func (qs TodoQuerySet) TitleNotlike(title string) TodoQuerySet {
206 | return qs.w(qs.db.Where("title NOT LIKE ?", title))
207 | }
208 |
209 | // UserIDEq is an autogenerated method
210 | // nolint: dupl
211 | func (qs TodoQuerySet) UserIDEq(userID uint64) TodoQuerySet {
212 | return qs.w(qs.db.Where("user_id = ?", userID))
213 | }
214 |
215 | // UserIDGt is an autogenerated method
216 | // nolint: dupl
217 | func (qs TodoQuerySet) UserIDGt(userID uint64) TodoQuerySet {
218 | return qs.w(qs.db.Where("user_id > ?", userID))
219 | }
220 |
221 | // UserIDGte is an autogenerated method
222 | // nolint: dupl
223 | func (qs TodoQuerySet) UserIDGte(userID uint64) TodoQuerySet {
224 | return qs.w(qs.db.Where("user_id >= ?", userID))
225 | }
226 |
227 | // UserIDIn is an autogenerated method
228 | // nolint: dupl
229 | func (qs TodoQuerySet) UserIDIn(userID ...uint64) TodoQuerySet {
230 | if len(userID) == 0 {
231 | qs.db.AddError(errors.New("must at least pass one userID in UserIDIn"))
232 | return qs.w(qs.db)
233 | }
234 | return qs.w(qs.db.Where("user_id IN (?)", userID))
235 | }
236 |
237 | // UserIDLt is an autogenerated method
238 | // nolint: dupl
239 | func (qs TodoQuerySet) UserIDLt(userID uint64) TodoQuerySet {
240 | return qs.w(qs.db.Where("user_id < ?", userID))
241 | }
242 |
243 | // UserIDLte is an autogenerated method
244 | // nolint: dupl
245 | func (qs TodoQuerySet) UserIDLte(userID uint64) TodoQuerySet {
246 | return qs.w(qs.db.Where("user_id <= ?", userID))
247 | }
248 |
249 | // UserIDNe is an autogenerated method
250 | // nolint: dupl
251 | func (qs TodoQuerySet) UserIDNe(userID uint64) TodoQuerySet {
252 | return qs.w(qs.db.Where("user_id != ?", userID))
253 | }
254 |
255 | // UserIDNotIn is an autogenerated method
256 | // nolint: dupl
257 | func (qs TodoQuerySet) UserIDNotIn(userID ...uint64) TodoQuerySet {
258 | if len(userID) == 0 {
259 | qs.db.AddError(errors.New("must at least pass one userID in UserIDNotIn"))
260 | return qs.w(qs.db)
261 | }
262 | return qs.w(qs.db.Where("user_id NOT IN (?)", userID))
263 | }
264 |
265 | // SetTitle is an autogenerated method
266 | // nolint: dupl
267 | func (u TodoUpdater) SetTitle(title string) TodoUpdater {
268 | u.fields[string(TodoDBSchema.Title)] = title
269 | return u
270 | }
271 |
272 | // SetUserID is an autogenerated method
273 | // nolint: dupl
274 | func (u TodoUpdater) SetUserID(userID uint64) TodoUpdater {
275 | u.fields[string(TodoDBSchema.UserID)] = userID
276 | return u
277 | }
278 |
279 | // Update is an autogenerated method
280 | // nolint: dupl
281 | func (u TodoUpdater) Update() error {
282 | return u.db.Updates(u.fields).Error
283 | }
284 |
285 | // UpdateNum is an autogenerated method
286 | // nolint: dupl
287 | func (u TodoUpdater) UpdateNum() (int64, error) {
288 | db := u.db.Updates(u.fields)
289 | return db.RowsAffected, db.Error
290 | }
291 |
292 | // ===== END of query set TodoQuerySet
293 |
294 | // ===== BEGIN of Todo modifiers
295 |
296 | // TodoDBSchemaField describes database schema field. It requires for method 'Update'
297 | type TodoDBSchemaField string
298 |
299 | // String method returns string representation of field.
300 | // nolint: dupl
301 | func (f TodoDBSchemaField) String() string {
302 | return string(f)
303 | }
304 |
305 | // TodoDBSchema stores db field names of Todo
306 | var TodoDBSchema = struct {
307 | UserID TodoDBSchemaField
308 | Title TodoDBSchemaField
309 | }{
310 |
311 | UserID: TodoDBSchemaField("user_id"),
312 | Title: TodoDBSchemaField("title"),
313 | }
314 |
315 | // Update updates Todo fields by primary key
316 | // nolint: dupl
317 | func (o *Todo) Update(db *gorm.DB, fields ...TodoDBSchemaField) error {
318 | dbNameToFieldName := map[string]interface{}{
319 | "user_id": o.UserID,
320 | "title": o.Title,
321 | }
322 | u := map[string]interface{}{}
323 | for _, f := range fields {
324 | fs := f.String()
325 | u[fs] = dbNameToFieldName[fs]
326 | }
327 | if err := db.Model(o).Updates(u).Error; err != nil {
328 | if err == gorm.ErrRecordNotFound {
329 | return err
330 | }
331 |
332 | return fmt.Errorf("can't update Todo %v fields %v: %s",
333 | o, fields, err)
334 | }
335 |
336 | return nil
337 | }
338 |
339 | // TodoUpdater is an Todo updates manager
340 | type TodoUpdater struct {
341 | fields map[string]interface{}
342 | db *gorm.DB
343 | }
344 |
345 | // NewTodoUpdater creates new Todo updater
346 | // nolint: dupl
347 | func NewTodoUpdater(db *gorm.DB) TodoUpdater {
348 | return TodoUpdater{
349 | fields: map[string]interface{}{},
350 | db: db.Model(&Todo{}),
351 | }
352 | }
353 |
354 | // ===== END of Todo modifiers
355 |
356 | // ===== END of all query sets
357 |
--------------------------------------------------------------------------------
/models/autogenerated_users.go:
--------------------------------------------------------------------------------
1 | // Code generated by go-queryset. DO NOT EDIT.
2 | package models
3 |
4 | import (
5 | "errors"
6 | "fmt"
7 | "strings"
8 |
9 | "github.com/jinzhu/gorm"
10 | )
11 |
12 | // ===== BEGIN of all query sets
13 |
14 | // ===== BEGIN of query set UsersQuerySet
15 |
16 | // UsersQuerySet is an queryset type for Users
17 | type UsersQuerySet struct {
18 | db *gorm.DB
19 | }
20 |
21 | // NewUsersQuerySet constructs new UsersQuerySet
22 | func NewUsersQuerySet(db *gorm.DB) UsersQuerySet {
23 | return UsersQuerySet{
24 | db: db.Model(&Users{}),
25 | }
26 | }
27 |
28 | func (qs UsersQuerySet) w(db *gorm.DB) UsersQuerySet {
29 | return NewUsersQuerySet(db)
30 | }
31 |
32 | func (qs UsersQuerySet) Select(fields ...UsersDBSchemaField) UsersQuerySet {
33 | names := []string{}
34 | for _, f := range fields {
35 | names = append(names, f.String())
36 | }
37 |
38 | return qs.w(qs.db.Select(strings.Join(names, ",")))
39 | }
40 |
41 | // Create is an autogenerated method
42 | // nolint: dupl
43 | func (o *Users) Create(db *gorm.DB) error {
44 | return db.Create(o).Error
45 | }
46 |
47 | // Delete is an autogenerated method
48 | // nolint: dupl
49 | func (o *Users) Delete(db *gorm.DB) error {
50 | return db.Delete(o).Error
51 | }
52 |
53 | // All is an autogenerated method
54 | // nolint: dupl
55 | func (qs UsersQuerySet) All(ret *[]Users) error {
56 | return qs.db.Find(ret).Error
57 | }
58 |
59 | // Count is an autogenerated method
60 | // nolint: dupl
61 | func (qs UsersQuerySet) Count() (int, error) {
62 | var count int
63 | err := qs.db.Count(&count).Error
64 | return count, err
65 | }
66 |
67 | // Delete is an autogenerated method
68 | // nolint: dupl
69 | func (qs UsersQuerySet) Delete() error {
70 | return qs.db.Delete(Users{}).Error
71 | }
72 |
73 | // DeleteNum is an autogenerated method
74 | // nolint: dupl
75 | func (qs UsersQuerySet) DeleteNum() (int64, error) {
76 | db := qs.db.Delete(Users{})
77 | return db.RowsAffected, db.Error
78 | }
79 |
80 | // DeleteNumUnscoped is an autogenerated method
81 | // nolint: dupl
82 | func (qs UsersQuerySet) DeleteNumUnscoped() (int64, error) {
83 | db := qs.db.Unscoped().Delete(Users{})
84 | return db.RowsAffected, db.Error
85 | }
86 |
87 | // GetDB is an autogenerated method
88 | // nolint: dupl
89 | func (qs UsersQuerySet) GetDB() *gorm.DB {
90 | return qs.db
91 | }
92 |
93 | // GetUpdater is an autogenerated method
94 | // nolint: dupl
95 | func (qs UsersQuerySet) GetUpdater() UsersUpdater {
96 | return NewUsersUpdater(qs.db)
97 | }
98 |
99 | // IDEq is an autogenerated method
100 | // nolint: dupl
101 | func (qs UsersQuerySet) IDEq(ID uint64) UsersQuerySet {
102 | return qs.w(qs.db.Where("id = ?", ID))
103 | }
104 |
105 | // IDGt is an autogenerated method
106 | // nolint: dupl
107 | func (qs UsersQuerySet) IDGt(ID uint64) UsersQuerySet {
108 | return qs.w(qs.db.Where("id > ?", ID))
109 | }
110 |
111 | // IDGte is an autogenerated method
112 | // nolint: dupl
113 | func (qs UsersQuerySet) IDGte(ID uint64) UsersQuerySet {
114 | return qs.w(qs.db.Where("id >= ?", ID))
115 | }
116 |
117 | // IDIn is an autogenerated method
118 | // nolint: dupl
119 | func (qs UsersQuerySet) IDIn(ID ...uint64) UsersQuerySet {
120 | if len(ID) == 0 {
121 | qs.db.AddError(errors.New("must at least pass one ID in IDIn"))
122 | return qs.w(qs.db)
123 | }
124 | return qs.w(qs.db.Where("id IN (?)", ID))
125 | }
126 |
127 | // IDLt is an autogenerated method
128 | // nolint: dupl
129 | func (qs UsersQuerySet) IDLt(ID uint64) UsersQuerySet {
130 | return qs.w(qs.db.Where("id < ?", ID))
131 | }
132 |
133 | // IDLte is an autogenerated method
134 | // nolint: dupl
135 | func (qs UsersQuerySet) IDLte(ID uint64) UsersQuerySet {
136 | return qs.w(qs.db.Where("id <= ?", ID))
137 | }
138 |
139 | // IDNe is an autogenerated method
140 | // nolint: dupl
141 | func (qs UsersQuerySet) IDNe(ID uint64) UsersQuerySet {
142 | return qs.w(qs.db.Where("id != ?", ID))
143 | }
144 |
145 | // IDNotIn is an autogenerated method
146 | // nolint: dupl
147 | func (qs UsersQuerySet) IDNotIn(ID ...uint64) UsersQuerySet {
148 | if len(ID) == 0 {
149 | qs.db.AddError(errors.New("must at least pass one ID in IDNotIn"))
150 | return qs.w(qs.db)
151 | }
152 | return qs.w(qs.db.Where("id NOT IN (?)", ID))
153 | }
154 |
155 | // Limit is an autogenerated method
156 | // nolint: dupl
157 | func (qs UsersQuerySet) Limit(limit int) UsersQuerySet {
158 | return qs.w(qs.db.Limit(limit))
159 | }
160 |
161 | // Offset is an autogenerated method
162 | // nolint: dupl
163 | func (qs UsersQuerySet) Offset(offset int) UsersQuerySet {
164 | return qs.w(qs.db.Offset(offset))
165 | }
166 |
167 | // One is used to retrieve one result. It returns gorm.ErrRecordNotFound
168 | // if nothing was fetched
169 | func (qs UsersQuerySet) One(ret *Users) error {
170 | return qs.db.First(ret).Error
171 | }
172 |
173 | // OrderAscByID is an autogenerated method
174 | // nolint: dupl
175 | func (qs UsersQuerySet) OrderAscByID() UsersQuerySet {
176 | return qs.w(qs.db.Order("id ASC"))
177 | }
178 |
179 | // OrderAscByPassword is an autogenerated method
180 | // nolint: dupl
181 | func (qs UsersQuerySet) OrderAscByPassword() UsersQuerySet {
182 | return qs.w(qs.db.Order("password ASC"))
183 | }
184 |
185 | // OrderAscByUsername is an autogenerated method
186 | // nolint: dupl
187 | func (qs UsersQuerySet) OrderAscByUsername() UsersQuerySet {
188 | return qs.w(qs.db.Order("username ASC"))
189 | }
190 |
191 | // OrderDescByID is an autogenerated method
192 | // nolint: dupl
193 | func (qs UsersQuerySet) OrderDescByID() UsersQuerySet {
194 | return qs.w(qs.db.Order("id DESC"))
195 | }
196 |
197 | // OrderDescByPassword is an autogenerated method
198 | // nolint: dupl
199 | func (qs UsersQuerySet) OrderDescByPassword() UsersQuerySet {
200 | return qs.w(qs.db.Order("password DESC"))
201 | }
202 |
203 | // OrderDescByUsername is an autogenerated method
204 | // nolint: dupl
205 | func (qs UsersQuerySet) OrderDescByUsername() UsersQuerySet {
206 | return qs.w(qs.db.Order("username DESC"))
207 | }
208 |
209 | // PasswordEq is an autogenerated method
210 | // nolint: dupl
211 | func (qs UsersQuerySet) PasswordEq(password string) UsersQuerySet {
212 | return qs.w(qs.db.Where("password = ?", password))
213 | }
214 |
215 | // PasswordGt is an autogenerated method
216 | // nolint: dupl
217 | func (qs UsersQuerySet) PasswordGt(password string) UsersQuerySet {
218 | return qs.w(qs.db.Where("password > ?", password))
219 | }
220 |
221 | // PasswordGte is an autogenerated method
222 | // nolint: dupl
223 | func (qs UsersQuerySet) PasswordGte(password string) UsersQuerySet {
224 | return qs.w(qs.db.Where("password >= ?", password))
225 | }
226 |
227 | // PasswordIn is an autogenerated method
228 | // nolint: dupl
229 | func (qs UsersQuerySet) PasswordIn(password ...string) UsersQuerySet {
230 | if len(password) == 0 {
231 | qs.db.AddError(errors.New("must at least pass one password in PasswordIn"))
232 | return qs.w(qs.db)
233 | }
234 | return qs.w(qs.db.Where("password IN (?)", password))
235 | }
236 |
237 | // PasswordLike is an autogenerated method
238 | // nolint: dupl
239 | func (qs UsersQuerySet) PasswordLike(password string) UsersQuerySet {
240 | return qs.w(qs.db.Where("password LIKE ?", password))
241 | }
242 |
243 | // PasswordLt is an autogenerated method
244 | // nolint: dupl
245 | func (qs UsersQuerySet) PasswordLt(password string) UsersQuerySet {
246 | return qs.w(qs.db.Where("password < ?", password))
247 | }
248 |
249 | // PasswordLte is an autogenerated method
250 | // nolint: dupl
251 | func (qs UsersQuerySet) PasswordLte(password string) UsersQuerySet {
252 | return qs.w(qs.db.Where("password <= ?", password))
253 | }
254 |
255 | // PasswordNe is an autogenerated method
256 | // nolint: dupl
257 | func (qs UsersQuerySet) PasswordNe(password string) UsersQuerySet {
258 | return qs.w(qs.db.Where("password != ?", password))
259 | }
260 |
261 | // PasswordNotIn is an autogenerated method
262 | // nolint: dupl
263 | func (qs UsersQuerySet) PasswordNotIn(password ...string) UsersQuerySet {
264 | if len(password) == 0 {
265 | qs.db.AddError(errors.New("must at least pass one password in PasswordNotIn"))
266 | return qs.w(qs.db)
267 | }
268 | return qs.w(qs.db.Where("password NOT IN (?)", password))
269 | }
270 |
271 | // PasswordNotlike is an autogenerated method
272 | // nolint: dupl
273 | func (qs UsersQuerySet) PasswordNotlike(password string) UsersQuerySet {
274 | return qs.w(qs.db.Where("password NOT LIKE ?", password))
275 | }
276 |
277 | // UsernameEq is an autogenerated method
278 | // nolint: dupl
279 | func (qs UsersQuerySet) UsernameEq(username string) UsersQuerySet {
280 | return qs.w(qs.db.Where("username = ?", username))
281 | }
282 |
283 | // UsernameGt is an autogenerated method
284 | // nolint: dupl
285 | func (qs UsersQuerySet) UsernameGt(username string) UsersQuerySet {
286 | return qs.w(qs.db.Where("username > ?", username))
287 | }
288 |
289 | // UsernameGte is an autogenerated method
290 | // nolint: dupl
291 | func (qs UsersQuerySet) UsernameGte(username string) UsersQuerySet {
292 | return qs.w(qs.db.Where("username >= ?", username))
293 | }
294 |
295 | // UsernameIn is an autogenerated method
296 | // nolint: dupl
297 | func (qs UsersQuerySet) UsernameIn(username ...string) UsersQuerySet {
298 | if len(username) == 0 {
299 | qs.db.AddError(errors.New("must at least pass one username in UsernameIn"))
300 | return qs.w(qs.db)
301 | }
302 | return qs.w(qs.db.Where("username IN (?)", username))
303 | }
304 |
305 | // UsernameLike is an autogenerated method
306 | // nolint: dupl
307 | func (qs UsersQuerySet) UsernameLike(username string) UsersQuerySet {
308 | return qs.w(qs.db.Where("username LIKE ?", username))
309 | }
310 |
311 | // UsernameLt is an autogenerated method
312 | // nolint: dupl
313 | func (qs UsersQuerySet) UsernameLt(username string) UsersQuerySet {
314 | return qs.w(qs.db.Where("username < ?", username))
315 | }
316 |
317 | // UsernameLte is an autogenerated method
318 | // nolint: dupl
319 | func (qs UsersQuerySet) UsernameLte(username string) UsersQuerySet {
320 | return qs.w(qs.db.Where("username <= ?", username))
321 | }
322 |
323 | // UsernameNe is an autogenerated method
324 | // nolint: dupl
325 | func (qs UsersQuerySet) UsernameNe(username string) UsersQuerySet {
326 | return qs.w(qs.db.Where("username != ?", username))
327 | }
328 |
329 | // UsernameNotIn is an autogenerated method
330 | // nolint: dupl
331 | func (qs UsersQuerySet) UsernameNotIn(username ...string) UsersQuerySet {
332 | if len(username) == 0 {
333 | qs.db.AddError(errors.New("must at least pass one username in UsernameNotIn"))
334 | return qs.w(qs.db)
335 | }
336 | return qs.w(qs.db.Where("username NOT IN (?)", username))
337 | }
338 |
339 | // UsernameNotlike is an autogenerated method
340 | // nolint: dupl
341 | func (qs UsersQuerySet) UsernameNotlike(username string) UsersQuerySet {
342 | return qs.w(qs.db.Where("username NOT LIKE ?", username))
343 | }
344 |
345 | // SetID is an autogenerated method
346 | // nolint: dupl
347 | func (u UsersUpdater) SetID(ID uint64) UsersUpdater {
348 | u.fields[string(UsersDBSchema.ID)] = ID
349 | return u
350 | }
351 |
352 | // SetPassword is an autogenerated method
353 | // nolint: dupl
354 | func (u UsersUpdater) SetPassword(password string) UsersUpdater {
355 | u.fields[string(UsersDBSchema.Password)] = password
356 | return u
357 | }
358 |
359 | // SetUsername is an autogenerated method
360 | // nolint: dupl
361 | func (u UsersUpdater) SetUsername(username string) UsersUpdater {
362 | u.fields[string(UsersDBSchema.Username)] = username
363 | return u
364 | }
365 |
366 | // Update is an autogenerated method
367 | // nolint: dupl
368 | func (u UsersUpdater) Update() error {
369 | return u.db.Updates(u.fields).Error
370 | }
371 |
372 | // UpdateNum is an autogenerated method
373 | // nolint: dupl
374 | func (u UsersUpdater) UpdateNum() (int64, error) {
375 | db := u.db.Updates(u.fields)
376 | return db.RowsAffected, db.Error
377 | }
378 |
379 | // ===== END of query set UsersQuerySet
380 |
381 | // ===== BEGIN of Users modifiers
382 |
383 | // UsersDBSchemaField describes database schema field. It requires for method 'Update'
384 | type UsersDBSchemaField string
385 |
386 | // String method returns string representation of field.
387 | // nolint: dupl
388 | func (f UsersDBSchemaField) String() string {
389 | return string(f)
390 | }
391 |
392 | // UsersDBSchema stores db field names of Users
393 | var UsersDBSchema = struct {
394 | ID UsersDBSchemaField
395 | Username UsersDBSchemaField
396 | Password UsersDBSchemaField
397 | }{
398 |
399 | ID: UsersDBSchemaField("id"),
400 | Username: UsersDBSchemaField("username"),
401 | Password: UsersDBSchemaField("password"),
402 | }
403 |
404 | // Update updates Users fields by primary key
405 | // nolint: dupl
406 | func (o *Users) Update(db *gorm.DB, fields ...UsersDBSchemaField) error {
407 | dbNameToFieldName := map[string]interface{}{
408 | "id": o.ID,
409 | "username": o.Username,
410 | "password": o.Password,
411 | }
412 | u := map[string]interface{}{}
413 | for _, f := range fields {
414 | fs := f.String()
415 | u[fs] = dbNameToFieldName[fs]
416 | }
417 | if err := db.Model(o).Updates(u).Error; err != nil {
418 | if err == gorm.ErrRecordNotFound {
419 | return err
420 | }
421 |
422 | return fmt.Errorf("can't update Users %v fields %v: %s",
423 | o, fields, err)
424 | }
425 |
426 | return nil
427 | }
428 |
429 | // UsersUpdater is an Users updates manager
430 | type UsersUpdater struct {
431 | fields map[string]interface{}
432 | db *gorm.DB
433 | }
434 |
435 | // NewUsersUpdater creates new Users updater
436 | // nolint: dupl
437 | func NewUsersUpdater(db *gorm.DB) UsersUpdater {
438 | return UsersUpdater{
439 | fields: map[string]interface{}{},
440 | db: db.Model(&Users{}),
441 | }
442 | }
443 |
444 | // ===== END of Users modifiers
445 |
446 | // ===== END of all query sets
447 |
--------------------------------------------------------------------------------
/models/gen_queryset.sh:
--------------------------------------------------------------------------------
1 | #!/bin/bash
2 | rm -f autogenerated_*
3 |
4 | FILES="./*.go"
5 | for f in $FILES
6 | do
7 | goqueryset -in ${f:2}
8 | done
9 |
--------------------------------------------------------------------------------
/models/setup.go:
--------------------------------------------------------------------------------
1 | package models
2 |
3 | import (
4 | "fmt"
5 | "log"
6 |
7 | _ "github.com/go-sql-driver/mysql"
8 | "github.com/jinzhu/gorm"
9 | _ "github.com/jinzhu/gorm/dialects/mysql"
10 | "github.com/spf13/viper"
11 | )
12 |
13 | var gGormDB *gorm.DB
14 |
15 | func CloseDB() {
16 | if gGormDB != nil {
17 | gGormDB.Close()
18 | }
19 | }
20 |
21 | func Init() {
22 | db, err := gorm.Open("mysql", dbConnString())
23 | if err != nil {
24 | log.Println("[DB Error] ", err)
25 | panic(err)
26 | }
27 | gGormDB = db
28 |
29 | log.Println("[DB] Start DB Migration ... ")
30 | log.Println("[DB] Start DB ... ")
31 |
32 | if err := db.AutoMigrate(&Users{}, &Todo{}).Error; err != nil {
33 | log.Println("[DB Error] ", err)
34 | panic(err)
35 | }
36 | }
37 |
38 | func dbConnString() (dbConnString string) {
39 | dbHost := viper.GetString(`database.host`)
40 | dbHost = "127.0.0.1"
41 | dbUser := viper.GetString(`database.user`)
42 | dbPass := viper.GetString(`database.pass`)
43 | dbName := viper.GetString(`database.name`)
44 |
45 | dbConnString = fmt.Sprintf("%s:%s@tcp(%s)/%s?charset=utf8mb4&parseTime=True&loc=Local",
46 | dbUser, dbPass, dbHost, dbName,
47 | )
48 |
49 | return
50 | }
51 |
--------------------------------------------------------------------------------
/models/todo.go:
--------------------------------------------------------------------------------
1 | package models
2 |
3 | // gen:qs
4 | type Todo struct {
5 | UserID uint64 `json:"user_id"`
6 | Title string `json:"title"`
7 | }
8 |
--------------------------------------------------------------------------------
/models/users.go:
--------------------------------------------------------------------------------
1 | package models
2 |
3 | // gen:qs
4 | type Users struct {
5 | ID uint64 `json:"id"`
6 | Username string `json:"username"`
7 | Password string `json:"password"`
8 | }
9 |
--------------------------------------------------------------------------------
/requests/todo.go:
--------------------------------------------------------------------------------
1 | package requests
2 |
3 | type Todo struct {
4 | UserID uint64 `json:"user_id"`
5 | Title string `json:"title"`
6 | }
7 |
--------------------------------------------------------------------------------
/routers/routers.go:
--------------------------------------------------------------------------------
1 | package routers
2 |
3 | import (
4 | "github.com/gin-gonic/gin"
5 | router "github.com/golang-crew/Bolierplate-JWT-Auth/routers/v1"
6 |
7 | ginSwagger "github.com/swaggo/gin-swagger"
8 | swaggerFiles "github.com/swaggo/gin-swagger/swaggerFiles"
9 | )
10 |
11 | func Init(r *gin.Engine) {
12 | r.GET("/swagger/*any", ginSwagger.WrapHandler(swaggerFiles.Handler))
13 | v1 := r.Group("/v1")
14 | {
15 | {
16 | router.ApplyRoutes(v1)
17 | }
18 | }
19 |
20 | }
21 |
--------------------------------------------------------------------------------
/routers/v1/router.go:
--------------------------------------------------------------------------------
1 | package router
2 |
3 | import (
4 | "github.com/gin-gonic/gin"
5 | "github.com/golang-crew/Bolierplate-JWT-Auth/handler"
6 | )
7 |
8 | func ApplyRoutes(r *gin.RouterGroup) {
9 | test := r.Group("/ping")
10 | {
11 | test.GET("/", func(c *gin.Context) {
12 | c.String(200, "pong")
13 | })
14 | }
15 | auth := r.Group("/auth")
16 | {
17 | auth.POST("/login", handler.Login)
18 | auth.POST("/logout", handler.Logout)
19 | }
20 | todo := r.Group("/todo")
21 | {
22 | todo.POST("/todo", handler.CreateTodo)
23 | }
24 | }
25 |
--------------------------------------------------------------------------------