Documentation ¶
Overview ¶
Package db defines the interface that fleetspeak expects from its persistence layer. Each installation will need to choose and configure a Store implementation. An example implementation meant for testing and small scale deployments is in the server/sqlite directory.
It also includes some utility methods and types meant for use by Store implementations.
SECURITY NOTE:
The endpoints provide much of the data passed through this interface. Implementations are responsible for using safe coding practices to prevent SQL injection and similar attacks.
Index ¶
- Constants
- func ClientRetryTime() time.Time
- func ComputeBroadcastAllocation(messageLimit, allocated, sent uint64, frac float32) (toAllocate, newAllocated uint64)
- func ComputeBroadcastAllocationCleanup(allocationLimit, allocated uint64) (uint64, error)
- func Now() time.Time
- func NowProto() *tspb.Timestamp
- func ServerRetryTime(retryCount uint32) time.Time
- type AllocationInfo
- type BroadcastInfo
- type BroadcastStore
- type ClientData
- type ClientStore
- type ContactData
- type ContactID
- type FileStore
- type MessageProcessor
- type MessageStore
- type NOOPCloser
- type ReadSeekerCloser
- type Store
Constants ¶
const ( BroadcastDisabled = uint64(0) BroadcastUnlimited = uint64(math.MaxInt64) // The sqlite datastore's uint64 doesn't support full uint64 range. )
Broadcast limits with special meaning.
Variables ¶
This section is empty.
Functions ¶
func ClientRetryTime ¶
ClientRetryTime returns when a client message, being sent to the client approximately Now(), will be considered timed out and eligible to be sent again. It should be used by MessageStore implementations to determine when a message can next be provided by ClientMessagesForProcessing.
func ComputeBroadcastAllocation ¶
func ComputeBroadcastAllocation(messageLimit, allocated, sent uint64, frac float32) (toAllocate, newAllocated uint64)
ComputeBroadcastAllocation computes how large a new allocation should be. It is meant to be used by implementations of Broadcaststore.
It takes the allocation's current message limit, also the number already sent, the number already allocated, and the target fraction of the allocation to claim. It returns the number of messages that should be allocated to a new allocation, also the new total number of messages allocated.
func ComputeBroadcastAllocationCleanup ¶
ComputeBroadcastAllocationCleanup computes the new number of messages allocated when cleaning up an allocation. It takes the number of messages that were allocated to the allocation, also the current total number of messages allocated from the broadcast.
func Now ¶
Now is the clock used by the server. Normally just time.Now, but can be replaced to support testing. It should be used by db.Store implementations to determine the time.
func ServerRetryTime ¶
ServerRetryTime returns when server message, whose processing is about to start, will be considered timed out and eligible to be processed again. It should be used by MessageStore implementations to determine when a message can next be provided to a MessageProcessor.
Types ¶
type AllocationInfo ¶
type AllocationInfo struct { ID ids.AllocationID Limit uint64 Expiry time.Time }
An AllocationInfo describes an allocation. An allocation is the right to send some broadcast to up to Limit machines before Expiry.
type BroadcastInfo ¶
A BroadcastInfo describes a broadcast and contains the static broadcast information, plus the current limit and count of messages sent.
type BroadcastStore ¶
type BroadcastStore interface { // CreateBroadcast stores a new broadcast message. CreateBroadcast(ctx context.Context, b *spb.Broadcast, limit uint64) error // SetBroadcastLimit adjusts the limit of an existing broadcast. SetBroadcastLimit(ctx context.Context, id ids.BroadcastID, limit uint64) error // SaveBroadcastMessage saves a new broadcast message. SaveBroadcastMessage(ctx context.Context, msg *fspb.Message, bid ids.BroadcastID, cid common.ClientID, aid ids.AllocationID) error // ListActiveBroadcasts lists broadcasts which could be sent to some // client. ListActiveBroadcasts(ctx context.Context) ([]*BroadcastInfo, error) // ListSentBroadcasts returns identifiers for those broadcasts which have already been sent to a client. ListSentBroadcasts(ctx context.Context, id common.ClientID) ([]ids.BroadcastID, error) // CreateAllocation creates an allocation for a given Broadcast, // reserving frac of the unallocated broadcast limit until // expiry. Return nil if there is no message allocation available. CreateAllocation(ctx context.Context, id ids.BroadcastID, frac float32, expiry time.Time) (*AllocationInfo, error) // CleanupAllocation deletes the identified allocation record and // updates the broadcast sent count according to the number that were // actually sent under the given allocation. CleanupAllocation(ctx context.Context, bid ids.BroadcastID, aid ids.AllocationID) error }
BroadcastStore provides methods to store and retrieve information about broadcasts.
type ClientData ¶
type ClientData struct { Key []byte // The der encoded public key for the client. Labels []*fspb.Label // The client's labels. // Whether the client_id has been blacklisted. Once blacklisted any contact // from this client_id will result in an rekey request. Blacklisted bool }
ClientData contains basic data about a client.
type ClientStore ¶
type ClientStore interface { // ListClients returns basic information about clients. If ids is empty, it // returns all clients. ListClients(ctx context.Context, ids []common.ClientID) ([]*spb.Client, error) // StreamClientIds streams the IDs of all available clients, optionally with last contact after a given timestamp. StreamClientIds(ctx context.Context, includeBlacklisted bool, lastContactAfter *time.Time, callback func(common.ClientID) error) error // GetClientData retrieves the current data about the client identified // by id. GetClientData(ctx context.Context, id common.ClientID) (*ClientData, error) // AddClient creates a new client. AddClient(ctx context.Context, id common.ClientID, data *ClientData) error // AddClientLabel records that a client now has a label. AddClientLabel(ctx context.Context, id common.ClientID, l *fspb.Label) error // RemoveLabel records that a client no longer has a label. RemoveClientLabel(ctx context.Context, id common.ClientID, l *fspb.Label) error // BlacklistClient records that a client_id is no longer trusted and should be // recreated. BlacklistClient(ctx context.Context, id common.ClientID) error // RecordClientContact records an authenticated contact with a // client. On success provides a contact id - an opaque string which can // be used to link messages to a contact. RecordClientContact(ctx context.Context, data ContactData) (ContactID, error) // ListClientContacts lists all of the contacts in the database for a given // client. // // NOTE: This method is explicitly permitted to return data up to 30 seconds // stale. Also, it is normal (and expected) for a datastore to delete contact // older than a few weeks. ListClientContacts(ctx context.Context, id common.ClientID) ([]*spb.ClientContact, error) // StreamClientContacts is a streaming version of ListClientContacts. StreamClientContacts(ctx context.Context, id common.ClientID, callback func(*spb.ClientContact) error) error // LinkMessagesToContact associates messages with a contact - it records // that they were sent or received during the given contact. LinkMessagesToContact(ctx context.Context, contact ContactID, msgs []common.MessageID) error // Writes resource-usage data received from a client to the data-store. RecordResourceUsageData(ctx context.Context, id common.ClientID, rud *mpb.ResourceUsageData) error // Fetches resource-usage records for a given client and time range from the data-store. FetchResourceUsageRecords(ctx context.Context, id common.ClientID, startTimestamp, endTimestamp *tspb.Timestamp) ([]*spb.ClientResourceUsageRecord, error) }
ClientStore provides methods to store and retrieve information about clients.
type ContactData ¶
type ContactData struct { ClientID common.ClientID // ID of the client. NonceSent, NonceReceived uint64 // Nonce sent to the client and received from the client. Addr string // Observed client network address. ClientClock *tspb.Timestamp // Client's report of its current clock setting. // If non-empty, indicates that the contact is or was a streaming contact to // the listed FS server. (As defined by notifications module being used.) StreamingTo string }
ContactData provides basic information about a client's contact with a FS server.
type ContactID ¶
type ContactID string
A ContactID identifies a communication with a client. The form is determined by the Datastore implementation - it is treated as an opaque string by the rest of the FS system.
type FileStore ¶
type FileStore interface { // StoreFile stores data into the Filestore, organized by service and name. StoreFile(ctx context.Context, service, name string, data io.Reader) error // StatFile returns the modification time of a file previously stored by // StoreFile. Returns ErrNotFound if not found. StatFile(ctx context.Context, servce, name string) (time.Time, error) // ReadFile returns the data and modification time of file previously // stored by StoreFile. Caller is responsible for closing data. // // Note: Calls to data are permitted to fail if ctx is canceled or expired. ReadFile(ctx context.Context, service, name string) (data ReadSeekerCloser, modtime time.Time, err error) }
FileStore provides methods to store and retrieve files. Files are keyed by an associated service and name.
SECURITY NOTES:
Fleetspeak doesn't provide any ACL support for files - all files are readable by any client.
Implementations are responsible for validating and/or sanitizing the identifiers provided. For example, an implementation backed by a filesystem would need to protect against path traversal vulnerabilities.
type MessageProcessor ¶
type MessageProcessor interface { // ProcessMessage is called by the Datastore to indicate that the // provided message is overdue and that processing should be attempted // again. // // This call will be repeated until MarkMessage(Processed|Failed) is // successfully called on m. ProcessMessages(msgs []*fspb.Message) }
A MessageProcessor receives messages that are overdue and should be reprocessing.
type MessageStore ¶
type MessageStore interface { // StoreMessages records msgs. If contact is not the empty string, it attaches // them to the associated contact. // // It is not an error for a message to already exist. In this case a success result // will overwrite any result already present, and a failed result will overwrite // an empty result. Otherwise the operation is silently dropped. // // A message is eligible to be returned by ClientMessagesForProcessing or the // registered MessageProcessor iff it does not yet have a Result. Also, // setting a Result will delete the message's Data field payload. StoreMessages(ctx context.Context, msgs []*fspb.Message, contact ContactID) error // DeletePendingMessages removes all pending messages for given clients. DeletePendingMessages(ctx context.Context, ids []common.ClientID) error // GetPendingMessageCount returns the number of pending messages for the given clients. GetPendingMessageCount(ctx context.Context, ids []common.ClientID) (uint64, error) // GetPendingMessages returns pending messages for given clients. // If wantData is true, message data is retrieved as well. // offset and limit are optional, set them to 0 to retrieve all pending messages. GetPendingMessages(ctx context.Context, ids []common.ClientID, offset uint64, limit uint64, wantData bool) ([]*fspb.Message, error) // ClientMessagesForProcessing returns messages that are due to be // processed by a client. It also increments the time at which the // messages will again become overdue using rp. // // The lim parameter indicates the maximum number of messages to to // retrieve. If non-nil, serviceLimits further limits the number returned // for each service. // // Note that if an error occurs partway through the loading of messages, // the already loaded messages may be returned along with the error. In // particular, datastore implementations might do this if the ctx times // out before all messages are found and updated. Any such are message // is valid and ready for processing. ClientMessagesForProcessing(ctx context.Context, id common.ClientID, lim uint64, serviceLimits map[string]uint64) ([]*fspb.Message, error) // GetMessages retrieves specific messages. GetMessages(ctx context.Context, ids []common.MessageID, wantData bool) ([]*fspb.Message, error) // GetMessageResult retrieves the current status of a message. GetMessageResult(ctx context.Context, id common.MessageID) (*fspb.MessageResult, error) // SetMessageResult retrieves the current status of a message. The dest // parameter identifies the destination client, and must be the empty id for // messages addressed to the server. SetMessageResult(ctx context.Context, dest common.ClientID, id common.MessageID, res *fspb.MessageResult) error // RegisterMessageProcessor installs a MessageProcessor which will be // called when a message is overdue for processing. RegisterMessageProcessor(mp MessageProcessor) // StopMessageProcessor causes the datastore to stop making calls to the // registered MessageProcessor. It only returns once all existing calls // to MessageProcessor have completed. StopMessageProcessor() }
MessageStore provides methods to store and query messages.
Notionally, a MessageStore is backed by a table where each row is a fspb.Message record, along with with one of the following:
1) due time and retry count - If the message is not processed or delivered before due time, it will be tried again. A count of the number of retries is maintained and used to compute the next due time.
2) completion time - When a record is processed or acknowledged by a client, the message is marked as completed by saving a completion time.
Furthermore it is possible to register a MessageProcessor with each MessageStore which then receives notifications that server messages are ready for processing. In multi-server installations, the datastore should attempt to provide eactly one notification to some Fleetspeak server each time the message becomes overdue.
type NOOPCloser ¶
type NOOPCloser struct {
io.ReadSeeker
}
NOOPCloser wraps an io.ReadSeeker to trivially turn it into a ReadSeekerCloser.
type ReadSeekerCloser ¶
type ReadSeekerCloser interface { io.ReadSeeker io.Closer }
ReadSeekerCloser groups io.ReadSeeker and io.Closer.
type Store ¶
type Store interface { MessageStore ClientStore BroadcastStore FileStore // IsNotFound returns whether an error returned by the Datastore indicates that // a record was not found. IsNotFound(error) bool // Close shuts down the Store, releasing any held resources. Close() error }
A Store describes the full persistence mechanism required by the base fleetspeak system. These operations must be thread safe. These must also be all-or-nothing, fully committed on success, and are otherwise trusted to be individually transactional.