diff --git a/examples/telegram/.gitignore b/examples/telegram/.gitignore new file mode 100644 index 0000000..0cffcb3 --- /dev/null +++ b/examples/telegram/.gitignore @@ -0,0 +1 @@ +config.json \ No newline at end of file diff --git a/examples/telegram/config.go b/examples/telegram/config.go new file mode 100644 index 0000000..ca9ad61 --- /dev/null +++ b/examples/telegram/config.go @@ -0,0 +1,42 @@ +package main + +import ( + "encoding/json" + "github.com/go-playground/validator/v10" + "log" + "os" + "strings" +) + +var AppConfig Config + +type Config struct { + Listen string `json:"listen"` + BaseURL string `json:"baseUrl" validate:"required,url"` + System string `json:"system" validate:"required,url"` + APIKey string `json:"apiKey" validate:"required"` + TGBotToken string `json:"tgBotToken" validate:"required"` +} + +func LoadConfig(src string) { + file, err := os.Open(src) + if err != nil { + panic(err) + } + defer func() { _ = file.Close() }() + if err := json.NewDecoder(file).Decode(&AppConfig); err != nil { + panic(err) + } + validate := validator.New(validator.WithRequiredStructEnabled()) + err = validate.Struct(AppConfig) + if err != nil { + log.Fatalln(err) + } + if strings.HasSuffix(AppConfig.BaseURL, "/") { + AppConfig.BaseURL = AppConfig.BaseURL[:len(AppConfig.BaseURL)-1] + } + if AppConfig.Listen == "" { + AppConfig.Listen = ":8080" + } + log.Println("loaded configuration from", src) +} diff --git a/examples/telegram/config.json.dist b/examples/telegram/config.json.dist new file mode 100644 index 0000000..f93c4a7 --- /dev/null +++ b/examples/telegram/config.json.dist @@ -0,0 +1,7 @@ +{ + "addr": ":8080", + "baseUrl": "https://demo.localhost.run", + "system": "https://test.retailcrm.pro", + "apiKey": "apiKey", + "tgBotToken": "telegram" +} \ No newline at end of file diff --git a/examples/telegram/go.mod b/examples/telegram/go.mod new file mode 100644 index 0000000..f050597 --- /dev/null +++ b/examples/telegram/go.mod @@ -0,0 +1,41 @@ +module github.com/retailcrm/mg-transport-api-client-go/examples/telegram + +go 1.21.5 + +require ( + github.com/gin-gonic/gin v1.9.1 + github.com/go-playground/validator/v10 v10.16.0 + github.com/go-telegram-bot-api/telegram-bot-api/v5 v5.5.1 + github.com/retailcrm/api-client-go/v2 v2.1.12 + github.com/retailcrm/mg-transport-api-client-go v1.3.8 +) + +require ( + github.com/bytedance/sonic v1.9.1 // indirect + github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311 // indirect + github.com/davecgh/go-spew v1.1.1 // indirect + github.com/gabriel-vasile/mimetype v1.4.2 // indirect + github.com/gin-contrib/sse v0.1.0 // indirect + github.com/go-playground/locales v0.14.1 // indirect + github.com/go-playground/universal-translator v0.18.1 // indirect + github.com/goccy/go-json v0.10.2 // indirect + github.com/google/go-querystring v1.1.0 // indirect + github.com/json-iterator/go v1.1.12 // indirect + github.com/klauspost/cpuid/v2 v2.2.4 // indirect + github.com/leodido/go-urn v1.2.4 // indirect + github.com/mattn/go-isatty v0.0.19 // 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.0.8 // indirect + github.com/pmezard/go-difflib v1.0.0 // indirect + github.com/stretchr/testify v1.8.3 // indirect + github.com/twitchyliquid64/golang-asm v0.15.1 // indirect + github.com/ugorji/go/codec v1.2.11 // indirect + golang.org/x/arch v0.3.0 // indirect + golang.org/x/crypto v0.9.0 // indirect + golang.org/x/net v0.10.0 // indirect + golang.org/x/sys v0.8.0 // indirect + golang.org/x/text v0.9.0 // indirect + google.golang.org/protobuf v1.30.0 // indirect + gopkg.in/yaml.v3 v3.0.1 // indirect +) diff --git a/examples/telegram/go.sum b/examples/telegram/go.sum new file mode 100644 index 0000000..1d62e5d --- /dev/null +++ b/examples/telegram/go.sum @@ -0,0 +1,106 @@ +github.com/bytedance/sonic v1.5.0/go.mod h1:ED5hyg4y6t3/9Ku1R6dU/4KyJ48DZ4jPhfY1O2AihPM= +github.com/bytedance/sonic v1.9.1 h1:6iJ6NqdoxCDr6mbY8h18oSO+cShGSMRGCEo7F2h0x8s= +github.com/bytedance/sonic v1.9.1/go.mod h1:i736AoUSYt75HyZLoJW9ERYxcy6eaN6h4BZXU064P/U= +github.com/chenzhuoyu/base64x v0.0.0-20211019084208-fb5309c8db06/go.mod h1:DH46F32mSOjUmXrMHnKwZdA8wcEefY7UVqBKYGjpdQY= +github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311 h1:qSGYFH7+jGhDF8vLC+iwCD4WpbV1EBDSzWkJODFLams= +github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311/go.mod h1:b583jCggY9gE99b6G5LEC39OIiVsWj+R97kbl5odCEk= +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/gabriel-vasile/mimetype v1.4.2 h1:w5qFW6JKBz9Y393Y4q372O9A7cUSequkh1Q7OhCmWKU= +github.com/gabriel-vasile/mimetype v1.4.2/go.mod h1:zApsH/mKG4w07erKIaJPFiX0Tsq9BFQgN3qGY5GnNgA= +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.9.1 h1:4idEAncQnU5cB7BeOkPtxjfCSye0AAm1R0RVIqJ+Jmg= +github.com/gin-gonic/gin v1.9.1/go.mod h1:hPrL7YrpYKXt5YId3A/Tnip5kqbEAP+KLuI3SUcPTeU= +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.16.0 h1:x+plE831WK4vaKHO/jpgUGsvLKIqRRkz6M78GuJAfGE= +github.com/go-playground/validator/v10 v10.16.0/go.mod h1:9iXMNT7sEkjXb0I+enO7QXmzG6QCsPWY4zveKFVRSyU= +github.com/go-telegram-bot-api/telegram-bot-api/v5 v5.5.1 h1:wG8n/XJQ07TmjbITcGiUaOtXxdrINDz1b0J1w0SzqDc= +github.com/go-telegram-bot-api/telegram-bot-api/v5 v5.5.1/go.mod h1:A2S0CWkNylc2phvKXWBBdD3K0iGnDBGbzRpISP2zBl8= +github.com/goccy/go-json v0.10.2 h1:CrxCmQqYDkv1z7lO7Wbh2HN93uovUHgrECaO5ZrCXAU= +github.com/goccy/go-json v0.10.2/go.mod h1:6MelG93GURQebXPDq3khkgXZkazVtN9CRI+MGFi0w8I= +github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk= +github.com/google/go-cmp v0.5.2/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= +github.com/google/go-cmp v0.5.5 h1:Khx7svrCpmxxtHBq5j2mp/xVjsi8hQMfNLvJFAlrGgU= +github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= +github.com/google/go-querystring v1.0.0/go.mod h1:odCYkC5MyYFN7vkCjXpyrEuKhc/BUO6wN/zVPAxq5ck= +github.com/google/go-querystring v1.1.0 h1:AnCroh3fv4ZBgVIf1Iwtovgjaw/GiKJo8M8yD/fhyJ8= +github.com/google/go-querystring v1.1.0/go.mod h1:Kcdr2DB4koayq7X8pmAG4sNG59So17icRSOU623lUBU= +github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg= +github.com/h2non/parth v0.0.0-20190131123155-b4df798d6542 h1:2VTzZjLZBgl62/EtslCrtky5vbi9dd7HrQPQIx6wqiw= +github.com/h2non/parth v0.0.0-20190131123155-b4df798d6542/go.mod h1:Ow0tF8D4Kplbc8s8sSb3V2oUCygFHVp8gC3Dn6U4MNI= +github.com/joho/godotenv v1.3.0 h1:Zjp+RcGpHhGlrMbJzXTrZZPrWj+1vfm90La1wgB6Bhc= +github.com/joho/godotenv v1.3.0/go.mod h1:7hK45KPybAkOC6peb+G5yklZfMxEjkZhHbwpqxOKXbg= +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.4 h1:acbojRNwl3o09bUq+yDCtZFc1aiwaAAxtcn8YkZXnvk= +github.com/klauspost/cpuid/v2 v2.2.4/go.mod h1:RVVoqg1df56z8g3pUjL/3lE5UfnlrJX8tyFgg4nqhuY= +github.com/leodido/go-urn v1.2.4 h1:XlAE/cm/ms7TE/VMVoduSpNBoyc2dOxHs5MZSwAN63Q= +github.com/leodido/go-urn v1.2.4/go.mod h1:7ZrI8mTSeBSHl/UaRyKQW1qZeMgak41ANeCNaVckg+4= +github.com/mattn/go-isatty v0.0.19 h1:JITubQf0MOLdlGRuRq+jtsDlekdYPia9ZFsB8h/APPA= +github.com/mattn/go-isatty v0.0.19/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/nbio/st v0.0.0-20140626010706-e9e8d9816f32/go.mod h1:9wM+0iRr9ahx58uYLpLIr5fm8diHn0JbqRycJi6w0Ms= +github.com/pelletier/go-toml/v2 v2.0.8 h1:0ctb6s9mE31h0/lhu+J6OPmVeDxJn+kYnJc2jZR9tGQ= +github.com/pelletier/go-toml/v2 v2.0.8/go.mod h1:vuYfssBdrU2XDZ9bYydBu6t+6a6PYNcZljzZR9VXg+4= +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/retailcrm/api-client-go/v2 v2.1.12 h1:eiFfSvxgjOyWEYEhg44XoCGf7ud30jVLxGr8t9/3YKc= +github.com/retailcrm/api-client-go/v2 v2.1.12/go.mod h1:1yTZl9+gd3+/k0kAJe7sYvC+mL4fqMwIwtnSgSWZlkQ= +github.com/retailcrm/mg-transport-api-client-go v1.3.8 h1:tHR6ePZONmjYHaEQx28yGEVWQh4jVTa0K/dmq++15YY= +github.com/retailcrm/mg-transport-api-client-go v1.3.8/go.mod h1:gDe/tj7t3Hr/uwIFSBVgGAmP85PoLajVl1A+skBo1Ro= +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/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= +github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4= +github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= +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.2/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4= +github.com/stretchr/testify v1.8.3 h1:RP3t2pwF7cMEbC1dqtB6poj3niw/9gnV4Cjg5oW5gtY= +github.com/stretchr/testify v1.8.3/go.mod h1:sz/lmYIOXD/1dqDmKjjqLyZ2RngseejIcXlSw2iwfAo= +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.11 h1:BMaWp1Bb6fHwEtbplGBGJ498wD+LKlNSl25MjdZY4dU= +github.com/ugorji/go/codec v1.2.11/go.mod h1:UNopzCgEMSXjBc6AOMqYvWC1ktqTAfzJZUZgYf6w6lg= +golang.org/x/arch v0.0.0-20210923205945-b76863e36670/go.mod h1:5om86z9Hs0C8fWVUuoMHwpExlXzs5Tkyp9hOrfG7pp8= +golang.org/x/arch v0.3.0 h1:02VY4/ZcO/gBOH6PUaoiptASxtXU10jazRCP865E97k= +golang.org/x/arch v0.3.0/go.mod h1:5om86z9Hs0C8fWVUuoMHwpExlXzs5Tkyp9hOrfG7pp8= +golang.org/x/crypto v0.9.0 h1:LF6fAI+IutBocDJ2OT0Q1g8plpYljMZ4+lty+dsqw3g= +golang.org/x/crypto v0.9.0/go.mod h1:yrmDGqONDYtNj3tH8X9dzUun2m2lzPa9ngI6/RUPGR0= +golang.org/x/net v0.10.0 h1:X2//UzNDwYmtCLn7To6G58Wr6f5ahEAQgKNzv9Y951M= +golang.org/x/net v0.10.0/go.mod h1:0qNGK6F8kojg2nk9dLZ2mShWaEBan6FAoqfSigmmuDg= +golang.org/x/sys v0.0.0-20220704084225-05e143d24a9e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.8.0 h1:EBmGv8NaZBZTWvrbjNoL6HVt+IVy3QDQpJs7VRIw3tU= +golang.org/x/sys v0.8.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/text v0.9.0 h1:2sjJmO8cDvYveuX97RDLsxlyUxLl+GHoLxBiRdHllBE= +golang.org/x/text v0.9.0/go.mod h1:e1OnstbJyHTd6l/uOt8jFFHp6TRDWZR/bV3emEE/zU8= +golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543 h1:E7g+9GITq07hpfrRu66IVDexMakfv52eLZ2CXBWiKr4= +golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= +google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw= +google.golang.org/protobuf v1.30.0 h1:kPPoIgf3TsEvrm0PFe15JQ+570QVxYzEvvHqChK+cng= +google.golang.org/protobuf v1.30.0/go.mod h1:HV8QOd/L58Z+nl8r43ehVNZIU/HEI6OcFqwMG9pJV4I= +gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405 h1:yhCVgyC4o1eVCa2tZl7eS0r+SDo693bJlVdllGtEeKM= +gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= +gopkg.in/h2non/gock.v1 v1.1.2 h1:jBbHXgGBK/AoPVfJh5x4r/WxIrElvbLel8TCZkkZJoY= +gopkg.in/h2non/gock.v1 v1.1.2/go.mod h1:n7UGz/ckNChHiK05rDoiC4MYSunEC/lyaUm2WWaDva0= +gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= +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= +rsc.io/pdf v0.1.1/go.mod h1:n8OzWcQ6Sp37PL01nO98y4iUCRdTGarVfzxY20ICaU4= diff --git a/examples/telegram/main.go b/examples/telegram/main.go new file mode 100644 index 0000000..336b681 --- /dev/null +++ b/examples/telegram/main.go @@ -0,0 +1,10 @@ +package main + +func main() { + LoadConfig("config.json") + RegisterSystem() + InitTGBotAPI() + SetTGWebhook() + RegisterChannel() + Listen() +} diff --git a/examples/telegram/register.go b/examples/telegram/register.go new file mode 100644 index 0000000..2b27ecc --- /dev/null +++ b/examples/telegram/register.go @@ -0,0 +1,104 @@ +package main + +import ( + retailcrm "github.com/retailcrm/api-client-go/v2" + v1 "github.com/retailcrm/mg-transport-api-client-go/v1" + "log" +) + +func RegisterSystem() { + client := retailcrm.New(AppConfig.System, AppConfig.APIKey) + resp, _, err := client.IntegrationModuleEdit(retailcrm.IntegrationModule{ + Code: "telegram-bot-integration-example", + IntegrationCode: "telegram-bot-integration-example", + Active: v1.BoolPtr(true), + Name: "Telegram Bot Integration Example", + ClientID: "telegram-bot-integration-example", + BaseURL: AppConfig.BaseURL, + AccountURL: AppConfig.BaseURL, + Integrations: &retailcrm.Integrations{ + MgTransport: &retailcrm.MgTransport{ + WebhookURL: AppConfig.BaseURL + "/api/v1/webhook", + }, + }, + }) + if err != nil { + log.Fatalln("cannot edit integration module:", err) + } + MG = v1.New(resp.Info.MgTransportInfo.EndpointURL, resp.Info.MgTransportInfo.Token) + log.Println("updated integration module") +} + +func RegisterChannel() { + channels, _, err := MG.TransportChannels(v1.Channels{}) + if err != nil { + log.Fatalln("cannot get channels:", err) + } + channel := v1.Channel{ + Type: "telegram", + Name: "@" + TG.Self.UserName, + Settings: getChannelSettings(), + } + for _, ch := range channels { + if ch.Name != nil && *ch.Name == "@"+TG.Self.UserName { + channel.ID = ch.ID + _, _, err := MG.UpdateTransportChannel(channel) + if err != nil { + log.Fatalln("cannot update channel:", err) + } + Channel = &channel + log.Println("updated MG channel") + return + } + } + resp, _, err := MG.ActivateTransportChannel(channel) + if err != nil { + log.Fatalln("cannot activate channel:", err) + } + log.Println("activated MG channel with id:", resp.ChannelID) + channel.ID = resp.ChannelID + Channel = &channel +} + +func getChannelSettings() v1.ChannelSettings { + return v1.ChannelSettings{ + Status: v1.Status{ + Delivered: v1.ChannelFeatureNone, + Read: v1.ChannelFeatureNone, + }, + Text: v1.ChannelSettingsText{ + Creating: v1.ChannelFeatureBoth, + Editing: v1.ChannelFeatureNone, + Quoting: v1.ChannelFeatureNone, + Deleting: v1.ChannelFeatureNone, + MaxCharsCount: 2000, + }, + Product: v1.Product{ + Creating: v1.ChannelFeatureNone, + Editing: v1.ChannelFeatureNone, + Deleting: v1.ChannelFeatureNone, + }, + Order: v1.Order{ + Creating: v1.ChannelFeatureNone, + Editing: v1.ChannelFeatureNone, + Deleting: v1.ChannelFeatureNone, + }, + File: v1.ChannelSettingsFilesBase{ + Creating: v1.ChannelFeatureNone, + Editing: v1.ChannelFeatureNone, + Quoting: v1.ChannelFeatureNone, + Deleting: v1.ChannelFeatureNone, + }, + Image: v1.ChannelSettingsFilesBase{ + Creating: v1.ChannelFeatureNone, + Editing: v1.ChannelFeatureNone, + Quoting: v1.ChannelFeatureNone, + Deleting: v1.ChannelFeatureNone, + }, + Audio: v1.ChannelSettingsAudio{ + Creating: v1.ChannelFeatureNone, + Quoting: v1.ChannelFeatureNone, + Deleting: v1.ChannelFeatureNone, + }, + } +} diff --git a/examples/telegram/server.go b/examples/telegram/server.go new file mode 100644 index 0000000..4017fda --- /dev/null +++ b/examples/telegram/server.go @@ -0,0 +1,130 @@ +package main + +import ( + "context" + "errors" + "fmt" + "github.com/gin-gonic/gin" + tgbotapi "github.com/go-telegram-bot-api/telegram-bot-api/v5" + v1 "github.com/retailcrm/mg-transport-api-client-go/v1" + "log" + "net/http" + "os" + "os/signal" + "strconv" + "strings" + "syscall" + "time" +) + +const ChatIDPrefix = "tg_" + +func Listen() { + router := gin.Default() + router.POST("/api/v1/mg", MGWebhookHandler) + router.POST("/api/v1/tg", TGWebhookHandler) + + srv := &http.Server{ + Addr: AppConfig.Listen, + Handler: router, + } + go func() { + log.Printf("listening on %s", AppConfig.Listen) + if err := srv.ListenAndServe(); err != nil && !errors.Is(err, http.ErrServerClosed) { + log.Fatalf("listen: %s\n", err) + } + }() + + quit := make(chan os.Signal) + signal.Notify(quit, syscall.SIGINT, syscall.SIGTERM) + <-quit + + ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second) + defer cancel() + if err := srv.Shutdown(ctx); err != nil { + log.Fatal("shutting down:", err) + } + select { + case <-ctx.Done(): + log.Println("quitting...") + } +} + +func MGWebhookHandler(c *gin.Context) { + var wh v1.WebhookRequest + if err := c.ShouldBindJSON(&wh); err != nil { + c.AbortWithStatusJSON(http.StatusBadRequest, + v1.NewTransportErrorResponse(v1.MessageErrorGeneral, "invalid webhook data")) + return + } + + if wh.Type != v1.MessageSendWebhookType { + c.AbortWithStatusJSON(http.StatusUnprocessableEntity, + v1.NewTransportErrorResponse(v1.MessageErrorGeneral, "unsupported webhook type")) + return + } + + whMsg := wh.MessageWebhookData() + if strings.HasPrefix(whMsg.ExternalChatID, ChatIDPrefix) { + c.AbortWithStatusJSON(http.StatusUnprocessableEntity, + v1.NewTransportErrorResponse(v1.MessageErrorGeneral, "unexpected chat ID")) + return + } + + chatID, err := strconv.ParseInt(whMsg.ExternalChatID[len(ChatIDPrefix):], 10, 64) + if err != nil { + c.AbortWithStatusJSON(http.StatusUnprocessableEntity, + v1.NewTransportErrorResponse(v1.MessageErrorGeneral, "unparsable chat ID")) + return + } + + resp, err := TG.Send(tgbotapi.NewMessage(chatID, whMsg.Content)) + if err != nil { + c.AbortWithStatusJSON(http.StatusUnprocessableEntity, + v1.NewTransportErrorResponse(v1.MessageErrorGeneral, err.Error())) + return + } + c.JSON(http.StatusOK, v1.NewSentMessageResponse(strconv.Itoa(resp.MessageID))) +} + +func TGWebhookHandler(c *gin.Context) { + var update tgbotapi.Update + if err := c.ShouldBindJSON(&update); err != nil { + c.AbortWithStatusJSON(http.StatusBadRequest, gin.H{"error": "invalid request"}) + return + } + + if update.Message == nil { + c.AbortWithStatus(http.StatusOK) + return + } + + _, _, err := MG.Messages(v1.SendData{ + Message: v1.Message{ + ExternalID: newExternalMessageID(update.Message.MessageID), + Type: v1.MsgTypeText, + Text: update.Message.Text, + }, + Originator: v1.OriginatorCustomer, + Customer: v1.Customer{ + ExternalID: strconv.FormatInt(update.Message.From.ID, 10), + Nickname: update.Message.From.UserName, + Firstname: update.Message.From.FirstName, + Lastname: update.Message.From.LastName, + ProfileURL: fmt.Sprintf("https://t.me/%s", update.Message.From.UserName), + Language: update.Message.From.LanguageCode, + }, + Channel: Channel.ID, + ExternalChatID: ChatIDPrefix + strconv.FormatInt(update.Message.Chat.ID, 10), + }) + if err != nil { + log.Printf("error: cannot send message: %s", err) + c.AbortWithStatusJSON(http.StatusInternalServerError, gin.H{"error": "cannot send message"}) + return + } + c.JSON(http.StatusOK, gin.H{}) +} + +func newExternalMessageID(messageID int) string { + return fmt.Sprintf("%d-%d", TG.Self.ID, messageID) +} diff --git a/examples/telegram/services.go b/examples/telegram/services.go new file mode 100644 index 0000000..795d4cf --- /dev/null +++ b/examples/telegram/services.go @@ -0,0 +1,12 @@ +package main + +import ( + tgbotapi "github.com/go-telegram-bot-api/telegram-bot-api/v5" + v1 "github.com/retailcrm/mg-transport-api-client-go/v1" +) + +var ( + Channel *v1.Channel + MG *v1.MgClient + TG *tgbotapi.BotAPI +) diff --git a/examples/telegram/tg.go b/examples/telegram/tg.go new file mode 100644 index 0000000..66f7eaf --- /dev/null +++ b/examples/telegram/tg.go @@ -0,0 +1,27 @@ +package main + +import ( + "fmt" + tgbotapi "github.com/go-telegram-bot-api/telegram-bot-api/v5" + "log" +) + +func InitTGBotAPI() { + botAPI, err := tgbotapi.NewBotAPI(AppConfig.TGBotToken) + if err != nil { + log.Fatalln("cannot initialize TG Bot API:", err) + } + TG = botAPI + log.Printf("initialized Telegram Bot API for bot @%s", botAPI.Self.UserName) +} + +func SetTGWebhook() { + wh, err := tgbotapi.NewWebhook(fmt.Sprintf("%s/api/v1/tg", AppConfig.BaseURL)) + if err != nil { + log.Fatalln("cannot initialize webhook data:", err) + } + _, err = TG.Request(wh) + if err != nil { + log.Fatalln("cannot register TG webhook:", err) + } +}