diff --git a/go.mod b/go.mod index 8b7d858..4c55ed1 100644 --- a/go.mod +++ b/go.mod @@ -4,8 +4,8 @@ go 1.19 require ( github.com/go-redis/redis/v8 v8.11.5 - github.com/gofiber/fiber/v2 v2.42.0 - github.com/mcstatus-io/mcutil v0.0.0-20230228062837-c41a7bd81011 + github.com/gofiber/fiber/v2 v2.43.0 + github.com/mcstatus-io/mcutil v0.0.0-20230331002438-59f81e936131 gopkg.in/yaml.v3 v3.0.1 ) @@ -14,17 +14,17 @@ require ( github.com/cespare/xxhash/v2 v2.2.0 // indirect github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f // indirect github.com/google/uuid v1.3.0 // indirect - github.com/klauspost/compress v1.16.0 // indirect + github.com/klauspost/compress v1.16.3 // indirect github.com/mattn/go-colorable v0.1.13 // indirect - github.com/mattn/go-isatty v0.0.17 // indirect + github.com/mattn/go-isatty v0.0.18 // indirect github.com/mattn/go-runewidth v0.0.14 // indirect - github.com/philhofer/fwd v1.1.1 // indirect + github.com/philhofer/fwd v1.1.2 // indirect github.com/rivo/uniseg v0.4.4 // indirect github.com/savsgio/dictpool v0.0.0-20221023140959-7bf2e61cea94 // indirect - github.com/savsgio/gotils v0.0.0-20220530130905-52f3993e8d6d // indirect - github.com/tinylib/msgp v1.1.6 // indirect + github.com/savsgio/gotils v0.0.0-20230208104028-c358bd845dee // indirect + github.com/tinylib/msgp v1.1.8 // indirect github.com/valyala/bytebufferpool v1.0.0 // indirect - github.com/valyala/fasthttp v1.44.0 // indirect + github.com/valyala/fasthttp v1.45.0 // indirect github.com/valyala/tcplisten v1.0.0 // indirect - golang.org/x/sys v0.5.0 // indirect + golang.org/x/sys v0.6.0 // indirect ) diff --git a/go.sum b/go.sum index 41e322a..53546dd 100644 --- a/go.sum +++ b/go.sum @@ -1,9 +1,5 @@ -github.com/andybalholm/brotli v1.0.4 h1:V7DdXeJtZscaqfNuAdSRuRFzuiKlHSC/Zh3zl9qY3JY= -github.com/andybalholm/brotli v1.0.4/go.mod h1:fO7iG3H7G2nSZ7m0zPUDn85XEX2GTukHGRSepvi9Eig= github.com/andybalholm/brotli v1.0.5 h1:8uQZIdzKmjc/iuPu7O2ioW48L81FgatrcpfFmiq/cCs= github.com/andybalholm/brotli v1.0.5/go.mod h1:fO7iG3H7G2nSZ7m0zPUDn85XEX2GTukHGRSepvi9Eig= -github.com/cespare/xxhash/v2 v2.1.2 h1:YRXhKfTDauu4ajMg1TPgFO5jnlC2HCbmLXMcTG5cbYE= -github.com/cespare/xxhash/v2 v2.1.2/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= github.com/cespare/xxhash/v2 v2.2.0 h1:DC2CZ1Ep5Y4k3ZQ899DldepgrayRUGE6BBZ/cd9Cj44= github.com/cespare/xxhash/v2 v2.2.0/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f h1:lO4WD4F/rVNCu3HqELle0jiPLLBs70cWOduZpkS1E78= @@ -11,86 +7,90 @@ github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f/go.mod h1:cu github.com/fsnotify/fsnotify v1.4.9 h1:hsms1Qyu0jgnwNXIxa+/V/PDsU6CfLf6CNO8H7IWoS4= github.com/go-redis/redis/v8 v8.11.5 h1:AcZZR7igkdvfVmQTPnu9WE37LRrO/YrBH5zWyjDC0oI= github.com/go-redis/redis/v8 v8.11.5/go.mod h1:gREzHqY1hg6oD9ngVRbLStwAWKhA0FEgq8Jd4h5lpwo= -github.com/gofiber/fiber/v2 v2.40.1 h1:pc7n9VVpGIqNsvg9IPLQhyFEMJL8gCs1kneH5D1pIl4= -github.com/gofiber/fiber/v2 v2.40.1/go.mod h1:Gko04sLksnHbzLSRBFWPFdzM9Ws9pRxvvIaohJK1dsk= -github.com/gofiber/fiber/v2 v2.42.0 h1:Fnp7ybWvS+sjNQsFvkhf4G8OhXswvB6Vee8hM/LyS+8= -github.com/gofiber/fiber/v2 v2.42.0/go.mod h1:3+SGNjqMh5VQH5Vz2Wdi43zTIV16ktlFd3x3R6O1Zlc= +github.com/gofiber/fiber/v2 v2.43.0 h1:yit3E4kHf178B60p5CQBa/3v+WVuziWMa/G2ZNyLJB0= +github.com/gofiber/fiber/v2 v2.43.0/go.mod h1:mpS1ZNE5jU+u+BA4FbM+KKnUzJ4wzTK+FT2tG3tU+6I= github.com/google/uuid v1.3.0 h1:t6JiXgmwXMjEs8VusXIJk2BXHsn+wx8BZdTaoZ5fu7I= github.com/google/uuid v1.3.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= -github.com/klauspost/compress v1.15.9 h1:wKRjX6JRtDdrE9qwa4b/Cip7ACOshUI4smpCQanqjSY= -github.com/klauspost/compress v1.15.9/go.mod h1:PhcZ0MbTNciWF3rruxRgKxI5NkcHHrHUDtV4Yw2GlzU= -github.com/klauspost/compress v1.16.0 h1:iULayQNOReoYUe+1qtKOqw9CwJv3aNQu8ivo7lw1HU4= -github.com/klauspost/compress v1.16.0/go.mod h1:ntbaceVETuRiXiv4DpjP66DpAtAGkEQskQzEyD//IeE= +github.com/klauspost/compress v1.16.3 h1:XuJt9zzcnaz6a16/OU53ZjWp/v7/42WcR5t2a0PcNQY= +github.com/klauspost/compress v1.16.3/go.mod h1:ntbaceVETuRiXiv4DpjP66DpAtAGkEQskQzEyD//IeE= github.com/mattn/go-colorable v0.1.13 h1:fFA4WZxdEF4tXPZVKMLwD8oUnCTTo08duU7wxecdEvA= github.com/mattn/go-colorable v0.1.13/go.mod h1:7S9/ev0klgBDR4GtXTXX8a3vIGJpMovkB8vQcUbaXHg= -github.com/mattn/go-isatty v0.0.16 h1:bq3VjFmv/sOjHtdEhmkEV4x1AJtvUvOJ2PFAZ5+peKQ= github.com/mattn/go-isatty v0.0.16/go.mod h1:kYGgaQfpe5nmfYZH+SKPsOc2e4SrIfOl2e/yFXSvRLM= -github.com/mattn/go-isatty v0.0.17 h1:BTarxUcIeDqL27Mc+vyvdWYSL28zpIhv3RoTdsLMPng= -github.com/mattn/go-isatty v0.0.17/go.mod h1:kYGgaQfpe5nmfYZH+SKPsOc2e4SrIfOl2e/yFXSvRLM= +github.com/mattn/go-isatty v0.0.18 h1:DOKFKCQ7FNG2L1rbrmstDN4QVRdS89Nkh85u68Uwp98= +github.com/mattn/go-isatty v0.0.18/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y= github.com/mattn/go-runewidth v0.0.14 h1:+xnbZSEeDbOIg5/mE6JF0w6n9duR1l3/WmbinWVwUuU= github.com/mattn/go-runewidth v0.0.14/go.mod h1:Jdepj2loyihRzMpdS35Xk/zdY8IAYHsh153qUoGf23w= -github.com/mcstatus-io/mcutil v0.0.0-20230226191507-0c73abf4a4d2 h1:zR0nGnTWJe/SqA+SvMVtLqu+9Owlq3oUEpAQn8lcMCY= -github.com/mcstatus-io/mcutil v0.0.0-20230226191507-0c73abf4a4d2/go.mod h1:VUB87/x9EYITmQVXZO4eS+egaZOdvxId4IdpU4L5LoA= -github.com/mcstatus-io/mcutil v0.0.0-20230228062837-c41a7bd81011 h1:A15qdovJAC2QxtNmtMIoaIHBFDIAXGUC+Fmo3od/gL0= -github.com/mcstatus-io/mcutil v0.0.0-20230228062837-c41a7bd81011/go.mod h1:VUB87/x9EYITmQVXZO4eS+egaZOdvxId4IdpU4L5LoA= +github.com/mcstatus-io/mcutil v0.0.0-20230326225412-9e5254880e3c h1:/8+KhqSkpQR26FLuuxlV2Yw17T+9w9VlnrwuOgolPVo= +github.com/mcstatus-io/mcutil v0.0.0-20230326225412-9e5254880e3c/go.mod h1:VUB87/x9EYITmQVXZO4eS+egaZOdvxId4IdpU4L5LoA= +github.com/mcstatus-io/mcutil v0.0.0-20230331002438-59f81e936131 h1:h5aBODlRBQTHRAvjbdNdYWt98BlP5e2GijGJnSP5b7g= +github.com/mcstatus-io/mcutil v0.0.0-20230331002438-59f81e936131/go.mod h1:VUB87/x9EYITmQVXZO4eS+egaZOdvxId4IdpU4L5LoA= github.com/nxadm/tail v1.4.8 h1:nPr65rt6Y5JFSKQO7qToXr7pePgD6Gwiw05lkbyAQTE= github.com/onsi/ginkgo v1.16.5 h1:8xi0RTUf59SOSfEtZMvwTvXYMzG4gV23XVHOZiXNtnE= github.com/onsi/gomega v1.18.1 h1:M1GfJqGRrBrrGGsbxzV5dqM2U2ApXefZCQpkukxYRLE= -github.com/philhofer/fwd v1.1.1 h1:GdGcTjf5RNAxwS4QLsiMzJYj5KEvPJD3Abr261yRQXQ= github.com/philhofer/fwd v1.1.1/go.mod h1:gk3iGcWd9+svBvR0sR+KPcfE+RNWozjowpeBVG3ZVNU= -github.com/rivo/uniseg v0.2.0 h1:S1pD9weZBuJdFmowNwbpi7BJ8TNftyUImj/0WQi72jY= +github.com/philhofer/fwd v1.1.2 h1:bnDivRJ1EWPjUIRXV5KfORO897HTbpFAQddBdE8t7Gw= +github.com/philhofer/fwd v1.1.2/go.mod h1:qkPdfjR2SIEbspLqpe1tO4n5yICnr2DY7mqEx2tUTP0= github.com/rivo/uniseg v0.2.0/go.mod h1:J6wj4VEh+S6ZtnVlnTBMWIodfgj8LQOQFoIToxlJtxc= github.com/rivo/uniseg v0.4.4 h1:8TfxU8dW6PdqD27gjM8MVNuicgxIjxpm4K7x4jp8sis= github.com/rivo/uniseg v0.4.4/go.mod h1:FN3SvrM+Zdj16jyLfmOkMNblXMcoc8DfTHruCPUcx88= github.com/savsgio/dictpool v0.0.0-20221023140959-7bf2e61cea94 h1:rmMl4fXJhKMNWl+K+r/fq4FbbKI+Ia2m9hYBLm2h4G4= github.com/savsgio/dictpool v0.0.0-20221023140959-7bf2e61cea94/go.mod h1:90zrgN3D/WJsDd1iXHT96alCoN2KJo6/4x1DZC3wZs8= -github.com/savsgio/gotils v0.0.0-20220530130905-52f3993e8d6d h1:Q+gqLBOPkFGHyCJxXMRqtUgUbTjI8/Ze8vu8GGyNFwo= github.com/savsgio/gotils v0.0.0-20220530130905-52f3993e8d6d/go.mod h1:Gy+0tqhJvgGlqnTF8CVGP0AaGRjwBtXs/a5PA0Y3+A4= -github.com/tinylib/msgp v1.1.6 h1:i+SbKraHhnrf9M5MYmvQhFnbLhAXSDWF8WWsuyRdocw= +github.com/savsgio/gotils v0.0.0-20230208104028-c358bd845dee h1:8Iv5m6xEo1NR1AvpV+7XmhI4r39LGNzwUL4YpMuL5vk= +github.com/savsgio/gotils v0.0.0-20230208104028-c358bd845dee/go.mod h1:qwtSXrKuJh/zsFQ12yEE89xfCrGKK63Rr7ctU/uCo4g= github.com/tinylib/msgp v1.1.6/go.mod h1:75BAfg2hauQhs3qedfdDZmWAPcFMAvJE5b9rGOMufyw= +github.com/tinylib/msgp v1.1.8 h1:FCXC1xanKO4I8plpHGH2P7koL/RzZs12l/+r7vakfm0= +github.com/tinylib/msgp v1.1.8/go.mod h1:qkpG+2ldGg4xRFmx+jfTvZPxfGFhi64BcnL9vkCm/Tw= github.com/valyala/bytebufferpool v1.0.0 h1:GqA5TC/0021Y/b9FG4Oi9Mr3q7XYx6KllzawFIhcdPw= github.com/valyala/bytebufferpool v1.0.0/go.mod h1:6bBcMArwyJ5K/AmCkWv1jt77kVWyCJ6HpOuEn7z0Csc= -github.com/valyala/fasthttp v1.41.0 h1:zeR0Z1my1wDHTRiamBCXVglQdbUwgb9uWG3k1HQz6jY= -github.com/valyala/fasthttp v1.41.0/go.mod h1:f6VbjjoI3z1NDOZOv17o6RvtRSWxC77seBFc2uWtgiY= -github.com/valyala/fasthttp v1.44.0 h1:R+gLUhldIsfg1HokMuQjdQ5bh9nuXHPIfvkYUu9eR5Q= -github.com/valyala/fasthttp v1.44.0/go.mod h1:f6VbjjoI3z1NDOZOv17o6RvtRSWxC77seBFc2uWtgiY= +github.com/valyala/fasthttp v1.45.0 h1:zPkkzpIn8tdHZUrVa6PzYd0i5verqiPSkgTd3bSUcpA= +github.com/valyala/fasthttp v1.45.0/go.mod h1:k2zXd82h/7UZc3VOdJ2WaUqt1uZ/XpXAfE9i+HBC3lA= github.com/valyala/tcplisten v1.0.0 h1:rBHj/Xf+E1tRGZyWIWwJDiRY0zc1Js+CV5DqwacVSA8= github.com/valyala/tcplisten v1.0.0/go.mod h1:T0xQ8SeCZGxckz9qRXTfG43PvQ/mcWh7FwZEA7Ioqkc= github.com/yuin/goldmark v1.2.1/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= +github.com/yuin/goldmark v1.4.13/go.mod h1:6yULJ656Px+3vBD8DxQVa3kxgyrAnzto9xy5taEt/CY= golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= -golang.org/x/crypto v0.0.0-20220214200702-86341886e292/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4= +golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= golang.org/x/mod v0.3.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= +golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4= +golang.org/x/mod v0.7.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs= golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= golang.org/x/net v0.0.0-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= -golang.org/x/net v0.0.0-20211112202133-69e39bad7dc2/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= -golang.org/x/net v0.0.0-20220906165146-f3363e06e74c h1:yKufUcDwucU5urd+50/Opbt4AYpqthk7wHpHok8f1lo= -golang.org/x/net v0.0.0-20220906165146-f3363e06e74c/go.mod h1:YDH+HFinaLZZlnHAfSS6ZXJJ9M9t4Dl22yv3iI2vPwk= +golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= +golang.org/x/net v0.0.0-20220722155237-a158d28d115b/go.mod h1:XRhObCWvk6IyKnWLug+ECip1KBveYUHfp+8e9klMJ9c= +golang.org/x/net v0.3.0/go.mod h1:MBQ8lrhLObU/6UmLb4fmbmk5OcyYmqtbGd/9yIeKjEE= +golang.org/x/net v0.8.0 h1:Zrh2ngAOFYneWTAIAPethzeaQLuHwhuBkuV6ZiRnUaQ= golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20201020160332-67f06af15bc9/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20220722155255-886fb9371eb4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.1.0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200930185726-fdedc70b468f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20210423082822-04245dca01da/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.0.0-20220728004956-3c1f35247d10/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.0.0-20220811171246-fbc7d0a398ab h1:2QkjZIsXupsJbJIdSjjUOgWK3aEtzyuh2mPt3l/CkeU= +golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20220722155257-8c9f86f7a55f/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220811171246-fbc7d0a398ab/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.5.0 h1:MUK/U/4lj1t1oPg0HfuXDN/Z1wv31ZJ/YcPiGccS4DU= -golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.3.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.6.0 h1:MVltZSvRTcU2ljQOhs94SXPftV6DCNnZViHeQps87pQ= +golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= +golang.org/x/term v0.3.0/go.mod h1:q750SLmJuPmVoN1blW3UFBPREJfb1KmY3vwxfr+nFDA= golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= -golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= -golang.org/x/text v0.3.7 h1:olpwvP2KacW1ZWvsR7uQhoyTYvKAupfQrRGBFM352Gk= golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ= +golang.org/x/text v0.5.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8= +golang.org/x/text v0.8.0 h1:57P1ETyNKtuIjB4SRd15iJxuhj8Gc416Y78H3qgMh68= golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= golang.org/x/tools v0.0.0-20201022035929-9cf592e881e9/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= +golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc= +golang.org/x/tools v0.4.0/go.mod h1:UE5sM2OK9E/d67R0ANs2xJizIymRP5gJU295PvKXxjQ= golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= diff --git a/src/config.go b/src/config.go index c2f0483..4d94dbe 100644 --- a/src/config.go +++ b/src/config.go @@ -9,17 +9,35 @@ import ( "gopkg.in/yaml.v3" ) +var ( + // DefaultConfig is the default configuration values used by the application. + DefaultConfig *Config = &Config{ + Environment: "production", + Host: "127.0.0.1", + Port: 3001, + Redis: nil, + Cache: ConfigCache{ + JavaStatusDuration: time.Minute, + BedrockStatusDuration: time.Minute, + IconDuration: time.Minute * 15, + }, + } +) + // Config represents the application configuration. type Config struct { - Environment string `yaml:"environment"` - Host string `yaml:"host"` - Port uint16 `yaml:"port"` - Redis *string `yaml:"redis"` - Cache struct { - JavaStatusDuration time.Duration `yaml:"java_status_duration" json:"java_status_duration"` - BedrockStatusDuration time.Duration `yaml:"bedrock_status_duration" json:"bedrock_status_duration"` - IconDuration time.Duration `yaml:"icon_duration" json:"icon_duration"` - } `yaml:"cache"` + Environment string `yaml:"environment"` + Host string `yaml:"host"` + Port uint16 `yaml:"port"` + Redis *string `yaml:"redis"` + Cache ConfigCache `yaml:"cache"` +} + +// ConfigCache represents the caching durations of various responses. +type ConfigCache struct { + JavaStatusDuration time.Duration `yaml:"java_status_duration" json:"java_status_duration"` + BedrockStatusDuration time.Duration `yaml:"bedrock_status_duration" json:"bedrock_status_duration"` + IconDuration time.Duration `yaml:"icon_duration" json:"icon_duration"` } // ReadFile reads the configuration from the given file and overrides values using environment variables. @@ -37,6 +55,17 @@ func (c *Config) ReadFile(file string) error { return c.overrideWithEnvVars() } +// WriteFile writes the configuration values to a file. +func (c *Config) WriteFile(file string) error { + data, err := yaml.Marshal(c) + + if err != nil { + return err + } + + return os.WriteFile(file, data, 0777) +} + // overrideWithEnvVars overrides configuration values using environment variables. func (c *Config) overrideWithEnvVars() error { if value := os.Getenv("ENVIRONMENT"); value != "" { diff --git a/src/main.go b/src/main.go index 2ce8059..454b4af 100644 --- a/src/main.go +++ b/src/main.go @@ -1,9 +1,11 @@ package main import ( + "errors" "fmt" "log" "net/http" + "os" "github.com/gofiber/fiber/v2" "github.com/gofiber/fiber/v2/middleware/cors" @@ -20,24 +22,32 @@ var ( return ctx.SendStatus(http.StatusInternalServerError) }, }) - r *Redis = &Redis{} - config *Config = &Config{} + r *Redis = &Redis{} + conf *Config = DefaultConfig ) func init() { - if err := config.ReadFile("config.yml"); err != nil { - log.Fatalf("failed to read config file: %v", err) + if err := conf.ReadFile("config.yml"); err != nil { + if errors.Is(err, os.ErrNotExist) { + log.Printf("config.yml does not exist, writing default config\n") + + if err = conf.WriteFile("config.yml"); err != nil { + log.Fatalf("Failed to write config file: %v", err) + } + } else { + log.Printf("Failed to read config file: %v", err) + } } if err := GetBlockedServerList(); err != nil { - log.Fatalf("failed to retrieve EULA blocked servers: %v", err) + log.Fatalf("Failed to retrieve EULA blocked servers: %v", err) } log.Println("Successfully retrieved EULA blocked servers") - if config.Redis != nil { + if conf.Redis != nil { if err := r.Connect(); err != nil { - log.Fatalf("failed to connect to Redis: %v", err) + log.Fatalf("Failed to connect to Redis: %v", err) } log.Println("Successfully connected to Redis") @@ -47,7 +57,7 @@ func init() { EnableStackTrace: true, })) - if config.Environment == "development" { + if conf.Environment == "development" { app.Use(cors.New(cors.Config{ AllowOrigins: "*", AllowMethods: "HEAD,OPTIONS,GET", @@ -64,9 +74,9 @@ func init() { func main() { defer r.Close() - log.Printf("Listening on %s:%d\n", config.Host, config.Port) + log.Printf("Listening on %s:%d\n", conf.Host, conf.Port) - if err := app.Listen(fmt.Sprintf("%s:%d", config.Host, config.Port)); err != nil { + if err := app.Listen(fmt.Sprintf("%s:%d", conf.Host, conf.Port)); err != nil { log.Fatalf("failed to start server: %v", err) } } diff --git a/src/redis.go b/src/redis.go index 1a8bbc0..bae2403 100644 --- a/src/redis.go +++ b/src/redis.go @@ -17,11 +17,12 @@ type Redis struct { // Connect establishes a connection to the Redis server using the configuration. func (r *Redis) Connect() error { - if config.Redis == nil { + if conf.Redis == nil { return errors.New("missing Redis configuration") } - opts, err := redis.ParseURL(*config.Redis) + opts, err := redis.ParseURL(*conf.Redis) + if err != nil { return err } @@ -29,6 +30,7 @@ func (r *Redis) Connect() error { r.Client = redis.NewClient(opts) ctx, cancel := context.WithTimeout(context.Background(), defaultTimeout) + defer cancel() return r.Client.Ping(ctx).Err() @@ -41,6 +43,7 @@ func (r *Redis) Get(key string) ([]byte, time.Duration, error) { } ctx, cancel := context.WithTimeout(context.Background(), defaultTimeout) + defer cancel() p := r.Client.Pipeline() @@ -68,6 +71,7 @@ func (r *Redis) Set(key string, value interface{}, ttl time.Duration) error { } ctx, cancel := context.WithTimeout(context.Background(), defaultTimeout) + defer cancel() return r.Client.Set(ctx, key, value, ttl).Err() @@ -80,6 +84,7 @@ func (r *Redis) Increment(key string) error { } ctx, cancel := context.WithTimeout(context.Background(), defaultTimeout) + defer cancel() return r.Client.Incr(ctx, key).Err() diff --git a/src/status.go b/src/status.go index f1769d9..07da24e 100644 --- a/src/status.go +++ b/src/status.go @@ -10,6 +10,7 @@ import ( "github.com/mcstatus-io/mcutil" ) +// StatusResponse is the root response for returning any status response from the API. type StatusResponse struct { Online bool `json:"online"` Host string `json:"host"` @@ -19,11 +20,13 @@ type StatusResponse struct { ExpiresAt int64 `json:"expires_at"` } +// JavaStatusResponse is the combined response of the root response and the Java Edition status response. type JavaStatusResponse struct { StatusResponse *JavaStatus } +// JavaStatus is the status response properties for Java Edition. type JavaStatus struct { Version *JavaVersion `json:"version"` Players JavaPlayers `json:"players"` @@ -32,11 +35,13 @@ type JavaStatus struct { Mods []Mod `json:"mods"` } +// BedrockStatusResponse is the combined response of the root response and the Bedrock Edition status response. type BedrockStatusResponse struct { StatusResponse *BedrockStatus } +// BedrockStatus is the status response properties for Bedrock Edition. type BedrockStatus struct { Version *BedrockVersion `json:"version"` Players *BedrockPlayers `json:"players"` @@ -46,29 +51,34 @@ type BedrockStatus struct { Edition *string `json:"edition"` } +// JavaVersion holds the properties for the version of Java Edition responses. type JavaVersion struct { NameRaw string `json:"name_raw"` NameClean string `json:"name_clean"` NameHTML string `json:"name_html"` - Protocol int `json:"protocol"` + Protocol int64 `json:"protocol"` } +// BedrockVersion holds the properties for the version of Bedrock Edition responses. type BedrockVersion struct { Name *string `json:"name"` Protocol *int64 `json:"protocol"` } +// JavaPlayers holds the properties for the players of Java Edition responses. type JavaPlayers struct { - Online int `json:"online"` - Max int `json:"max"` + Online *int64 `json:"online"` + Max *int64 `json:"max"` List []Player `json:"list"` } +// BedrockPlayers holds the properties for the players of Bedrock Edition responses. type BedrockPlayers struct { Online *int64 `json:"online"` Max *int64 `json:"max"` } +// Player is a single sample player used in Java Edition status responses. type Player struct { UUID string `json:"uuid"` NameRaw string `json:"name_raw"` @@ -76,17 +86,20 @@ type Player struct { NameHTML string `json:"name_html"` } +// MOTD is a group of formatted and unformatted properties for status responses. type MOTD struct { Raw string `json:"raw"` Clean string `json:"clean"` HTML string `json:"html"` } +// Mod is a single Forge mod installed on any Java Edition status response. type Mod struct { Name string `json:"name"` Version string `json:"version"` } +// GetJavaStatus returns the status response of a Java Edition server, either using cache or fetching a fresh status. func GetJavaStatus(host string, port uint16) (*JavaStatusResponse, time.Duration, error) { cacheKey := fmt.Sprintf("java:%s-%d", host, port) @@ -116,13 +129,14 @@ func GetJavaStatus(host string, port uint16) (*JavaStatusResponse, time.Duration return nil, 0, err } - if err := r.Set(cacheKey, data, config.Cache.JavaStatusDuration); err != nil { + if err := r.Set(cacheKey, data, conf.Cache.JavaStatusDuration); err != nil { return nil, 0, err } return response, 0, nil } +// GetBedrockStatus returns the status response of a Bedrock Edition server, either using cache or fetching a fresh status. func GetBedrockStatus(host string, port uint16) (*BedrockStatusResponse, time.Duration, error) { cacheKey := fmt.Sprintf("bedrock:%s-%d", host, port) @@ -152,13 +166,14 @@ func GetBedrockStatus(host string, port uint16) (*BedrockStatusResponse, time.Du return nil, 0, err } - if err := r.Set(cacheKey, data, config.Cache.BedrockStatusDuration); err != nil { + if err := r.Set(cacheKey, data, conf.Cache.BedrockStatusDuration); err != nil { return nil, 0, err } return response, 0, nil } +// GetServerIcon returns the icon image of a Java Edition server, either using cache or fetching a fresh image. func GetServerIcon(host string, port uint16) ([]byte, time.Duration, error) { cacheKey := fmt.Sprintf("icon:%s-%d", host, port) @@ -186,13 +201,14 @@ func GetServerIcon(host string, port uint16) ([]byte, time.Duration, error) { icon = data } - if err := r.Set(cacheKey, icon, config.Cache.IconDuration); err != nil { + if err := r.Set(cacheKey, icon, conf.Cache.IconDuration); err != nil { return nil, 0, err } return icon, 0, nil } +// FetchJavaStatus fetches a fresh status of a Java Edition server. func FetchJavaStatus(host string, port uint16) (*JavaStatusResponse, error) { status, err := mcutil.Status(host, port) @@ -207,7 +223,7 @@ func FetchJavaStatus(host string, port uint16) (*JavaStatusResponse, error) { Port: port, EULABlocked: IsBlockedAddress(host), RetrievedAt: time.Now().UnixMilli(), - ExpiresAt: time.Now().Add(config.Cache.JavaStatusDuration).UnixMilli(), + ExpiresAt: time.Now().Add(conf.Cache.JavaStatusDuration).UnixMilli(), }, }, nil } @@ -219,13 +235,13 @@ func FetchJavaStatus(host string, port uint16) (*JavaStatusResponse, error) { Port: port, EULABlocked: IsBlockedAddress(host), RetrievedAt: time.Now().UnixMilli(), - ExpiresAt: time.Now().Add(config.Cache.JavaStatusDuration).UnixMilli(), + ExpiresAt: time.Now().Add(conf.Cache.JavaStatusDuration).UnixMilli(), }, JavaStatus: &JavaStatus{ Version: nil, Players: JavaPlayers{ - Online: statusLegacy.Players.Online, - Max: statusLegacy.Players.Max, + Online: &statusLegacy.Players.Online, + Max: &statusLegacy.Players.Max, List: make([]Player, 0), }, MOTD: MOTD{ @@ -281,7 +297,7 @@ func FetchJavaStatus(host string, port uint16) (*JavaStatusResponse, error) { Port: port, EULABlocked: IsBlockedAddress(host), RetrievedAt: time.Now().UnixMilli(), - ExpiresAt: time.Now().Add(config.Cache.JavaStatusDuration).UnixMilli(), + ExpiresAt: time.Now().Add(conf.Cache.JavaStatusDuration).UnixMilli(), }, JavaStatus: &JavaStatus{ Version: &JavaVersion{ @@ -306,6 +322,7 @@ func FetchJavaStatus(host string, port uint16) (*JavaStatusResponse, error) { }, nil } +// FetchBedrockStatus fetches a fresh status of a Bedrock Edition server. func FetchBedrockStatus(host string, port uint16) (*BedrockStatusResponse, error) { status, err := mcutil.StatusBedrock(host, port) @@ -317,7 +334,7 @@ func FetchBedrockStatus(host string, port uint16) (*BedrockStatusResponse, error Port: port, EULABlocked: IsBlockedAddress(host), RetrievedAt: time.Now().UnixMilli(), - ExpiresAt: time.Now().Add(config.Cache.BedrockStatusDuration).UnixMilli(), + ExpiresAt: time.Now().Add(conf.Cache.BedrockStatusDuration).UnixMilli(), }, }, nil } @@ -329,7 +346,7 @@ func FetchBedrockStatus(host string, port uint16) (*BedrockStatusResponse, error Port: port, EULABlocked: IsBlockedAddress(host), RetrievedAt: time.Now().UnixMilli(), - ExpiresAt: time.Now().Add(config.Cache.BedrockStatusDuration).UnixMilli(), + ExpiresAt: time.Now().Add(conf.Cache.BedrockStatusDuration).UnixMilli(), }, BedrockStatus: &BedrockStatus{ Version: nil,