diff --git a/bin/server b/bin/server index dfcd1bf..e72ed25 100755 Binary files a/bin/server and b/bin/server differ diff --git a/go.mod b/go.mod index 47d3957..60c2492 100644 --- a/go.mod +++ b/go.mod @@ -3,6 +3,8 @@ module madsky.ru/go-finance go 1.24.0 require ( + github.com/doganarif/govisual v0.1.8 + github.com/gin-gonic/gin v1.10.0 github.com/goccy/go-json v0.10.5 github.com/ilyakaznacheev/cleanenv v1.5.0 github.com/jackc/pgx/v5 v5.7.2 @@ -12,17 +14,52 @@ require ( require ( github.com/BurntSushi/toml v1.2.1 // indirect + github.com/bytedance/sonic v1.11.6 // indirect + github.com/bytedance/sonic/loader v0.1.1 // indirect + github.com/cenkalti/backoff/v4 v4.2.1 // indirect + github.com/cespare/xxhash/v2 v2.3.0 // indirect + github.com/cloudwego/base64x v0.1.4 // indirect + github.com/cloudwego/iasm v0.2.0 // indirect + github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f // indirect + github.com/gabriel-vasile/mimetype v1.4.3 // indirect + github.com/gin-contrib/sse v0.1.0 // indirect + github.com/go-logr/logr v1.4.2 // indirect + github.com/go-logr/stdr v1.2.2 // indirect + github.com/go-playground/locales v0.14.1 // indirect + github.com/go-playground/universal-translator v0.18.1 // indirect + github.com/go-playground/validator/v10 v10.20.0 // indirect + github.com/go-redis/redis/v8 v8.11.5 // indirect + github.com/google/uuid v1.6.0 // indirect + github.com/grpc-ecosystem/grpc-gateway/v2 v2.19.0 // indirect github.com/jackc/pgpassfile v1.0.0 // indirect github.com/jackc/pgservicefile v0.0.0-20240606120523-5a60cdf6a761 // indirect github.com/jackc/puddle/v2 v2.2.2 // indirect github.com/joho/godotenv v1.5.1 // indirect - github.com/kr/text v0.2.0 // indirect + github.com/json-iterator/go v1.1.12 // indirect + github.com/klauspost/cpuid/v2 v2.2.7 // indirect + github.com/leodido/go-urn v1.4.0 // indirect + github.com/lib/pq v1.10.9 // indirect + github.com/mattn/go-isatty v0.0.20 // indirect + github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd // indirect + github.com/modern-go/reflect2 v1.0.2 // indirect + github.com/pelletier/go-toml/v2 v2.2.2 // indirect github.com/rogpeppe/go-internal v1.14.1 // indirect + github.com/twitchyliquid64/golang-asm v0.15.1 // indirect + github.com/ugorji/go/codec v1.2.12 // indirect + go.opentelemetry.io/otel v1.32.0 // indirect + go.opentelemetry.io/otel/exporters/otlp/otlptrace v1.24.0 // indirect + go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracegrpc v1.24.0 // indirect + go.opentelemetry.io/otel/metric v1.32.0 // indirect + go.opentelemetry.io/otel/sdk v1.32.0 // indirect + go.opentelemetry.io/otel/trace v1.32.0 // indirect + go.opentelemetry.io/proto/otlp v1.1.0 // indirect + golang.org/x/arch v0.8.0 // indirect golang.org/x/crypto v0.31.0 // indirect golang.org/x/net v0.32.0 // indirect - golang.org/x/sync v0.10.0 // indirect - golang.org/x/sys v0.28.0 // indirect - golang.org/x/text v0.21.0 // indirect + golang.org/x/sync v0.13.0 // indirect + golang.org/x/sys v0.32.0 // indirect + golang.org/x/text v0.24.0 // indirect + google.golang.org/genproto/googleapis/api v0.0.0-20241202173237-19429a94021a // indirect google.golang.org/genproto/googleapis/rpc v0.0.0-20241202173237-19429a94021a // indirect gopkg.in/yaml.v3 v3.0.1 // indirect olympos.io/encoding/edn v0.0.0-20201019073823-d3554ca0b0a3 // indirect diff --git a/go.sum b/go.sum index ceba882..c71bedc 100644 --- a/go.sum +++ b/go.sum @@ -1,21 +1,58 @@ github.com/BurntSushi/toml v1.2.1 h1:9F2/+DoOYIOksmaJFPw1tGFy1eDnIJXg+UHjuD8lTak= github.com/BurntSushi/toml v1.2.1/go.mod h1:CxXYINrC8qIiEnFrOxCa7Jy5BFHlXnUU2pbicEuybxQ= -github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E= +github.com/bytedance/sonic v1.11.6 h1:oUp34TzMlL+OY1OUWxHqsdkgC/Zfc85zGqw9siXjrc0= +github.com/bytedance/sonic v1.11.6/go.mod h1:LysEHSvpvDySVdC2f87zGWf6CIKJcAvqab1ZaiQtds4= +github.com/bytedance/sonic/loader v0.1.1 h1:c+e5Pt1k/cy5wMveRDyk2X4B9hF4g7an8N3zCYjJFNM= +github.com/bytedance/sonic/loader v0.1.1/go.mod h1:ncP89zfokxS5LZrJxl5z0UJcsk4M4yY2JpfqGeCtNLU= +github.com/cenkalti/backoff/v4 v4.2.1 h1:y4OZtCnogmCPw98Zjyt5a6+QwPLGkiQsYW5oUqylYbM= +github.com/cenkalti/backoff/v4 v4.2.1/go.mod h1:Y3VNntkOUPxTVeUxJ/G5vcM//AlwfmyYozVcomhLiZE= +github.com/cespare/xxhash/v2 v2.3.0 h1:UL815xU9SqsFlibzuggzjXhog7bL6oX9BbNZnL2UFvs= +github.com/cespare/xxhash/v2 v2.3.0/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= +github.com/cloudwego/base64x v0.1.4 h1:jwCgWpFanWmN8xoIUHa2rtzmkd5J2plF/dnLS6Xd/0Y= +github.com/cloudwego/base64x v0.1.4/go.mod h1:0zlkT4Wn5C6NdauXdJRhSKRlJvmclQ1hhJgA0rcu/8w= +github.com/cloudwego/iasm v0.2.0 h1:1KNIy1I1H9hNNFEEH3DVnI4UujN+1zjpuk6gwHLTssg= +github.com/cloudwego/iasm v0.2.0/go.mod h1:8rXZaNYT2n95jn+zTI1sDr+IgcD2GVs0nlbbQPiEFhY= github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= +github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f h1:lO4WD4F/rVNCu3HqELle0jiPLLBs70cWOduZpkS1E78= +github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f/go.mod h1:cuUVRXasLTGF7a8hSLbxyZXjz+1KgoB3wDUb6vlszIc= +github.com/doganarif/govisual v0.1.8 h1:qAimY3yJNPl84U1ZNEzua4EP5+DxBnLQpdtZmsmd8ig= +github.com/doganarif/govisual v0.1.8/go.mod h1:UC4PGlP6cZRjoTlIUFPOOKSEwbLgPF9kDIoIO1Y4LJs= +github.com/fsnotify/fsnotify v1.4.9 h1:hsms1Qyu0jgnwNXIxa+/V/PDsU6CfLf6CNO8H7IWoS4= +github.com/fsnotify/fsnotify v1.4.9/go.mod h1:znqG4EE+3YCdAaPaxE2ZRY/06pZUdp0tY4IgpuI1SZQ= +github.com/gabriel-vasile/mimetype v1.4.3 h1:in2uUcidCuFcDKtdcBxlR0rJ1+fsokWf+uqxgUFjbI0= +github.com/gabriel-vasile/mimetype v1.4.3/go.mod h1:d8uq/6HKRL6CGdk+aubisF/M5GcPfT7nKyLpA0lbSSk= +github.com/gin-contrib/sse v0.1.0 h1:Y/yl/+YNO8GZSjAhjMsSuLt29uWRFHdHYUb5lYOV9qE= +github.com/gin-contrib/sse v0.1.0/go.mod h1:RHrZQHXnP2xjPF+u1gW/2HnVO7nvIa9PG3Gm+fLHvGI= +github.com/gin-gonic/gin v1.10.0 h1:nTuyha1TYqgedzytsKYqna+DfLos46nTv2ygFy86HFU= +github.com/gin-gonic/gin v1.10.0/go.mod h1:4PMNQiOhvDRa013RKVbsiNwoyezlm2rm0uX/T7kzp5Y= +github.com/go-logr/logr v1.2.2/go.mod h1:jdQByPbusPIv2/zmleS9BjJVeZ6kBagPoEUsqbVz/1A= github.com/go-logr/logr v1.4.2 h1:6pFjapn8bFcIbiKo3XT4j/BhANplGihG6tvd+8rYgrY= github.com/go-logr/logr v1.4.2/go.mod h1:9T104GzyrTigFIr8wt5mBrctHMim0Nb2HLGrmQ40KvY= github.com/go-logr/stdr v1.2.2 h1:hSWxHoqTgW2S2qGc0LTAI563KZ5YKYRhT3MFKZMbjag= github.com/go-logr/stdr v1.2.2/go.mod h1:mMo/vtBO5dYbehREoey6XUKy/eSumjCCveDpRre4VKE= +github.com/go-playground/assert/v2 v2.2.0 h1:JvknZsQTYeFEAhQwI4qEt9cyV5ONwRHC+lYKSsYSR8s= +github.com/go-playground/assert/v2 v2.2.0/go.mod h1:VDjEfimB/XKnb+ZQfWdccd7VUvScMdVu0Titje2rxJ4= +github.com/go-playground/locales v0.14.1 h1:EWaQ/wswjilfKLTECiXz7Rh+3BjFhfDFKv/oXslEjJA= +github.com/go-playground/locales v0.14.1/go.mod h1:hxrqLVvrK65+Rwrd5Fc6F2O76J/NuW9t0sjnWqG1slY= +github.com/go-playground/universal-translator v0.18.1 h1:Bcnm0ZwsGyWbCzImXv+pAJnYK9S473LQFuzCbDbfSFY= +github.com/go-playground/universal-translator v0.18.1/go.mod h1:xekY+UJKNuX9WP91TpwSH2VMlDf28Uj24BCp08ZFTUY= +github.com/go-playground/validator/v10 v10.20.0 h1:K9ISHbSaI0lyB2eWMPJo+kOS/FBExVwjEviJTixqxL8= +github.com/go-playground/validator/v10 v10.20.0/go.mod h1:dbuPbCMFw/DrkbEynArYaCwl3amGuJotoKCe95atGMM= +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/goccy/go-json v0.10.5 h1:Fq85nIqj+gXn/S5ahsiTlK3TmC85qgirsdTP/+DeaC4= github.com/goccy/go-json v0.10.5/go.mod h1:oq7eo15ShAhp70Anwd5lgX2pLfOS3QCiwU/PULtXL6M= github.com/golang/protobuf v1.5.4 h1:i7eJL8qZTpSEXOPTxNKhASYpMn+8e5Q6AdndVa1dWek= github.com/golang/protobuf v1.5.4/go.mod h1:lnTiLA8Wa4RWRcIUkrtSVa5nRhsEGBg48fD6rSs7xps= github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI= github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= +github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg= github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0= github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= +github.com/grpc-ecosystem/grpc-gateway/v2 v2.19.0 h1:Wqo399gCIufwto+VfwCSvsnfGpF/w5E9CNxSwbpD6No= +github.com/grpc-ecosystem/grpc-gateway/v2 v2.19.0/go.mod h1:qmOFXW2epJhM0qSnUUYpldc7gVz2KMQwJ/QYCDIa7XU= github.com/ilyakaznacheev/cleanenv v1.5.0 h1:0VNZXggJE2OYdXE87bfSSwGxeiGt9moSR2lOrsHHvr4= github.com/ilyakaznacheev/cleanenv v1.5.0/go.mod h1:a5aDzaJrLCQZsazHol1w8InnDcOX0OColm64SlIi6gk= github.com/jackc/pgpassfile v1.0.0 h1:/6Hmqy13Ss2zCq62VdNG8tM1wchn8zjSGOBJ6icpsIM= @@ -28,21 +65,61 @@ github.com/jackc/puddle/v2 v2.2.2 h1:PR8nw+E/1w0GLuRFSmiioY6UooMp6KJv0/61nB7icHo github.com/jackc/puddle/v2 v2.2.2/go.mod h1:vriiEXHvEE654aYKXXjOvZM39qJ0q+azkZFrfEOc3H4= github.com/joho/godotenv v1.5.1 h1:7eLL/+HRGLY0ldzfGMeQkb7vMd0as4CfYvUVzLqw0N0= github.com/joho/godotenv v1.5.1/go.mod h1:f4LDr5Voq0i2e/R5DDNOoa2zzDfwtkZa6DnEwAbqwq4= -github.com/kr/pretty v0.3.0 h1:WgNl7dwNpEZ6jJ9k1snq4pZsg7DOEN8hP9Xw0Tsjwk0= -github.com/kr/pretty v0.3.0/go.mod h1:640gp4NfQd8pI5XOwp5fnNeVWj67G7CFk/SaSQn7NBk= +github.com/json-iterator/go v1.1.12 h1:PV8peI4a0ysnczrg+LtxykD8LfKY9ML6u2jnxaEnrnM= +github.com/json-iterator/go v1.1.12/go.mod h1:e30LSqwooZae/UwlEbR2852Gd8hjQvJoHmT4TnhNGBo= +github.com/klauspost/cpuid/v2 v2.0.9/go.mod h1:FInQzS24/EEf25PyTYn52gqo7WaD8xa0213Md/qVLRg= +github.com/klauspost/cpuid/v2 v2.2.7 h1:ZWSB3igEs+d0qvnxR/ZBzXVmxkgt8DdzP6m9pfuVLDM= +github.com/klauspost/cpuid/v2 v2.2.7/go.mod h1:Lcz8mBdAVJIBVzewtcLocK12l3Y+JytZYpaMropDUws= +github.com/knz/go-libedit v1.10.1/go.mod h1:MZTVkCWyz0oBc7JOWP3wNAzd002ZbM/5hgShxwh4x8M= +github.com/kr/pretty v0.3.1 h1:flRD4NNwYAUpkphVc1HcthR4KEIFJ65n8Mw5qdRn3LE= +github.com/kr/pretty v0.3.1/go.mod h1:hoEshYVHaxMs3cyo3Yncou5ZscifuDolrwPKZanG3xk= github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= +github.com/leodido/go-urn v1.4.0 h1:WT9HwE9SGECu3lg4d/dIA+jxlljEa1/ffXKmRjqdmIQ= +github.com/leodido/go-urn v1.4.0/go.mod h1:bvxc+MVxLKB4z00jd1z+Dvzr47oO32F/QSNjSBOlFxI= +github.com/lib/pq v1.10.9 h1:YXG7RB+JIjhP29X+OtkiDnYaXQwpS4JEWq7dtCCRUEw= +github.com/lib/pq v1.10.9/go.mod h1:AlVN5x4E4T544tWzH6hKfbfQvm3HdbOxrmggDNAPY9o= +github.com/mattn/go-isatty v0.0.20 h1:xfD0iDuEKnDkl03q4limB+vH+GxLEtL/jb4xVJSWWEY= +github.com/mattn/go-isatty v0.0.20/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y= +github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= +github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd h1:TRLaZ9cD/w8PVh93nsPXa1VrQ6jlwL5oN8l14QlcNfg= +github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= +github.com/modern-go/reflect2 v1.0.2 h1:xBagoLtFs94CBntxluKeaWgTMpvLxC4ur3nMaC9Gz0M= +github.com/modern-go/reflect2 v1.0.2/go.mod h1:yWuevngMOJpCy52FWWMvUC8ws7m/LJsjYzDa0/r8luk= +github.com/nxadm/tail v1.4.8 h1:nPr65rt6Y5JFSKQO7qToXr7pePgD6Gwiw05lkbyAQTE= +github.com/nxadm/tail v1.4.8/go.mod h1:+ncqLTQzXmGhMZNUePPaPqPvBxHAIsmXswZKocGu+AU= +github.com/onsi/ginkgo v1.16.5 h1:8xi0RTUf59SOSfEtZMvwTvXYMzG4gV23XVHOZiXNtnE= +github.com/onsi/ginkgo v1.16.5/go.mod h1:+E8gABHa3K6zRBolWtd+ROzc/U5bkGt0FwiG042wbpU= +github.com/onsi/gomega v1.18.1 h1:M1GfJqGRrBrrGGsbxzV5dqM2U2ApXefZCQpkukxYRLE= +github.com/onsi/gomega v1.18.1/go.mod h1:0q+aL8jAiMXy9hbwj2mr5GziHiwhAIQpFmmtT5hitRs= +github.com/pelletier/go-toml/v2 v2.2.2 h1:aYUidT7k73Pcl9nb2gScu7NSrKCSHIDE89b3+6Wq+LM= +github.com/pelletier/go-toml/v2 v2.2.2/go.mod h1:1t835xjRzz80PqgE6HHgN2JOsmgYu/h4qDAS4n929Rs= github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= github.com/rogpeppe/go-internal v1.14.1 h1:UQB4HGPB6osV0SQTLymcB4TgvyWu6ZyliaW0tI/otEQ= github.com/rogpeppe/go-internal v1.14.1/go.mod h1:MaRKkUm5W0goXpeCfT7UZI6fk/L7L7so1lCWt35ZSgc= github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= +github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw= +github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo= +github.com/stretchr/objx v0.5.2/go.mod h1:FRsXN1f5AsAjCGJKqEizvkpNtU+EGNCLh3NxZ/8L+MA= github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= -github.com/stretchr/testify v1.8.1 h1:w7B6lhMri9wdJUVmEZPGGhZzrYTPvgJArz7wNPgYKsk= +github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= +github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU= github.com/stretchr/testify v1.8.1/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4= +github.com/stretchr/testify v1.8.4/go.mod h1:sz/lmYIOXD/1dqDmKjjqLyZ2RngseejIcXlSw2iwfAo= +github.com/stretchr/testify v1.9.0 h1:HtqpIVDClZ4nwg75+f6Lvsy/wHu+3BoSGCbBAcpTsTg= +github.com/stretchr/testify v1.9.0/go.mod h1:r2ic/lqez/lEtzL7wO/rwa5dbSLXVDPFyf8C91i36aY= +github.com/twitchyliquid64/golang-asm v0.15.1 h1:SU5vSMR7hnwNxj24w34ZyCi/FmDZTkS4MhqMhdFk5YI= +github.com/twitchyliquid64/golang-asm v0.15.1/go.mod h1:a1lVb/DtPvCB8fslRZhAngC2+aY1QWCk3Cedj/Gdt08= +github.com/ugorji/go/codec v1.2.12 h1:9LC83zGrHhuUA9l16C9AHXAqEV/2wBQ4nkvumAE65EE= +github.com/ugorji/go/codec v1.2.12/go.mod h1:UNopzCgEMSXjBc6AOMqYvWC1ktqTAfzJZUZgYf6w6lg= go.opentelemetry.io/otel v1.32.0 h1:WnBN+Xjcteh0zdk01SVqV55d/m62NJLJdIyb4y/WO5U= go.opentelemetry.io/otel v1.32.0/go.mod h1:00DCVSB0RQcnzlwyTfqtxSm+DRr9hpYrHjNGiBHVQIg= +go.opentelemetry.io/otel/exporters/otlp/otlptrace v1.24.0 h1:t6wl9SPayj+c7lEIFgm4ooDBZVb01IhLB4InpomhRw8= +go.opentelemetry.io/otel/exporters/otlp/otlptrace v1.24.0/go.mod h1:iSDOcsnSA5INXzZtwaBPrKp/lWu/V14Dd+llD0oI2EA= +go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracegrpc v1.24.0 h1:Mw5xcxMwlqoJd97vwPxA8isEaIoxsta9/Q51+TTJLGE= +go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracegrpc v1.24.0/go.mod h1:CQNu9bj7o7mC6U7+CA/schKEYakYXWr79ucDHTMGhCM= go.opentelemetry.io/otel/metric v1.32.0 h1:xV2umtmNcThh2/a/aCP+h64Xx5wsj8qqnkYZktzNa0M= go.opentelemetry.io/otel/metric v1.32.0/go.mod h1:jH7CIbbK6SH2V2wE16W05BHCtIDzauciCRLoc/SyMv8= go.opentelemetry.io/otel/sdk v1.32.0 h1:RNxepc9vK59A8XsgZQouW8ue8Gkb4jpWtJm9ge5lEG4= @@ -51,16 +128,27 @@ go.opentelemetry.io/otel/sdk/metric v1.32.0 h1:rZvFnvmvawYb0alrYkjraqJq0Z4ZUJAiy go.opentelemetry.io/otel/sdk/metric v1.32.0/go.mod h1:PWeZlq0zt9YkYAp3gjKZ0eicRYvOh1Gd+X99x6GHpCQ= go.opentelemetry.io/otel/trace v1.32.0 h1:WIC9mYrXf8TmY/EXuULKc8hR17vE+Hjv2cssQDe03fM= go.opentelemetry.io/otel/trace v1.32.0/go.mod h1:+i4rkvCraA+tG6AzwloGaCtkx53Fa+L+V8e9a7YvhT8= +go.opentelemetry.io/proto/otlp v1.1.0 h1:2Di21piLrCqJ3U3eXGCTPHE9R8Nh+0uglSnOyxikMeI= +go.opentelemetry.io/proto/otlp v1.1.0/go.mod h1:GpBHCBWiqvVLDqmHZsoMM3C5ySeKTC7ej/RNTae6MdY= +go.uber.org/goleak v1.3.0 h1:2K3zAYmnTNqV73imy9J1T3WC+gmCePx2hEGkimedGto= +go.uber.org/goleak v1.3.0/go.mod h1:CoHD4mav9JJNrW/WLlf7HGZPjdw8EucARQHekz1X6bE= +golang.org/x/arch v0.0.0-20210923205945-b76863e36670/go.mod h1:5om86z9Hs0C8fWVUuoMHwpExlXzs5Tkyp9hOrfG7pp8= +golang.org/x/arch v0.8.0 h1:3wRIsP3pM4yUptoR96otTUOXI367OS0+c9eeRi9doIc= +golang.org/x/arch v0.8.0/go.mod h1:FEVrYAQjsQXMVJ1nsMoVVXPZg6p2JE2mx8psSWTDQys= golang.org/x/crypto v0.31.0 h1:ihbySMvVjLAeSH1IbfcRTkD/iNscyz8rGzjF/E5hV6U= golang.org/x/crypto v0.31.0/go.mod h1:kDsLvtWBEx7MV9tJOj9bnXsPbxwJQ6csT/x4KIN4Ssk= golang.org/x/net v0.32.0 h1:ZqPmj8Kzc+Y6e0+skZsuACbx+wzMgo5MQsJh9Qd6aYI= golang.org/x/net v0.32.0/go.mod h1:CwU0IoeOlnQQWJ6ioyFrfRuomB8GKF6KbYXZVyeXNfs= -golang.org/x/sync v0.10.0 h1:3NQrjDixjgGwUOCaF8w2+VYHv0Ve/vGYSbdkTa98gmQ= -golang.org/x/sync v0.10.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk= -golang.org/x/sys v0.28.0 h1:Fksou7UEQUWlKvIdsqzJmUmCX3cZuD2+P3XyyzwMhlA= -golang.org/x/sys v0.28.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= -golang.org/x/text v0.21.0 h1:zyQAAkrwaneQ066sspRyJaG9VNi/YJ1NfzcGB3hZ/qo= -golang.org/x/text v0.21.0/go.mod h1:4IBbMaMmOPCJ8SecivzSH54+73PCFmPWxNTLm+vZkEQ= +golang.org/x/sync v0.13.0 h1:AauUjRAJ9OSnvULf/ARrrVywoJDy0YS2AwQ98I37610= +golang.org/x/sync v0.13.0/go.mod h1:1dzgHSNfp02xaA81J2MS99Qcpr2w7fw1gpm99rleRqA= +golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.32.0 h1:s77OFDvIQeibCmezSnk/q6iAfkdiQaJi4VzroCFrN20= +golang.org/x/sys v0.32.0/go.mod h1:BJP2sWEmIv4KK5OTEluFJCKSidICx8ciO85XgH3Ak8k= +golang.org/x/text v0.24.0 h1:dd5Bzh4yt5KYA8f9CJHCP4FB4D51c2c6JvN37xJJkJ0= +golang.org/x/text v0.24.0/go.mod h1:L8rBsPeo2pSS+xqN0d5u2ikmjtmoJbDBT1b7nHvFCdU= +google.golang.org/genproto/googleapis/api v0.0.0-20241202173237-19429a94021a h1:OAiGFfOiA0v9MRYsSidp3ubZaBnteRUyn3xB2ZQ5G/E= +google.golang.org/genproto/googleapis/api v0.0.0-20241202173237-19429a94021a/go.mod h1:jehYqy3+AhJU9ve55aNOaSml7wUXjF9x6z2LcCfpAhY= google.golang.org/genproto/googleapis/rpc v0.0.0-20241202173237-19429a94021a h1:hgh8P4EuoxpsuKMXX/To36nOFD7vixReXgn8lPGnt+o= google.golang.org/genproto/googleapis/rpc v0.0.0-20241202173237-19429a94021a/go.mod h1:5uTbfoYQed2U9p3KIj2/Zzm02PYhndfdmML0qC3q3FU= google.golang.org/grpc v1.70.0 h1:pWFv03aZoHzlRKHWicjsZytKAiYCtNS0dHbXnIdq7jQ= @@ -70,8 +158,14 @@ google.golang.org/protobuf v1.35.2/go.mod h1:9fA7Ob0pmnwhb644+1+CVWFRbNajQ6iRojt gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk= gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c/go.mod h1:JHkPIbrfpd72SG/EVd6muEfDQjcINNoR0C8j2r3qZ4Q= +gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7 h1:uRGJdciOHaEIrze2W8Q3AKkepLTh2hOroT7a+7czfdQ= +gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7/go.mod h1:dt/ZhP58zS4L8KSrWDmTeBkI65Dw0HsyUHuEVlX15mw= +gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY= +gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ= gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= +nullprogram.com/x/optparse v1.0.0/go.mod h1:KdyPE+Igbe0jQUrVfMqDMeJQIJZEuyV7pjYmp6pbG50= olympos.io/encoding/edn v0.0.0-20201019073823-d3554ca0b0a3 h1:slmdOY3vp8a7KQbHkL+FLbvbkgMqmXojpFUO/jENuqQ= olympos.io/encoding/edn v0.0.0-20201019073823-d3554ca0b0a3/go.mod h1:oVgVk4OWVDi43qWBEyGhXgYxt7+ED4iYNpTngSLX2Iw= +rsc.io/pdf v0.1.1/go.mod h1:n8OzWcQ6Sp37PL01nO98y4iUCRdTGarVfzxY20ICaU4= diff --git a/internal/auth/dto.go b/internal/auth/dto.go new file mode 100644 index 0000000..877bd2d --- /dev/null +++ b/internal/auth/dto.go @@ -0,0 +1,6 @@ +package auth + +type LoginDto struct { + Login string `json:"login"` + Password string `json:"password"` +} diff --git a/internal/repository/user/user.go b/internal/repository/user/user.go new file mode 100644 index 0000000..a17a7f0 --- /dev/null +++ b/internal/repository/user/user.go @@ -0,0 +1,50 @@ +package user + +import ( + "context" + "madsky.ru/go-finance/internal/database" + "madsky.ru/go-finance/internal/user" +) + +type Repository interface { + Find(ctx context.Context) ([]*user.User, error) + FindOne(ctx context.Context, id uint64) (*user.User, error) + Create(ctx context.Context, dto *user.CreateUserDTO) (*user.User, error) + Update(ctx context.Context, id uint64, issue *user.CreateUserDTO) (*user.User, error) + Remove(ctx context.Context, id uint64) (uint64, error) +} + +type repository struct { + client database.Client +} + +func NewRepository(client database.Client) Repository { + return &repository{ + client: client, + } +} + +func (r *repository) Find(ctx context.Context) ([]*user.User, error) { + //TODO implement me + panic("implement me") +} + +func (r *repository) FindOne(ctx context.Context, id uint64) (*user.User, error) { + //TODO implement me + panic("implement me") +} + +func (r *repository) Create(ctx context.Context, dto *user.CreateUserDTO) (*user.User, error) { + //TODO implement me + panic("implement me") +} + +func (r *repository) Update(ctx context.Context, id uint64, issue *user.CreateUserDTO) (*user.User, error) { + //TODO implement me + panic("implement me") +} + +func (r repository) Remove(ctx context.Context, id uint64) (uint64, error) { + //TODO implement me + panic("implement me") +} diff --git a/internal/server/handlers/auth.go b/internal/server/handlers/auth.go new file mode 100644 index 0000000..b87ac6b --- /dev/null +++ b/internal/server/handlers/auth.go @@ -0,0 +1,35 @@ +package handlers + +import ( + "context" + "fmt" + "github.com/goccy/go-json" + "madsky.ru/go-finance/internal/auth" + repository "madsky.ru/go-finance/internal/repository/user" + "madsky.ru/go-finance/internal/server/response" + "net/http" +) + +func RegisterAuthRoutes(mux *http.ServeMux, ctx context.Context, repository repository.Repository) { + mux.HandleFunc("POST /api/login", Login(ctx, repository)) +} + +func Login(ctx context.Context, repository repository.Repository) http.HandlerFunc { + return func(w http.ResponseWriter, r *http.Request) { + p := 0 + + dec := json.NewDecoder(r.Body) + dec.DisallowUnknownFields() + + var loginDto auth.LoginDto + + if err := dec.Decode(&loginDto); err != nil { + response.Error(w, err, http.StatusBadRequest) + return + } + + fmt.Println(loginDto) + + response.WriteJSON(w, nil, http.StatusCreated, &p) + } +} diff --git a/internal/server/handlers/issue.go b/internal/server/handlers/issue.go index e8fbd0a..4c53570 100644 --- a/internal/server/handlers/issue.go +++ b/internal/server/handlers/issue.go @@ -10,6 +10,14 @@ import ( "net/http" ) +func RegisterIssueRoutes(mux *http.ServeMux, ctx context.Context, repository issue.Repository) { + mux.HandleFunc("GET /api/issues/", FindIssues(ctx, repository)) + mux.HandleFunc("GET /api/issues/{id}", FindIssuesByID(ctx, repository)) + mux.HandleFunc("POST /api/issues", CreateIssues(ctx, repository)) + mux.HandleFunc("POST /api/issues/positions", UpdatePositions(ctx, repository)) + mux.HandleFunc("DELETE /api/issues/{id}", DeleteIssues(ctx, repository)) +} + func FindIssues(ctx context.Context, repository issue.Repository) http.HandlerFunc { return func(w http.ResponseWriter, r *http.Request) { i, err := repository.Find(ctx) diff --git a/internal/server/handlers/project.go b/internal/server/handlers/project.go index 813d7be..f479025 100644 --- a/internal/server/handlers/project.go +++ b/internal/server/handlers/project.go @@ -10,6 +10,14 @@ import ( "net/http" ) +func RegisterProjectRoutes(mux *http.ServeMux, ctx context.Context, repository project.Repository) { + mux.HandleFunc("GET /api/projects", FindProjects(ctx, repository)) + mux.HandleFunc("GET /api/projects/{id}", FindProjectByID(ctx, repository)) + mux.HandleFunc("POST /api/projects", CreateProject(ctx, repository)) + mux.HandleFunc("PUT /api/projects/{id}", UpdateProject(ctx, repository)) + mux.HandleFunc("DELETE /api/projects/{id}", DeleteProject(ctx, repository)) +} + func FindProjects(ctx context.Context, repository project.Repository) http.HandlerFunc { return func(w http.ResponseWriter, r *http.Request) { p, err := repository.Find(ctx) diff --git a/internal/server/handlers/status.go b/internal/server/handlers/status.go index 2d32d1b..baa5832 100644 --- a/internal/server/handlers/status.go +++ b/internal/server/handlers/status.go @@ -10,6 +10,13 @@ import ( "net/http" ) +func RegisterStatusRoutes(mux *http.ServeMux, ctx context.Context, repository status.Repository) { + mux.HandleFunc("GET /api/statuses/", FindStatuses(ctx, repository)) + mux.HandleFunc("GET /api/statuses/{id}", FindStatusById(ctx, repository)) + mux.HandleFunc("POST /api/statuses", CreateStatus(ctx, repository)) + mux.HandleFunc("DELETE /api/statuses/{id}", DeleteStatus(ctx, repository)) +} + func FindStatuses(ctx context.Context, repository status.Repository) http.HandlerFunc { return func(w http.ResponseWriter, r *http.Request) { s, err := repository.Find(ctx) diff --git a/internal/server/server.go b/internal/server/server.go index 7e8713b..e353cc7 100644 --- a/internal/server/server.go +++ b/internal/server/server.go @@ -2,14 +2,20 @@ package server import ( "context" + "io/fs" + "madsky.ru/go-finance/web" + "os" + "strings" + + //"github.com/doganarif/govisual" + "github.com/gin-gonic/gin" "github.com/jackc/pgx/v5/pgxpool" "log/slog" "madsky.ru/go-finance/internal/repository/issue" "madsky.ru/go-finance/internal/repository/project" "madsky.ru/go-finance/internal/repository/status" + "madsky.ru/go-finance/internal/repository/user" "madsky.ru/go-finance/internal/server/handlers" - "madsky.ru/go-finance/internal/server/middleware" - "madsky.ru/go-finance/web" "net/http" "time" ) @@ -17,54 +23,74 @@ import ( type Server struct { http *http.Server logger *slog.Logger -} - -func RegisterProjectRoutes(mux *http.ServeMux, ctx context.Context, repository project.Repository) { - mux.HandleFunc("GET /api/projects", handlers.FindProjects(ctx, repository)) - mux.HandleFunc("GET /api/projects/{id}", handlers.FindProjectByID(ctx, repository)) - mux.HandleFunc("POST /api/projects", handlers.CreateProject(ctx, repository)) - mux.HandleFunc("PUT /api/projects/{id}", handlers.UpdateProject(ctx, repository)) - mux.HandleFunc("DELETE /api/projects/{id}", handlers.DeleteProject(ctx, repository)) -} - -func RegisterStatusRoutes(mux *http.ServeMux, ctx context.Context, repository status.Repository) { - mux.HandleFunc("GET /api/statuses/", handlers.FindStatuses(ctx, repository)) - mux.HandleFunc("GET /api/statuses/{id}", handlers.FindStatusById(ctx, repository)) - mux.HandleFunc("POST /api/statuses", handlers.CreateStatus(ctx, repository)) - mux.HandleFunc("DELETE /api/statuses/{id}", handlers.DeleteStatus(ctx, repository)) -} - -func RegisterIssueRoutes(mux *http.ServeMux, ctx context.Context, repository issue.Repository) { - mux.HandleFunc("GET /api/issues/", handlers.FindIssues(ctx, repository)) - mux.HandleFunc("GET /api/issues/{id}", handlers.FindIssuesByID(ctx, repository)) - mux.HandleFunc("POST /api/issues", handlers.CreateIssues(ctx, repository)) - mux.HandleFunc("POST /api/issues/positions", handlers.UpdatePositions(ctx, repository)) - mux.HandleFunc("DELETE /api/issues/{id}", handlers.DeleteIssues(ctx, repository)) + gin *gin.Engine } func NewServer(ctx context.Context, client *pgxpool.Pool, logger *slog.Logger) *Server { - const addr = "localhost:3000" - mux := http.NewServeMux() - mux.Handle("/", http.FileServer(http.FS(web.Dist))) + const addr = "0.0.0.0:3000" - handler := middleware.LoggingMiddleware(mux, logger) + //r := gin.Default() + // + //r.GET("/ping", func(c *gin.Context) { + // c.JSON(http.StatusOK, gin.H{ + // "message": "pong", + // }) + //}) + + mux := http.NewServeMux() + + mux.Handle("/", http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + dist, err := fs.Sub(web.DistDir, "dist") + if err != nil { + http.Error(w, err.Error(), http.StatusInternalServerError) + } + + f, err := dist.Open(strings.TrimPrefix(r.URL.Path, "/")) + if err == nil { + defer func(f fs.File) { + err := f.Close() + if err != nil { + logger.Error("error close file", err) + } + }(f) + } + + if os.IsNotExist(err) { + r.URL.Path = "/" + } + + http.FileServer(http.FS(dist)).ServeHTTP(w, r) + })) + + //http://localhost:8080/__viz + //handler := govisual.Wrap( + // mux, + // govisual.WithRequestBodyLogging(true), + // govisual.WithResponseBodyLogging(true), + //) + //handler := middleware.LoggingMiddleware(mux, logger) projectsRepository := project.NewRepository(client) - RegisterProjectRoutes(mux, ctx, projectsRepository) + handlers.RegisterProjectRoutes(mux, ctx, projectsRepository) statusRepository := status.NewRepository(client) - RegisterStatusRoutes(mux, ctx, statusRepository) + handlers.RegisterStatusRoutes(mux, ctx, statusRepository) issueRepository := issue.NewRepository(client) - RegisterIssueRoutes(mux, ctx, issueRepository) + handlers.RegisterIssueRoutes(mux, ctx, issueRepository) - logger.Info("start server", slog.String("addr", addr)) + userRepository := user.NewRepository(client) + handlers.RegisterAuthRoutes(mux, ctx, userRepository) + + //mux.Handle("/api/", http.StripPrefix("/api", apiHandler)) + //logger.Info("start server", slog.String("addr", addr)) return &Server{ logger: logger, + //gin: r, http: &http.Server{ Addr: addr, - Handler: handler, + Handler: mux, ReadTimeout: 10 * time.Second, WriteTimeout: 10 * time.Second, MaxHeaderBytes: 1 << 20, @@ -73,6 +99,7 @@ func NewServer(ctx context.Context, client *pgxpool.Pool, logger *slog.Logger) * } func (s *Server) Start() error { + //s.gin.Run('0.0.0.0:3000') return s.http.ListenAndServe() } diff --git a/internal/user/dto.go b/internal/user/dto.go new file mode 100644 index 0000000..a0df78d --- /dev/null +++ b/internal/user/dto.go @@ -0,0 +1,3 @@ +package user + +type CreateUserDTO struct{} diff --git a/internal/user/model.go b/internal/user/model.go new file mode 100644 index 0000000..2b93dd9 --- /dev/null +++ b/internal/user/model.go @@ -0,0 +1,6 @@ +package user + +type User struct { + ID uint32 `json:"id"` + Login string `json:"login"` +} diff --git a/web/embed.go b/web/embed.go index e7d28fe..5eacfd9 100644 --- a/web/embed.go +++ b/web/embed.go @@ -2,10 +2,7 @@ package web import ( "embed" - "io/fs" ) //go:embed all:dist -var distDir embed.FS - -var Dist, _ = fs.Sub(distDir, "dist") +var DistDir embed.FS diff --git a/web/src/components/EditableText.vue b/web/src/components/EditableText.vue new file mode 100644 index 0000000..b6f76b3 --- /dev/null +++ b/web/src/components/EditableText.vue @@ -0,0 +1,12 @@ + + + + + diff --git a/web/src/components/IssueCreateForm.vue b/web/src/components/IssueCreateForm.vue index 76ef00c..a955232 100644 --- a/web/src/components/IssueCreateForm.vue +++ b/web/src/components/IssueCreateForm.vue @@ -46,20 +46,19 @@ function onClickOutside() { hide-details placeholder="title" variant="plain" - clearable @keydown.enter="addIssue" v-model:model-value="issue.name" - > + /> - + - + - + diff --git a/web/src/components/IssueItemAddComment.vue b/web/src/components/IssueItemAddComment.vue new file mode 100644 index 0000000..0438ca5 --- /dev/null +++ b/web/src/components/IssueItemAddComment.vue @@ -0,0 +1,13 @@ + + + + + diff --git a/web/src/components/IssueItemAddons.vue b/web/src/components/IssueItemAddons.vue new file mode 100644 index 0000000..4fe904b --- /dev/null +++ b/web/src/components/IssueItemAddons.vue @@ -0,0 +1,41 @@ + + + + + diff --git a/web/src/components/IssueItemCommentsList.vue b/web/src/components/IssueItemCommentsList.vue new file mode 100644 index 0000000..bb9fecb --- /dev/null +++ b/web/src/components/IssueItemCommentsList.vue @@ -0,0 +1,9 @@ + + + + + diff --git a/web/src/components/IssueItemDetails.vue b/web/src/components/IssueItemDetails.vue index c2e6fc4..ee37777 100644 --- a/web/src/components/IssueItemDetails.vue +++ b/web/src/components/IssueItemDetails.vue @@ -3,6 +3,9 @@ import type { Issue } from '@/stores/issues.ts' import { computed } from 'vue' import StatusMenu from '@/components/StatusMenu.vue' +import EditableText from '@/components/EditableText.vue' +import IssueItemAddComment from '@/components/IssueItemAddComment.vue' +import IssueItemAddons from '@/components/IssueItemAddons.vue' const props = defineProps<{ selectedIssue: Issue }>() @@ -20,11 +23,11 @@ const issueId = computed(() => {