1
0
mirror of https://github.com/go-micro/go-micro.git synced 2026-04-30 19:15:24 +02:00
Files
2026-02-11 11:25:36 +00:00

145 lines
3.9 KiB
Go

package auth
import (
"context"
"go-micro.dev/v5/auth"
"go-micro.dev/v5/client"
"go-micro.dev/v5/metadata"
)
// ClientOptions for configuring the auth client wrapper
type ClientOptions struct {
// Auth provider for token generation
Auth auth.Auth
// Token to use (optional - if not provided, will be extracted from context)
Token string
}
// AuthClient returns a client Wrapper that adds authentication tokens to outgoing requests.
//
// For each outgoing request:
// 1. Extracts or uses provided token
// 2. Adds Bearer token to request metadata
// 3. Makes the RPC call
//
// Example usage:
//
// client := client.NewClient(
// client.Wrap(auth.AuthClient(auth.ClientOptions{
// Auth: myAuthProvider,
// Token: myToken,
// })),
// )
func AuthClient(opts ClientOptions) client.Wrapper {
return func(c client.Client) client.Client {
return &authClient{
Client: c,
opts: opts,
}
}
}
// authClient wraps a client to add authentication
type authClient struct {
client.Client
opts ClientOptions
}
// Call adds authentication token to the request
func (a *authClient) Call(ctx context.Context, req client.Request, rsp interface{}, opts ...client.CallOption) error {
// Get token from options or context
token := a.opts.Token
if token == "" && a.opts.Auth != nil {
// Try to get token from context account
if acc, ok := auth.AccountFromContext(ctx); ok {
// Generate token for this account
if t, err := a.opts.Auth.Token(auth.WithCredentials(acc.ID, acc.Secret)); err == nil {
token = t.AccessToken
}
}
}
// Add token to metadata if available
if token != "" {
md, ok := metadata.FromContext(ctx)
if !ok {
md = metadata.Metadata{}
}
md = TokenToMetadata(md, token)
ctx = metadata.NewContext(ctx, md)
}
return a.Client.Call(ctx, req, rsp, opts...)
}
// Stream adds authentication token to the stream request
func (a *authClient) Stream(ctx context.Context, req client.Request, opts ...client.CallOption) (client.Stream, error) {
// Get token from options or context
token := a.opts.Token
if token == "" && a.opts.Auth != nil {
// Try to get token from context account
if acc, ok := auth.AccountFromContext(ctx); ok {
// Generate token for this account
if t, err := a.opts.Auth.Token(auth.WithCredentials(acc.ID, acc.Secret)); err == nil {
token = t.AccessToken
}
}
}
// Add token to metadata if available
if token != "" {
md, ok := metadata.FromContext(ctx)
if !ok {
md = metadata.Metadata{}
}
md = TokenToMetadata(md, token)
ctx = metadata.NewContext(ctx, md)
}
return a.Client.Stream(ctx, req, opts...)
}
// Publish adds authentication token to the publish request
func (a *authClient) Publish(ctx context.Context, msg client.Message, opts ...client.PublishOption) error {
// Get token from options or context
token := a.opts.Token
if token == "" && a.opts.Auth != nil {
// Try to get token from context account
if acc, ok := auth.AccountFromContext(ctx); ok {
// Generate token for this account
if t, err := a.opts.Auth.Token(auth.WithCredentials(acc.ID, acc.Secret)); err == nil {
token = t.AccessToken
}
}
}
// Add token to metadata if available
if token != "" {
md, ok := metadata.FromContext(ctx)
if !ok {
md = metadata.Metadata{}
}
md = TokenToMetadata(md, token)
ctx = metadata.NewContext(ctx, md)
}
return a.Client.Publish(ctx, msg, opts...)
}
// FromToken creates a client wrapper with a static token.
// This is useful when you have a pre-generated token and don't need the auth provider.
func FromToken(token string) client.Wrapper {
return AuthClient(ClientOptions{
Token: token,
})
}
// FromContext creates a client wrapper that extracts the account from context
// and generates a token for each request. Useful for service-to-service auth.
func FromContext(authProvider auth.Auth) client.Wrapper {
return AuthClient(ClientOptions{
Auth: authProvider,
})
}