aboutsummaryrefslogtreecommitdiff
path: root/cmd/patreon-saasproxy/main.go
blob: 363fa80c37e8c3c5218702814113285a9429cbc0 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
package main

import (
	"context"
	"flag"
	"fmt"
	"log"
	"log/slog"
	"net"
	"net/http"
	"os"

	"github.com/facebookgo/flagenv"
	_ "github.com/joho/godotenv/autoload"
	"github.com/twitchtv/twirp"
	"golang.org/x/oauth2"
	"google.golang.org/protobuf/types/known/emptypb"
	"google.golang.org/protobuf/types/known/timestamppb"
	"gopkg.in/mxpv/patreon-go.v1"
	"xeiaso.net/v4/internal"
	"xeiaso.net/v4/internal/adminpb"
	"xeiaso.net/v4/internal/k8s"
)

var (
	bind          = flag.String("bind", ":80", "HTTP bind addr")
	clientID      = flag.String("client-id", "", "Patreon client ID")
	clientSecret  = flag.String("client-secret", "", "Patreon client secret")
	dataDir       = flag.String("data-dir", "./var", "Directory to store data in")
	k8sNamespace  = flag.String("kubernetes-namespace", "default", "Kubernetes namespace this app is running in")
	k8sSecretName = flag.String("kubernetes-secret-name", "xesite-patreon-saasproxy-state", "Kubernetes secret to store state data in")
)

func main() {
	flagenv.Parse()
	flag.Parse()
	internal.Slog()

	os.MkdirAll(*dataDir, 0700)

	config := oauth2.Config{
		ClientID:     *clientID,
		ClientSecret: *clientSecret,
		Endpoint: oauth2.Endpoint{
			AuthURL:  patreon.AuthorizationURL,
			TokenURL: patreon.AccessTokenURL,
		},
		Scopes: []string{"users", "pledges-to-me", "my-campaign"},
	}

	cts, err := k8s.TokenSource(*k8sNamespace, *k8sSecretName, &config)
	if err != nil {
		log.Fatalf("error making token source: %v", err)
	}

	s := &Server{
		cts: cts,
	}

	ph := adminpb.NewPatreonServer(s)
	http.Handle(adminpb.PatreonPathPrefix, ph)

	http.HandleFunc("/healthz", func(w http.ResponseWriter, r *http.Request) {
		fmt.Fprintln(w, "OK")
	})

	ln, err := net.Listen("tcp", *bind)
	if err != nil {
		log.Fatalf("can't listen over TCP: %v", err)
	}
	defer ln.Close()

	slog.Info("listening", "bind", *bind)

	log.Fatal(http.Serve(ln, nil))
}

type Server struct {
	cts oauth2.TokenSource
}

func (s *Server) GetToken(ctx context.Context, _ *emptypb.Empty) (*adminpb.PatreonToken, error) {
	token, err := s.cts.Token()
	if err != nil {
		slog.Error("token fetch failed", "err", err)
		return nil, twirp.InternalErrorWith(err)
	}

	return &adminpb.PatreonToken{
		AccessToken:  token.AccessToken,
		TokenType:    token.TokenType,
		RefreshToken: token.RefreshToken,
		Expiry:       timestamppb.New(token.Expiry),
	}, nil
}