mirror of
https://github.com/siderolabs/talos.git
synced 2025-08-21 14:41:12 +02:00
This PR brings our protobuf files into conformance with the protobuf style guide, and community conventions. It is purely renames, along with generated docs. Signed-off-by: Andrew Rynhard <andrew@andrewrynhard.com>
214 lines
6.2 KiB
Go
214 lines
6.2 KiB
Go
// This Source Code Form is subject to the terms of the Mozilla Public
|
|
// License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
// file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
|
|
package backend
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"strings"
|
|
"sync"
|
|
|
|
"github.com/gogo/protobuf/proto"
|
|
"github.com/talos-systems/grpc-proxy/proxy"
|
|
"google.golang.org/grpc"
|
|
"google.golang.org/grpc/credentials"
|
|
"google.golang.org/grpc/metadata"
|
|
|
|
"github.com/talos-systems/talos/api/common"
|
|
"github.com/talos-systems/talos/pkg/constants"
|
|
)
|
|
|
|
// APID backend performs proxying to another apid instance.
|
|
//
|
|
// Backend authenticates itself using given grpc credentials.
|
|
type APID struct {
|
|
target string
|
|
creds credentials.TransportCredentials
|
|
|
|
mu sync.Mutex
|
|
conn *grpc.ClientConn
|
|
}
|
|
|
|
// NewAPID creates new instance of APID backend
|
|
func NewAPID(target string, creds credentials.TransportCredentials) (*APID, error) {
|
|
// perform very basic validation on target
|
|
if target == "" || strings.Contains(target, ":") {
|
|
return nil, fmt.Errorf("invalid target %q", target)
|
|
}
|
|
|
|
return &APID{
|
|
target: target,
|
|
creds: creds,
|
|
}, nil
|
|
}
|
|
|
|
func (a *APID) String() string {
|
|
return a.target
|
|
}
|
|
|
|
// GetConnection returns a grpc connection to the backend.
|
|
func (a *APID) GetConnection(ctx context.Context) (context.Context, *grpc.ClientConn, error) {
|
|
origMd, ok := metadata.FromIncomingContext(ctx)
|
|
|
|
md := origMd.Copy()
|
|
delete(md, "nodes")
|
|
delete(md, ":authority")
|
|
|
|
if ok {
|
|
md.Set("proxyfrom", origMd[":authority"]...)
|
|
} else {
|
|
md.Set("proxyfrom", "unknown")
|
|
}
|
|
|
|
outCtx := metadata.NewOutgoingContext(ctx, md)
|
|
|
|
a.mu.Lock()
|
|
defer a.mu.Unlock()
|
|
|
|
if a.conn != nil {
|
|
return outCtx, a.conn, nil
|
|
}
|
|
|
|
var err error
|
|
a.conn, err = grpc.DialContext(
|
|
ctx,
|
|
fmt.Sprintf("%s:%d", a.target, constants.ApidPort),
|
|
grpc.WithTransportCredentials(a.creds),
|
|
grpc.WithCodec(proxy.Codec()), //nolint: staticcheck
|
|
)
|
|
|
|
return outCtx, a.conn, err
|
|
}
|
|
|
|
// AppendInfo is called to enhance response from the backend with additional data.
|
|
//
|
|
// AppendInfo enhances upstream response with node metadata (target).
|
|
//
|
|
// This method depends on grpc protobuf response structure, each response should
|
|
// look like:
|
|
//
|
|
// message SomeReply {
|
|
// repeated SomeResponse response = 1; // please note field ID == 1
|
|
// }
|
|
//
|
|
// message SomeResponse {
|
|
// common.Metadata metadata = 1;
|
|
// <other fields go here ...>
|
|
// }
|
|
//
|
|
// As 'SomeResponse' is repeated in 'SomeReply', if we concatenate protobuf representation
|
|
// of several 'SomeReply' messages, we still get valid 'SomeReply' representation but with more
|
|
// entries (feature of protobuf binary representation).
|
|
//
|
|
// If we look at binary representation of any 'SomeReply' message, it will always contain one
|
|
// protobuf field with field ID 1 (see above) and type 2 (embedded message SomeResponse is encoded
|
|
// as string with length). So if we want to add fields to 'SomeResponse', we can simply read field
|
|
// header, adjust length for new 'SomeResponse' representation, and prepend new field header.
|
|
//
|
|
// At the same time, we can add 'common.Metadata' structure to 'SomeResponse' by simply
|
|
// appending or prepending 'common.Metadata' as a single field. This requires 'metadata'
|
|
// field to be not defined in original response. (This is due to the fact that protobuf message
|
|
// representation is concatenation of each field representation).
|
|
//
|
|
// To build only single field (Metadata) we use helper message which contains exactly this
|
|
// field with same field ID as in every other 'Response':
|
|
//
|
|
// message EmptyResponse {
|
|
// common.Metadata metadata = 1;
|
|
// }
|
|
//
|
|
// As streaming responses are not wrapped into 'SomeReply' with 'repeated', handling is simpler: we just
|
|
// need to append EmptyResponse with details.
|
|
//
|
|
// So AppendInfo does the following: validates that reply contains field ID 1 encoded as string,
|
|
// cuts field header, rest is representation of some 'Response'. Marshal 'EmptyResponse' as protobuf,
|
|
// which builds 'common.Metadata' field, append it to original 'Response' message, build new header
|
|
// for new length of some 'Response', and add back new field header.
|
|
func (a *APID) AppendInfo(streaming bool, resp []byte) ([]byte, error) {
|
|
payload, err := proto.Marshal(&common.Empty{
|
|
Metadata: &common.Metadata{
|
|
Hostname: a.target,
|
|
},
|
|
})
|
|
|
|
if streaming {
|
|
return append(resp, payload...), err
|
|
}
|
|
|
|
const (
|
|
metadataField = 1 // field number in proto definition for repeated response
|
|
metadataType = 2 // "string" for embedded messages
|
|
)
|
|
|
|
// decode protobuf embedded header
|
|
typ, n1 := proto.DecodeVarint(resp)
|
|
_, n2 := proto.DecodeVarint(resp[n1:]) // length
|
|
|
|
if typ != (metadataField<<3)|metadataType {
|
|
return nil, fmt.Errorf("unexpected message format: %d", typ)
|
|
}
|
|
|
|
if n1+n2 > len(resp) {
|
|
return nil, fmt.Errorf("unexpected message size: %d", len(resp))
|
|
}
|
|
|
|
// cut off embedded message header
|
|
resp = resp[n1+n2:]
|
|
// build new embedded message header
|
|
prefix := append(proto.EncodeVarint((metadataField<<3)|metadataType), proto.EncodeVarint(uint64(len(resp)+len(payload)))...)
|
|
resp = append(prefix, resp...)
|
|
|
|
return append(resp, payload...), err
|
|
}
|
|
|
|
// BuildError is called to convert error from upstream into response field.
|
|
//
|
|
// BuildError converts upstream error into message from upstream, so that multiple
|
|
// successful and failure responses might be returned.
|
|
//
|
|
// This simply relies on the fact that any response contains 'EmptyReply' message.
|
|
// So if 'EmptyReply' is unmarshalled into any other 'Reply' message, all the fields
|
|
// are undefined but 'Metadata':
|
|
//
|
|
// message EmptyResponse {
|
|
// common.Metadata metadata = 1;
|
|
// }
|
|
//
|
|
// message EmptyReply {
|
|
// repeated EmptyResponse response = 1;
|
|
// }
|
|
//
|
|
// Streaming responses are not wrapped into EmptyReply, so we simply marshall EmptyResponse
|
|
// message.
|
|
func (a *APID) BuildError(streaming bool, err error) ([]byte, error) {
|
|
var resp proto.Message = &common.Empty{
|
|
Metadata: &common.Metadata{
|
|
Hostname: a.target,
|
|
Error: err.Error(),
|
|
},
|
|
}
|
|
|
|
if !streaming {
|
|
resp = &common.EmptyResponse{
|
|
Messages: []*common.Empty{
|
|
resp.(*common.Empty),
|
|
},
|
|
}
|
|
}
|
|
|
|
return proto.Marshal(resp)
|
|
}
|
|
|
|
// Close connection.
|
|
func (a *APID) Close() {
|
|
a.mu.Lock()
|
|
defer a.mu.Unlock()
|
|
|
|
if a.conn != nil {
|
|
a.conn.Close() //nolint: errcheck
|
|
a.conn = nil
|
|
}
|
|
}
|