| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138 |
- package cli
-
- import (
- "encoding/json"
- "fmt"
- "net"
- "net/url"
- "os"
- "strconv"
- "sync"
-
- "github.com/9seconds/mtg/v2/internal/config"
- "github.com/9seconds/mtg/v2/internal/utils"
- )
-
- type accessResponse struct {
- IPv4 *accessResponseURLs `json:"ipv4,omitempty"`
- IPv6 *accessResponseURLs `json:"ipv6,omitempty"`
- Secret struct {
- Hex string `json:"hex"`
- Base64 string `json:"base64"`
- } `json:"secret"`
- }
-
- type accessResponseURLs struct {
- IP net.IP `json:"ip"`
- Port uint `json:"port"`
- TgURL string `json:"tg_url"` //nolint: tagliatelle
- TgQrCode string `json:"tg_qrcode"` //nolint: tagliatelle
- TmeURL string `json:"tme_url"` //nolint: tagliatelle
- TmeQrCode string `json:"tme_qrcode"` //nolint: tagliatelle
- }
-
- type Access struct {
- ConfigPath string `kong:"arg,required,type='existingfile',help='Path to the configuration file.',name='config-path'"` //nolint: lll
- PublicIPv4 net.IP `kong:"help='Public IPv4 address for proxy. By default it is resolved via remote website',name='ipv4',short='i'"` //nolint: lll
- PublicIPv6 net.IP `kong:"help='Public IPv6 address for proxy. By default it is resolved via remote website',name='ipv6',short='I'"` //nolint: lll
- Port uint `kong:"help='Port number. Default port is taken from configuration file, bind-to parameter',type:'uint',short='p'"` //nolint: lll
- Hex bool `kong:"help='Print secret in hex encoding.',short='x'"`
- }
-
- func (a *Access) Run(cli *CLI, version string) error {
- conf, err := utils.ReadConfig(a.ConfigPath)
- if err != nil {
- return fmt.Errorf("cannot init config: %w", err)
- }
-
- resp := &accessResponse{}
- resp.Secret.Base64 = conf.Secret.Base64()
- resp.Secret.Hex = conf.Secret.Hex()
-
- ntw, err := makeNetwork(conf, version)
- if err != nil {
- return fmt.Errorf("cannot init network: %w", err)
- }
-
- wg := &sync.WaitGroup{}
-
- wg.Go(func() {
- ip := a.PublicIPv4
- if ip == nil {
- ip = getIP(ntw, "tcp4")
- }
-
- if ip != nil {
- ip = ip.To4()
- }
-
- resp.IPv4 = a.makeURLs(conf, ip)
- })
- wg.Go(func() {
- ip := a.PublicIPv6
- if ip == nil {
- ip = getIP(ntw, "tcp6")
- }
-
- if ip != nil {
- ip = ip.To16()
- }
-
- resp.IPv6 = a.makeURLs(conf, ip)
- })
-
- wg.Wait()
-
- encoder := json.NewEncoder(os.Stdout)
- encoder.SetEscapeHTML(false)
- encoder.SetIndent("", " ")
-
- if err := encoder.Encode(resp); err != nil {
- return fmt.Errorf("cannot dump access json: %w", err)
- }
-
- return nil
- }
-
- func (a *Access) makeURLs(conf *config.Config, ip net.IP) *accessResponseURLs {
- if ip == nil {
- return nil
- }
-
- portNo := a.Port
- if portNo == 0 {
- portNo = conf.BindTo.Port
- }
-
- values := url.Values{}
- values.Set("server", ip.String())
- values.Set("port", strconv.Itoa(int(portNo)))
-
- if a.Hex {
- values.Set("secret", conf.Secret.Hex())
- } else {
- values.Set("secret", conf.Secret.Base64())
- }
-
- urlQuery := values.Encode()
-
- rv := &accessResponseURLs{
- IP: ip,
- Port: portNo,
- TgURL: (&url.URL{
- Scheme: "tg",
- Host: "proxy",
- RawQuery: urlQuery,
- }).String(),
- TmeURL: (&url.URL{
- Scheme: "https",
- Host: "t.me",
- Path: "proxy",
- RawQuery: urlQuery,
- }).String(),
- }
- rv.TgQrCode = utils.MakeQRCodeURL(rv.TgURL)
- rv.TmeQrCode = utils.MakeQRCodeURL(rv.TmeURL)
-
- return rv
- }
|