package main import ( "context" "encoding/json" "fmt" "net/http" "os" plugin "github.com/hashicorp/go-plugin" "github.com/synctv-org/synctv/internal/provider" "github.com/synctv-org/synctv/internal/provider/plugins" "golang.org/x/oauth2" ) // Linux/Mac/Windows: // CGO_ENABLED=0 GOOS=linux GOARCH=amd64 go build ./internal/provider/plugins/example/example_authing/example_authing.go // CGO_ENABLED=0 GOOS=dawin GOARCH=amd64 go build ./internal/provider/plugins/example/example_authing/example_authing.go // CGO_ENABLED=0 GOOS=windows GOARCH=amd64 go build ./internal/provider/plugins/example/example_authing/example_authing.go // // mv gitee {data-dir}/plugins/oauth2/authing // // Authing:https://console.authing.cn/ // // config.yaml: // // oauth2_plugins: // - plugin_file: plugins/oauth2/authing // args: ["认证配置-认证地址(只需要你自定义的那个部分)"] type AuthingProvider struct { config oauth2.Config } func newAuthingProvider(AuthUrl string) provider.ProviderInterface { return &AuthingProvider{ config: oauth2.Config{ Scopes: []string{"profile"}, Endpoint: oauth2.Endpoint{ AuthURL: fmt.Sprintf("https://%s.authing.cn/oauth/auth", AuthUrl), // 授权码(authorization_code)获取接口 TokenURL: fmt.Sprintf("https://%s.authing.cn/oauth/token", AuthUrl), // Token端点 }, }, } } func (p *AuthingProvider) Init(c provider.Oauth2Option) { p.config.ClientID = c.ClientID p.config.ClientSecret = c.ClientSecret p.config.RedirectURL = c.RedirectURL } func (p *AuthingProvider) Provider() provider.OAuth2Provider { return "authing" //插件名 } func (p *AuthingProvider) NewAuthURL(ctx context.Context, state string) (string, error) { return p.config.AuthCodeURL(state, oauth2.AccessTypeOnline), nil } func (p *AuthingProvider) GetUserInfo(ctx context.Context, code string) (*provider.UserInfo, error) { tk, err := p.config.Exchange(ctx, code) if err != nil { return nil, err } client := p.config.Client(ctx, tk) req, err := http.NewRequestWithContext(ctx, http.MethodGet, "https://core.authing.cn/oauth/me", nil) // 身份端点 if err != nil { return nil, err } resp, err := client.Do(req) if err != nil { return nil, err } defer resp.Body.Close() ui := AuthingUserInfo{} err = json.NewDecoder(resp.Body).Decode(&ui) if err != nil { return nil, err } return &provider.UserInfo{ Username: ui.Name, ProviderUserID: ui.UnionId, }, nil } type AuthingUserInfo struct { UnionId string `json:"sub"` // Authing用户ID Name string `json:"name"` // Authing用户名 } func main() { args := os.Args var pluginMap = map[string]plugin.Plugin{ "Provider": &plugins.ProviderPlugin{Impl: newAuthingProvider(args[1])}, } plugin.Serve(&plugin.ServeConfig{ HandshakeConfig: plugins.HandshakeConfig, Plugins: pluginMap, GRPCServer: plugin.DefaultGRPCServer, }) }