/////////////////////////////////////////////////////////////////////////////// // Copyright © 2022 xx network SEZC // // // // Use of this source code is governed by a license that can be found in the // // LICENSE file // /////////////////////////////////////////////////////////////////////////////// package bindings import ( "crypto/ed25519" "encoding/base64" "encoding/json" "sync" "time" "github.com/pkg/errors" "gitlab.com/elixxir/client/channels" "gitlab.com/elixxir/client/cmix/rounds" "gitlab.com/elixxir/client/storage/utility" "gitlab.com/elixxir/client/storage/versioned" "gitlab.com/elixxir/client/xxdk" cryptoBroadcast "gitlab.com/elixxir/crypto/broadcast" cryptoChannel "gitlab.com/elixxir/crypto/channel" "gitlab.com/elixxir/crypto/rsa" "gitlab.com/xx_network/primitives/id" "gitlab.com/xx_network/primitives/id/ephemeral" "gitlab.com/xx_network/primitives/netTime" ) //////////////////////////////////////////////////////////////////////////////// // Singleton Tracker // //////////////////////////////////////////////////////////////////////////////// // channelManagerTrackerSingleton is used to track ChannelsManager objects // so that they can be referenced by ID back over the bindings. var channelManagerTrackerSingleton = &channelManagerTracker{ tracked: make(map[int]*ChannelsManager), count: 0, } // channelManagerTracker is a singleton used to keep track of extant // ChannelsManager objects, preventing race conditions created by passing it // over the bindings. type channelManagerTracker struct { tracked map[int]*ChannelsManager count int mux sync.RWMutex } // make create a ChannelsManager from an [channels.Manager], assigns it a unique // ID, and adds it to the channelManagerTracker. func (cmt *channelManagerTracker) make(c channels.Manager) *ChannelsManager { cmt.mux.Lock() defer cmt.mux.Unlock() chID := cmt.count cmt.count++ cmt.tracked[chID] = &ChannelsManager{ api: c, id: chID, } return cmt.tracked[chID] } // get an ChannelsManager from the channelManagerTracker given its ID. func (cmt *channelManagerTracker) get(id int) (*ChannelsManager, error) { cmt.mux.RLock() defer cmt.mux.RUnlock() c, exist := cmt.tracked[id] if !exist { return nil, errors.Errorf( "Cannot get ChannelsManager for ID %d, does not exist", id) } return c, nil } // delete removes a ChannelsManager from the channelManagerTracker. func (cmt *channelManagerTracker) delete(id int) { cmt.mux.Lock() defer cmt.mux.Unlock() delete(cmt.tracked, id) } //////////////////////////////////////////////////////////////////////////////// // Basic Channel API // //////////////////////////////////////////////////////////////////////////////// // ChannelsManager is a bindings-layer struct that wraps a [channels.Manager] // interface. type ChannelsManager struct { api channels.Manager id int } // GetID returns the channelManagerTracker ID for the ChannelsManager object. func (cm *ChannelsManager) GetID() int { return cm.id } // GenerateChannelIdentity creates a new private channel identity // ([channel.PrivateIdentity]). The public component can be retrieved as JSON // via [GetPublicChannelIdentityFromPrivate]. // // Parameters: // - cmixID - The tracked cmix object ID. This can be retrieved using // [Cmix.GetID]. // // Returns: // - Marshalled bytes of [channel.PrivateIdentity]. func GenerateChannelIdentity(cmixID int) ([]byte, error) { // Get user from singleton user, err := cmixTrackerSingleton.get(cmixID) if err != nil { return nil, err } rng := user.api.GetRng().GetStream() defer rng.Close() pi, err := cryptoChannel.GenerateIdentity(rng) if err != nil { return nil, err } return pi.Marshal(), nil } // ConstructIdentity constructs a [channel.Identity] from a user's public key // and codeset version. // // Parameters: // - pubKey - The Ed25519 public key. // - codesetVersion - The version of the codeset used to generate the identity. // // Returns: // - JSON of [channel.Identity]. func ConstructIdentity(pubKey []byte, codesetVersion int) ([]byte, error) { identity, err := cryptoChannel.ConstructIdentity( pubKey, uint8(codesetVersion)) if err != nil { return nil, err } return json.Marshal(identity) } // ImportPrivateIdentity generates a new [channel.PrivateIdentity] from exported // data. // // Parameters: // - password - The password used to encrypt the identity. // - data - The encrypted data. // // Returns: // - JSON of [channel.PrivateIdentity]. func ImportPrivateIdentity(password string, data []byte) ([]byte, error) { pi, err := cryptoChannel.ImportPrivateIdentity(password, data) if err != nil { return nil, err } return pi.Marshal(), nil } // GetPublicChannelIdentity constructs a public identity ([channel.Identity]) // from a bytes version and returns it JSON marshaled. // // Parameters: // - marshaledPublic - Bytes of the public identity ([channel.Identity]). // // Returns: // - JSON of the constructed [channel.Identity]. func GetPublicChannelIdentity(marshaledPublic []byte) ([]byte, error) { i, err := cryptoChannel.UnmarshalIdentity(marshaledPublic) if err != nil { return nil, err } return json.Marshal(&i) } // GetPublicChannelIdentityFromPrivate returns the public identity // ([channel.Identity]) contained in the given private identity // ([channel.PrivateIdentity]). // // Parameters: // - marshaledPrivate - Bytes of the private identity // (channel.PrivateIdentity]). // // Returns: // - JSON of the public [channel.Identity]. func GetPublicChannelIdentityFromPrivate(marshaledPrivate []byte) ([]byte, error) { pi, err := cryptoChannel.UnmarshalPrivateIdentity(marshaledPrivate) if err != nil { return nil, err } return json.Marshal(&pi.Identity) } // NewChannelsManagerGoEventModel creates a new [ChannelsManager] from a new // private identity ([channel.PrivateIdentity]). This is not compatible with // GoMobile Bindings because it receives the go event model. // // This is for creating a manager for an identity for the first time. For // generating a new one channel identity, use [GenerateChannelIdentity]. To // reload this channel manager, use [LoadChannelsManagerGoEventModel], passing // in the storage tag retrieved by [ChannelsManager.GetStorageTag]. // // Parameters: // - cmixID - The tracked Cmix object ID. This can be retrieved using // [Cmix.GetID]. // - privateIdentity - Bytes of a private identity ([channel.PrivateIdentity]) // that is generated by [GenerateChannelIdentity]. // - goEvent - A function that initialises and returns the event model that is // not compatible with GoMobile bindings. func NewChannelsManagerGoEventModel(cmixID int, privateIdentity []byte, goEventBuilder channels.EventModelBuilder) (*ChannelsManager, error) { pi, err := cryptoChannel.UnmarshalPrivateIdentity(privateIdentity) if err != nil { return nil, err } // Get user from singleton user, err := cmixTrackerSingleton.get(cmixID) if err != nil { return nil, err } // Construct new channels manager m, err := channels.NewManager(pi, user.api.GetStorage().GetKV(), user.api.GetCmix(), user.api.GetRng(), goEventBuilder) if err != nil { return nil, err } // Add channel to singleton and return return channelManagerTrackerSingleton.make(m), nil } // LoadChannelsManagerGoEventModel loads an existing ChannelsManager. This is not // compatible with GoMobile Bindings because it receives the go event model. // This is for creating a manager for an identity for the first time. // The channel manager should have first been created with // NewChannelsManagerGoEventModel and then the storage tag can be retrieved // with ChannelsManager.GetStorageTag // // Parameters: // - cmixID - The tracked cmix object ID. This can be retrieved using // [Cmix.GetID]. // - storageTag - retrieved with ChannelsManager.GetStorageTag // - goEvent - A function that initialises and returns the event model that is // not compatible with GoMobile bindings. func LoadChannelsManagerGoEventModel(cmixID int, storageTag string, goEventBuilder channels.EventModelBuilder) (*ChannelsManager, error) { // Get user from singleton user, err := cmixTrackerSingleton.get(cmixID) if err != nil { return nil, err } // Construct new channels manager m, err := channels.LoadManager(storageTag, user.api.GetStorage().GetKV(), user.api.GetCmix(), user.api.GetRng(), goEventBuilder) if err != nil { return nil, err } // Add channel to singleton and return return channelManagerTrackerSingleton.make(m), nil } // NewChannelsManager creates a new [ChannelsManager] from a new private // identity ([channel.PrivateIdentity]). // // This is for creating a manager for an identity for the first time. For // generating a new one channel identity, use [GenerateChannelIdentity]. To // reload this channel manager, use [LoadChannelsManager], passing in the // storage tag retrieved by [ChannelsManager.GetStorageTag]. // // Parameters: // - cmixID - The tracked Cmix object ID. This can be retrieved using // [Cmix.GetID]. // - privateIdentity - Bytes of a private identity ([channel.PrivateIdentity]) // that is generated by [GenerateChannelIdentity]. // - event - An interface that contains a function that initialises and returns // the event model that is bindings-compatible. func NewChannelsManager(cmixID int, privateIdentity []byte, eventBuilder EventModelBuilder) (*ChannelsManager, error) { pi, err := cryptoChannel.UnmarshalPrivateIdentity(privateIdentity) if err != nil { return nil, err } // Get user from singleton user, err := cmixTrackerSingleton.get(cmixID) if err != nil { return nil, err } eb := func(path string) (channels.EventModel, error) { return NewEventModel(eventBuilder.Build(path)), nil } // Construct new channels manager m, err := channels.NewManager(pi, user.api.GetStorage().GetKV(), user.api.GetCmix(), user.api.GetRng(), eb) if err != nil { return nil, err } // Add channel to singleton and return return channelManagerTrackerSingleton.make(m), nil } // LoadChannelsManager loads an existing [ChannelsManager]. // // This is for loading a manager for an identity that has already been created. // The channel manager should have previously been created with // [NewChannelsManager] and the storage is retrievable with // [ChannelsManager.GetStorageTag]. // // Parameters: // - cmixID - The tracked cmix object ID. This can be retrieved using // [Cmix.GetID]. // - storageTag - The storage tag associated with the previously created // channel manager and retrieved with [ChannelsManager.GetStorageTag]. // - event - An interface that contains a function that initialises and returns // the event model that is bindings-compatible. func LoadChannelsManager(cmixID int, storageTag string, eventBuilder EventModelBuilder) (*ChannelsManager, error) { // Get user from singleton user, err := cmixTrackerSingleton.get(cmixID) if err != nil { return nil, err } eb := func(path string) (channels.EventModel, error) { return NewEventModel(eventBuilder.Build(path)), nil } // Construct new channels manager m, err := channels.LoadManager(storageTag, user.api.GetStorage().GetKV(), user.api.GetCmix(), user.api.GetRng(), eb) if err != nil { return nil, err } // Add channel to singleton and return return channelManagerTrackerSingleton.make(m), nil } // ChannelGeneration contains information about a newly generated channel. It // contains the public channel info formatted in pretty print and the private // key for the channel in PEM format. // // Example JSON: // { // "Channel": "\u003cSpeakeasy-v3:name|description:desc|level:Public|created:1665489600000000000|secrets:zjHmrPPMDQ0tNSANjAmQfKhRpJIdJMU+Hz5hsZ+fVpk=|qozRNkADprqb38lsnU7WxCtGCq9OChlySCEgl4NHjI4=|2|328|7aZQAtuVjE84q4Z09iGytTSXfZj9NyTa6qBp0ueKjCI=\u003e", // "PrivateKey": "-----BEGIN RSA PRIVATE KEY-----\nMCYCAQACAwDVywIDAQABAgMAlVECAgDvAgIA5QICAJECAgCVAgIA1w==\n-----END RSA PRIVATE KEY-----" // } type ChannelGeneration struct { Channel string PrivateKey string } // GenerateChannel is used to create a channel a new channel of which you are // the admin. It is only for making new channels, not joining existing ones. // // It returns a pretty print of the channel and the private key. // // Parameters: // - cmixID - The tracked cmix object ID. This can be retrieved using // [Cmix.GetID]. // - name - The name of the new channel. The name must be between 3 and 24 // characters inclusive. It can only include upper and lowercase unicode // letters, digits 0 through 9, and underscores (_). It cannot be changed // once a channel is created. // - description - The description of a channel. The description is optional // but cannot be longer than 144 characters and can include all unicode // characters. It cannot be changed once a channel is created. // - privacyLevel - The broadcast.PrivacyLevel of the channel. 0 = public, // 1 = private, and 2 = secret. Refer to the comment below for more // information. // // Returns: // - []byte - [ChannelGeneration] describes a generated channel. It contains // both the public channel info and the private key for the channel in PEM // format. // // The [broadcast.PrivacyLevel] of a channel indicates the level of channel // information revealed when sharing it via URL. For any channel besides public // channels, the secret information is encrypted and a password is required to // share and join a channel. // - A privacy level of [broadcast.Public] reveals all the information // including the name, description, privacy level, public key and salt. // - A privacy level of [broadcast.Private] reveals only the name and // description. // - A privacy level of [broadcast.Secret] reveals nothing. func GenerateChannel(cmixID int, name, description string, privacyLevel int) ([]byte, error) { // Get cmix from singleton so its rng can be used cmix, err := cmixTrackerSingleton.get(cmixID) if err != nil { return nil, err } stream := cmix.api.GetRng().GetStream() defer stream.Close() level := cryptoBroadcast.PrivacyLevel(privacyLevel) c, pk, err := cryptoBroadcast.NewChannel(name, description, level, cmix.api.GetCmix().GetMaxMessageLength(), stream) if err != nil { return nil, err } gen := ChannelGeneration{ Channel: c.PrettyPrint(), PrivateKey: string(pk.MarshalPem()), } err = saveChannelPrivateKey(cmix, c.ReceptionID, pk) if err != nil { return nil, err } return json.Marshal(&gen) } const ( channelPrivateKeyStoreVersion = 0 channelPrivateKeyStoreKey = "channelPrivateKey" ) func saveChannelPrivateKey(cmix *Cmix, channelID *id.ID, pk rsa.PrivateKey) error { return cmix.api.GetStorage().Set( makeChannelPrivateKeyStoreKey(channelID), &versioned.Object{ Version: channelPrivateKeyStoreVersion, Timestamp: netTime.Now(), Data: pk.MarshalPem(), }) } // GetSavedChannelPrivateKeyUNSAFE loads the private key from storage for the // given channel ID. // // NOTE: This function is unsafe and only for debugging purposes only. // // Parameters: // - cmixID - ID of [Cmix] object in tracker. // - channelIdBase64 - The [id.ID] of the channel in base 64 encoding. // // Returns: // - The PEM file of the private key. func GetSavedChannelPrivateKeyUNSAFE(cmixID int, channelIdBase64 string) (string, error) { cmix, err := cmixTrackerSingleton.get(cmixID) if err != nil { return "", err } channelIdBytes, err := base64.StdEncoding.DecodeString(channelIdBase64) if err != nil { return "", errors.Errorf("failed to decode channel ID: %+v", err) } channelID, err := id.Unmarshal(channelIdBytes) if err != nil { return "", errors.Errorf("invalid channel ID: %+v", err) } privKey, err := loadChannelPrivateKey(cmix, channelID) if err != nil { return "", errors.Errorf( "failed to load private key from storage: %+v", err) } return string(privKey.MarshalPem()), nil } func loadChannelPrivateKey(cmix *Cmix, channelID *id.ID) (rsa.PrivateKey, error) { obj, err := cmix.api.GetStorage().Get( makeChannelPrivateKeyStoreKey(channelID)) if err != nil { return nil, err } return rsa.GetScheme().UnmarshalPrivateKeyPEM(obj.Data) } func makeChannelPrivateKeyStoreKey(channelID *id.ID) string { return channelPrivateKeyStoreKey + "/" + channelID.String() } // DecodePublicURL decodes the channel URL into a channel pretty print. This // function can only be used for public channel URLs. To get the privacy level // of a channel URL, use [GetShareUrlType]. // // Parameters: // - url - The channel's share URL. Should be received from another user or // generated via [GetShareURL]. // // Returns: // - The channel pretty print. func DecodePublicURL(url string) (string, error) { c, err := cryptoBroadcast.DecodeShareURL(url, "") if err != nil { return "", err } return c.PrettyPrint(), nil } // DecodePrivateURL decodes the channel URL, using the password, into a channel // pretty print. This function can only be used for private or secret channel // URLs. To get the privacy level of a channel URL, use [GetShareUrlType]. // // Parameters: // - url - The channel's share URL. Should be received from another user or // generated via [GetShareURL]. // - password - The password needed to decrypt the secret data in the URL. // // Returns: // - The channel pretty print. func DecodePrivateURL(url, password string) (string, error) { c, err := cryptoBroadcast.DecodeShareURL(url, password) if err != nil { return "", err } return c.PrettyPrint(), nil } // GetChannelJSON returns the JSON of the channel for the given pretty print. // // Parameters: // - prettyPrint - The pretty print of the channel. // // Returns: // - JSON of the [broadcast.Channel] object. // // Example JSON of [broadcast.Channel]: // { // "ReceptionID": "Ja/+Jh+1IXZYUOn+IzE3Fw/VqHOscomD0Q35p4Ai//kD", // "Name": "My_Channel", // "Description": "Here is information about my channel.", // "Salt": "+tlrU/htO6rrV3UFDfpQALUiuelFZ+Cw9eZCwqRHk+g=", // "RsaPubKeyHash": "PViT1mYkGBj6AYmE803O2RpA7BX24EjgBdldu3pIm4o=", // "RsaPubKeyLength": 5, // "RSASubPayloads": 1, // "Secret": "JxZt/wPx2luoPdHY6jwbXqNlKnixVU/oa9DgypZOuyI=", // "Level": 0 // } func GetChannelJSON(prettyPrint string) ([]byte, error) { c, err := cryptoBroadcast.NewChannelFromPrettyPrint(prettyPrint) if err != nil { return nil, nil } return json.Marshal(c) } // ChannelInfo contains information about a channel. // // Example of ChannelInfo JSON: // { // "Name": "Test Channel", // "Description": "This is a test channel", // "ChannelID": "RRnpRhmvXtW9ugS1nILJ3WfttdctDvC2jeuH43E0g/0D", // } type ChannelInfo struct { Name string Description string ChannelID string } // GetChannelInfo returns the info about a channel from its public description. // // Parameters: // - prettyPrint - The pretty print of the channel. // // The pretty print will be of the format: // <Speakeasy-v3:Test_Channel|description:Channel description.|level:Public|created:1666718081766741100|secrets:+oHcqDbJPZaT3xD5NcdLY8OjOMtSQNKdKgLPmr7ugdU=|rCI0wr01dHFStjSFMvsBzFZClvDIrHLL5xbCOPaUOJ0=|493|1|7cBhJxVfQxWo+DypOISRpeWdQBhuQpAZtUbQHjBm8NQ=> // // Returns: // - []byte - JSON of [ChannelInfo], which describes all relevant channel info. func GetChannelInfo(prettyPrint string) ([]byte, error) { _, bytes, err := getChannelInfo(prettyPrint) return bytes, err } func getChannelInfo(prettyPrint string) (*cryptoBroadcast.Channel, []byte, error) { c, err := cryptoBroadcast.NewChannelFromPrettyPrint(prettyPrint) if err != nil { return nil, nil, err } ci := &ChannelInfo{ Name: c.Name, Description: c.Description, ChannelID: c.ReceptionID.String(), } bytes, err := json.Marshal(ci) if err != nil { return nil, nil, err } return c, bytes, nil } // JoinChannel joins the given channel. It will fail if the channel has already // been joined. // // Parameters: // - channelPretty - A portable channel string. Should be received from // another user or generated via GenerateChannel. // // The pretty print will be of the format: // <Speakeasy-v3:Test_Channel|description:Channel description.|level:Public|created:1666718081766741100|secrets:+oHcqDbJPZaT3xD5NcdLY8OjOMtSQNKdKgLPmr7ugdU=|rCI0wr01dHFStjSFMvsBzFZClvDIrHLL5xbCOPaUOJ0=|493|1|7cBhJxVfQxWo+DypOISRpeWdQBhuQpAZtUbQHjBm8NQ=> // // Returns: // - []byte - JSON of [ChannelInfo], which describes all relevant channel info. func (cm *ChannelsManager) JoinChannel(channelPretty string) ([]byte, error) { c, info, err := getChannelInfo(channelPretty) if err != nil { return nil, err } // Join the channel using the API err = cm.api.JoinChannel(c) return info, err } // GetChannels returns the IDs of all channels that have been joined. // // Returns: // - []byte - A JSON marshalled list of IDs. // // JSON Example: // { // "U4x/lrFkvxuXu59LtHLon1sUhPJSCcnZND6SugndnVID", // "15tNdkKbYXoMn58NO6VbDMDWFEyIhTWEGsvgcJsHWAgD" // } func (cm *ChannelsManager) GetChannels() ([]byte, error) { channelIds := cm.api.GetChannels() return json.Marshal(channelIds) } // LeaveChannel leaves the given channel. It will return an error if the // channel was not previously joined. // // Parameters: // - marshalledChanId - A JSON marshalled channel ID ([id.ID]). func (cm *ChannelsManager) LeaveChannel(marshalledChanId []byte) error { // Unmarshal channel ID channelId, err := id.Unmarshal(marshalledChanId) if err != nil { return err } // Leave the channel return cm.api.LeaveChannel(channelId) } // ReplayChannel replays all messages from the channel within the network's // memory (~3 weeks) over the event model. // // Parameters: // - marshalledChanId - A JSON marshalled channel ID ([id.ID]). func (cm *ChannelsManager) ReplayChannel(marshalledChanId []byte) error { // Unmarshal channel ID chanId, err := id.Unmarshal(marshalledChanId) if err != nil { return err } // Replay channel return cm.api.ReplayChannel(chanId) } //////////////////////////////////////////////////////////////////////////////// // Channel Share URL // //////////////////////////////////////////////////////////////////////////////// // ShareURL is returned from ChannelsManager.GetShareURL. It includes the // channel's share URL and password, if it needs one. // // JSON example for a public channel: // { // "url": "https://internet.speakeasy.tech/?0Name=name&1Description=desc&2Level=Public&3Created=1665489600000000000&e=%2FWNZvuHPuv%2Bx23XbZXVNzCi7y8rUSxkh75MpR9UrsCo%3D&k=ddX1CH52xH%2F%2Fb6lKrbvDghdSmCQr90ktsOAZ%2FrhEonI%3D&l=2&m=0&p=328&s=%2FD%2FoQP2mio3XAWfhmWF0xmZrpj4nAsb9JLXj%2B0Mzq9Y%3D&v=1", // "password": "" // } // // JSON example for a private channel: // { // "url": "https://internet.speakeasy.tech/?0Name=name&1Description=desc&3Created=1665489600000000000&d=5AZQirb%2FYrmUITLn%2FFzCaGek1APfJnd2q0KwORGj%2BnbGg26kTShG6cfD3w6c%2BA3RDzxuKDSDN0zS4n1LbjiGe0KYdb8eJVeyRZtld516hfojNDXNAwZq8zbeZy4jjbF627fcLHRNS%2FaII4uJ5UB3gLUeBeZGraaybCCu3FIj1N4RbcJ5cQgT7hBf93bHmJc%3D&m=0&v=1", // "password": "tribune gangrene labrador italics nutmeg process exhume legal" // } // // JSON example for a secret channel: // { // "url": "https://internet.speakeasy.tech/?d=w5evLthm%2Fq2j11g6PPtV0QoLaAqNCIER0OqxhxL%2FhpGVJI0057ZPgGBrKoJNE1%2FdoVuU35%2FhohuW%2BWvGlx6IuHoN6mDj0HfNj6Lo%2B8GwIaD6jOEwUcH%2FMKGsKnoqFsMaMPd5gXYgdHvA8l5SRe0gSCVqGKUaG6JgL%2FDu4iyjY7v4ykwZdQ7soWOcBLHDixGEkVLpwsCrPVHkT2K0W6gV74GIrQ%3D%3D&m=0&v=1", // "password": "frenzy contort staple thicket consuming affiliate scion demeanor" // } type ShareURL struct { URL string `json:"url"` Password string `json:"password"` } // GetShareURL generates a URL that can be used to share this channel with // others on the given host. // // A URL comes in one of three forms based on the privacy level set when // generating the channel. Each privacy level hides more information than the // last with the lowest level revealing everything and the highest level // revealing nothing. For any level above the lowest, a password is returned, // which will be required when decoding the URL. // // The maxUses is the maximum number of times this URL can be used to join a // channel. If it is set to 0, then it can be shared unlimited times. The max // uses is set as a URL parameter using the key [broadcast.MaxUsesKey]. Note // that this number is also encoded in the secret data for private and secret // URLs, so if the number is changed in the URL, is will be verified when // calling [ChannelsManager.JoinChannelFromURL]. There is no enforcement for // public URLs. // // Parameters: // - cmixID - The tracked Cmix object ID. // - host - The URL to append the channel info to. // - maxUses - The maximum number of uses the link can be used (0 for // unlimited). // - marshalledChanId - A marshalled channel ID ([id.ID]). // // Returns: // - JSON of ShareURL. func (cm *ChannelsManager) GetShareURL(cmixID int, host string, maxUses int, marshalledChanId []byte) ([]byte, error) { // Unmarshal channel ID chanId, err := id.Unmarshal(marshalledChanId) if err != nil { return nil, err } // Get the channel from the ID ch, err := cm.api.GetChannel(chanId) if err != nil { return nil, err } // Get user from singleton user, err := cmixTrackerSingleton.get(cmixID) if err != nil { return nil, err } // Generate share URL and password rng := user.api.GetRng().GetStream() url, password, err := ch.ShareURL(host, maxUses, rng) rng.Close() if err != nil { return nil, err } su := ShareURL{ URL: url, Password: password, } return json.Marshal(su) } // GetShareUrlType determines the [broadcast.PrivacyLevel] of the channel URL. // If the URL is an invalid channel URL, an error is returned. // // Parameters: // - url - The channel share URL. // // Returns: // - An int that corresponds to the [broadcast.PrivacyLevel] as outlined below. // // Possible returns: // 0 = public channel // 1 = private channel // 2 = secret channel func GetShareUrlType(url string) (int, error) { level, err := cryptoBroadcast.GetShareUrlType(url) return int(level), err } //////////////////////////////////////////////////////////////////////////////// // Channel Sending Methods & Reports // //////////////////////////////////////////////////////////////////////////////// // ChannelSendReport is the bindings' representation of the return values of // ChannelsManager's Send operations. // // JSON Example: // { // "MessageId": "0kitNxoFdsF4q1VMSI/xPzfCnGB2l+ln2+7CTHjHbJw=", // "Rounds":[1,5,9], // "EphId": 0 // } type ChannelSendReport struct { MessageId []byte RoundsList EphId int64 } // SendGeneric is used to send a raw message over a channel. In general, it // should be wrapped in a function that defines the wire protocol. If the final // message, before being sent over the wire, is too long, this will return an // error. Due to the underlying encoding using compression, it isn't possible to // define the largest payload that can be sent, but it will always be possible // to send a payload of 802 bytes at minimum. The meaning of validUntil depends // on the use case. // // Parameters: // - marshalledChanId - A JSON marshalled channel ID ([id.ID]). // - messageType - The message type of the message. This will be a valid // [channels.MessageType]. // - message - The contents of the message. This need not be of data type // string, as the message could be a specified format that the channel may // recognize. // - leaseTimeMS - The lease of the message. This will be how long the message // is valid until, in milliseconds. As per the channels.Manager // documentation, this has different meanings depending on the use case. // These use cases may be generic enough that they will not be enumerated // here. // - cmixParamsJSON - A JSON marshalled [xxdk.CMIXParams]. This may be empty, // and GetDefaultCMixParams will be used internally. // // Returns: // - []byte - A JSON marshalled ChannelSendReport. func (cm *ChannelsManager) SendGeneric(marshalledChanId []byte, messageType int, message []byte, leaseTimeMS int64, cmixParamsJSON []byte) ([]byte, error) { // Unmarshal channel ID and parameters chanId, params, err := parseChannelsParameters( marshalledChanId, cmixParamsJSON) if err != nil { return nil, err } msgTy := channels.MessageType(messageType) // Send message chanMsgId, rnd, ephId, err := cm.api.SendGeneric(chanId, msgTy, message, time.Duration(leaseTimeMS), params.CMIX) if err != nil { return nil, err } // Construct send report return constructChannelSendReport(chanMsgId, rnd.ID, ephId) } // SendAdminGeneric is used to send a raw message over a channel encrypted with // admin keys, identifying it as sent by the admin. In general, it should be // wrapped in a function that defines the wire protocol. If the final message, // before being sent over the wire, is too long, this will return an error. The // message must be at most 510 bytes long. // // Parameters: // - adminPrivateKey - The PEM-encoded admin RSA private key. // - marshalledChanId - A JSON marshalled channel ID ([id.ID]). // - messageType - The message type of the message. This will be a valid // [channels.MessageType]. // - message - The contents of the message. The message should be at most 510 // bytes. This need not be of data type string, as the message could be a // specified format that the channel may recognize. // - leaseTimeMS - The lease of the message. This will be how long the message // is valid until, in milliseconds. As per the channels.Manager // documentation, this has different meanings depending on the use case. // These use cases may be generic enough that they will not be enumerated // here. // - cmixParamsJSON - A JSON marshalled [xxdk.CMIXParams]. This may be empty, // and GetDefaultCMixParams will be used internally. // // Returns: // - []byte - A JSON marshalled ChannelSendReport. func (cm *ChannelsManager) SendAdminGeneric(adminPrivateKey, marshalledChanId []byte, messageType int, message []byte, leaseTimeMS int64, cmixParamsJSON []byte) ([]byte, error) { // Load private key from file rsaPrivKey, err := rsa.GetScheme().UnmarshalPrivateKeyPEM(adminPrivateKey) if err != nil { return nil, err } // Unmarshal channel ID and parameters chanId, params, err := parseChannelsParameters( marshalledChanId, cmixParamsJSON) if err != nil { return nil, err } msgTy := channels.MessageType(messageType) // Send admin message chanMsgId, rnd, ephId, err := cm.api.SendAdminGeneric(rsaPrivKey, chanId, msgTy, message, time.Duration(leaseTimeMS), params.CMIX) // Construct send report return constructChannelSendReport(chanMsgId, rnd.ID, ephId) } // SendMessage is used to send a formatted message over a channel. // Due to the underlying encoding using compression, it isn't possible to define // the largest payload that can be sent, but it will always be possible to send // a payload of 798 bytes at minimum. // // The message will auto delete validUntil after the round it is sent in, // lasting forever if [channels.ValidForever] is used. // // Parameters: // - marshalledChanId - A JSON marshalled channel ID ([id.ID]). // - message - The contents of the message. The message should be at most 510 // bytes. This is expected to be Unicode, and thus a string data type is // expected // - leaseTimeMS - The lease of the message. This will be how long the message // is valid until, in milliseconds. As per the channels.Manager // documentation, this has different meanings depending on the use case. // These use cases may be generic enough that they will not be enumerated // here. // - cmixParamsJSON - A JSON marshalled [xxdk.CMIXParams]. This may be // empty, and GetDefaultCMixParams will be used internally. // // Returns: // - []byte - A JSON marshalled ChannelSendReport func (cm *ChannelsManager) SendMessage(marshalledChanId []byte, message string, leaseTimeMS int64, cmixParamsJSON []byte) ([]byte, error) { // Unmarshal channel ID and parameters chanId, params, err := parseChannelsParameters( marshalledChanId, cmixParamsJSON) if err != nil { return nil, err } // Send message chanMsgId, rnd, ephId, err := cm.api.SendMessage(chanId, message, time.Duration(leaseTimeMS), params.CMIX) if err != nil { return nil, err } // Construct send report return constructChannelSendReport(chanMsgId, rnd.ID, ephId) } // SendReply is used to send a formatted message over a channel. Due to the // underlying encoding using compression, it isn't possible to define the // largest payload that can be sent, but it will always be possible to send a // payload of 766 bytes at minimum. // // If the message ID the reply is sent to is nonexistent, the other side will // post the message as a normal message and not a reply. The message will auto // delete validUntil after the round it is sent in, lasting forever if // [channels.ValidForever] is used. // // Parameters: // - marshalledChanId - A JSON marshalled channel ID ([id.ID]). // - message - The contents of the message. The message should be at most 510 // bytes. This is expected to be Unicode, and thus a string data type is // expected. // - messageToReactTo - The marshalled [channel.MessageID] of the message you // wish to reply to. This may be found in the ChannelSendReport if replying // to your own. Alternatively, if reacting to another user's message, you may // retrieve it via the ChannelMessageReceptionCallback registered using // RegisterReceiveHandler. // - leaseTimeMS - The lease of the message. This will be how long the message // is valid until, in milliseconds. As per the channels.Manager // documentation, this has different meanings depending on the use case. // These use cases may be generic enough that they will not be enumerated // here. // - cmixParamsJSON - A JSON marshalled [xxdk.CMIXParams]. This may be empty, // and GetDefaultCMixParams will be used internally. // // Returns: // - []byte - A JSON marshalled ChannelSendReport func (cm *ChannelsManager) SendReply(marshalledChanId []byte, message string, messageToReactTo []byte, leaseTimeMS int64, cmixParamsJSON []byte) ([]byte, error) { // Unmarshal channel ID and parameters chanId, params, err := parseChannelsParameters( marshalledChanId, cmixParamsJSON) if err != nil { return nil, err } // Unmarshal message ID msgId := cryptoChannel.MessageID{} copy(msgId[:], messageToReactTo) // Send Reply chanMsgId, rnd, ephId, err := cm.api.SendReply(chanId, message, msgId, time.Duration(leaseTimeMS), params.CMIX) if err != nil { return nil, err } // Construct send report return constructChannelSendReport(chanMsgId, rnd.ID, ephId) } // SendReaction is used to send a reaction to a message over a channel. // The reaction must be a single emoji with no other characters, and will // be rejected otherwise. // Users will drop the reaction if they do not recognize the reactTo message. // // Parameters: // - marshalledChanId - A JSON marshalled channel ID ([id.ID]). // - reaction - The user's reaction. This should be a single emoji with no // other characters. As such, a Unicode string is expected. // - messageToReactTo - The marshalled [channel.MessageID] of the message you // wish to reply to. This may be found in the ChannelSendReport if replying // to your own. Alternatively, if reacting to another user's message, you may // retrieve it via the ChannelMessageReceptionCallback registered using // RegisterReceiveHandler. // - cmixParamsJSON - A JSON marshalled [xxdk.CMIXParams]. This may be empty, // and GetDefaultCMixParams will be used internally. // // Returns: // - []byte - A JSON marshalled ChannelSendReport. func (cm *ChannelsManager) SendReaction(marshalledChanId []byte, reaction string, messageToReactTo []byte, cmixParamsJSON []byte) ([]byte, error) { // Unmarshal channel ID and parameters chanId, params, err := parseChannelsParameters( marshalledChanId, cmixParamsJSON) if err != nil { return nil, err } // Unmarshal message ID msgId := cryptoChannel.MessageID{} copy(msgId[:], messageToReactTo) // Send reaction chanMsgId, rnd, ephId, err := cm.api.SendReaction(chanId, reaction, msgId, params.CMIX) if err != nil { return nil, err } // Construct send report return constructChannelSendReport(chanMsgId, rnd.ID, ephId) } // GetIdentity returns the marshaled public identity ([channel.Identity]) that // the channel is using. func (cm *ChannelsManager) GetIdentity() ([]byte, error) { i := cm.api.GetIdentity() return json.Marshal(&i) } // ExportPrivateIdentity encrypts and exports the private identity to a portable // string. func (cm *ChannelsManager) ExportPrivateIdentity(password string) ([]byte, error) { return cm.api.ExportPrivateIdentity(password) } // GetStorageTag returns the storage tag needed to reload the manager. func (cm *ChannelsManager) GetStorageTag() string { return cm.api.GetStorageTag() } // SetNickname sets the nickname for a given channel. The nickname must be valid // according to [IsNicknameValid]. func (cm *ChannelsManager) SetNickname(newNick string, ch []byte) error { chid, err := id.Unmarshal(ch) if err != nil { return err } return cm.api.SetNickname(newNick, chid) } // DeleteNickname deletes the nickname for a given channel. func (cm *ChannelsManager) DeleteNickname(ch []byte) error { chid, err := id.Unmarshal(ch) if err != nil { return err } return cm.api.DeleteNickname(chid) } // GetNickname returns the nickname set for a given channel. Returns an error if // there is no nickname set. func (cm *ChannelsManager) GetNickname(ch []byte) (string, error) { chid, err := id.Unmarshal(ch) if err != nil { return "", err } nick, exists := cm.api.GetNickname(chid) if !exists { return "", errors.New("no nickname found for the given channel") } return nick, nil } // IsNicknameValid checks if a nickname is valid. // // Rules: // 1. A nickname must not be longer than 24 characters. // 2. A nickname must not be shorter than 1 character. func IsNicknameValid(nick string) error { return channels.IsNicknameValid(nick) } // parseChannelsParameters is a helper function for the Send functions. It // parses the channel ID and the passed in parameters into their respective // objects. These objects are passed into the API via the internal send // functions. func parseChannelsParameters(marshalledChanId, cmixParamsJSON []byte) ( *id.ID, xxdk.CMIXParams, error) { // Unmarshal channel ID chanId, err := id.Unmarshal(marshalledChanId) if err != nil { return nil, xxdk.CMIXParams{}, err } // Unmarshal cmix params params, err := parseCMixParams(cmixParamsJSON) if err != nil { return nil, xxdk.CMIXParams{}, err } return chanId, params, nil } // constructChannelSendReport is a helper function which returns a JSON // marshalled ChannelSendReport. func constructChannelSendReport(channelMessageId cryptoChannel.MessageID, roundId id.Round, ephId ephemeral.Id) ([]byte, error) { // Construct send report chanSendReport := ChannelSendReport{ MessageId: channelMessageId.Bytes(), RoundsList: makeRoundsList(roundId), EphId: ephId.Int64(), } // Marshal send report return json.Marshal(chanSendReport) } //////////////////////////////////////////////////////////////////////////////// // Channel Receiving Logic and Callback Registration // //////////////////////////////////////////////////////////////////////////////// // ReceivedChannelMessageReport is a report structure returned via the // ChannelMessageReceptionCallback. This report gives the context for the // channel the message was sent to and the message itself. This is returned via // the callback as JSON marshalled bytes. // // JSON Example: // { // "ChannelId": "AQAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA", // "MessageId": "3S6DiVjWH9mLmjy1oaam/3x45bJQzOW6u2KgeUn59wA=", // "ReplyTo":"cxMyGUFJ+Ff1Xp2X+XkIpOnNAQEZmv8SNP5eYH4tCik=", // "MessageType": 42, // "SenderUsername": "hunter2", // "Content": "YmFuX2JhZFVTZXI=", // "Timestamp": 1662502150335283000, // "Lease": 25, // "Rounds": [ 1, 4, 9], // } type ReceivedChannelMessageReport struct { ChannelId []byte MessageId []byte MessageType int Nickname string PubKey []byte Codeset int Content []byte Timestamp int64 Lease int64 RoundsList } // ChannelMessageReceptionCallback is the callback that returns the context for // a channel message via the Callback. // It must return a unique UUID for the message by which it can be referenced // later type ChannelMessageReceptionCallback interface { Callback(receivedChannelMessageReport []byte, err error) int } // RegisterReceiveHandler is used to register handlers for non-default message // types. They can be processed by modules. It is important that such modules // sync up with the event model implementation. // // There can only be one handler per [channels.MessageType], and this will // return an error on any re-registration. // // Parameters: // - messageType - represents the [channels.MessageType] which will have a // registered listener. // - listenerCb - the callback which will be executed when a channel message // of messageType is received. func (cm *ChannelsManager) RegisterReceiveHandler(messageType int, listenerCb ChannelMessageReceptionCallback) error { // Wrap callback around backend interface cb := channels.MessageTypeReceiveMessage( func(channelID *id.ID, messageID cryptoChannel.MessageID, messageType channels.MessageType, nickname string, content []byte, pubKey ed25519.PublicKey, codeset uint8, timestamp time.Time, lease time.Duration, round rounds.Round, status channels.SentStatus) uint64 { rcm := ReceivedChannelMessageReport{ ChannelId: channelID.Marshal(), MessageId: messageID.Bytes(), MessageType: int(messageType), Nickname: nickname, PubKey: pubKey, Codeset: int(codeset), Content: content, Timestamp: timestamp.UnixNano(), Lease: int64(lease), RoundsList: makeRoundsList(round.ID), } return uint64(listenerCb.Callback(json.Marshal(rcm))) }) // Register handler return cm.api.RegisterReceiveHandler(channels.MessageType(messageType), cb) } //////////////////////////////////////////////////////////////////////////////// // Event Model Logic // //////////////////////////////////////////////////////////////////////////////// // EventModelBuilder builds an event model type EventModelBuilder interface { Build(path string) EventModel } // EventModel is an interface which an external party which uses the channels // system passed an object which adheres to in order to get events on the // channel. type EventModel interface { // JoinChannel is called whenever a channel is joined locally. // // Parameters: // - channel - Returns the pretty print representation of a channel. JoinChannel(channel string) // LeaveChannel is called whenever a channel is left locally. // // Parameters: // - ChannelId - The marshalled channel [id.ID]. LeaveChannel(channelID []byte) // ReceiveMessage is called whenever a message is received on a given // channel. It may be called multiple times on the same message. It is // incumbent on the user of the API to filter such called by message ID. // // Parameters: // - channelID - The marshalled channel [id.ID]. // - messageID - The bytes of the [channel.MessageID] of the received // message. // - nickname - The nickname of the sender of the message. // - text - The content of the message. // - timestamp - Time the message was received; represented as nanoseconds // since unix epoch. // - pubKey - The sender's Ed25519 public key. // - codeset - The codeset version. // - lease - The number of nanoseconds that the message is valid for. // - roundId - The ID of the round that the message was received on. // - mType - the type of the message, always 1 for this call // - status - the [channels.SentStatus] of the message. // // Statuses will be enumerated as such: // Sent = 0 // Delivered = 1 // Failed = 2 // // Returns a non-negative unique UUID for the message that it can be // referenced by later with [EventModel.UpdateSentStatus]. ReceiveMessage(channelID, messageID []byte, nickname, text string, pubKey []byte, codeset int, timestamp, lease, roundId, mType, status int64) int64 // ReceiveReply is called whenever a message is received that is a reply on // a given channel. It may be called multiple times on the same message. It // is incumbent on the user of the API to filter such called by message ID. // // Messages may arrive our of order, so a reply in theory can arrive before // the initial message. As a result, it may be important to buffer replies. // // Parameters: // - channelID - The marshalled channel [id.ID]. // - messageID - The bytes of the [channel.MessageID] of the received // message. // - reactionTo - The [channel.MessageID] for the message that received a // reply. // - nickname - The nickname of the sender of the message. // - text - The content of the message. // - pubKey - The sender's Ed25519 public key. // - codeset - The codeset version. // - timestamp - Time the message was received; represented as nanoseconds // since unix epoch. // - lease - The number of nanoseconds that the message is valid for. // - roundId - The ID of the round that the message was received on. // - mType - the type of the message, always 1 for this call // - status - the [channels.SentStatus] of the message. // // Statuses will be enumerated as such: // Sent = 0 // Delivered = 1 // Failed = 2 // // Returns a non-negative unique UUID for the message that it can be // referenced by later with [EventModel.UpdateSentStatus]. ReceiveReply(channelID, messageID, reactionTo []byte, nickname, text string, pubKey []byte, codeset int, timestamp, lease, roundId, mType, status int64) int64 // ReceiveReaction is called whenever a reaction to a message is received // on a given channel. It may be called multiple times on the same reaction. // It is incumbent on the user of the API to filter such called by message // ID. // // Messages may arrive our of order, so a reply in theory can arrive before // the initial message. As a result, it may be important to buffer // reactions. // // Parameters: // - channelID - The marshalled channel [id.ID]. // - messageID - The bytes of the [channel.MessageID] of the received // message. // - reactionTo - The [channel.MessageID] for the message that received a // reply. // - nickname - The nickname of the sender of the message. // - reaction - The contents of the reaction message. // - pubKey - The sender's Ed25519 public key. // - codeset - The codeset version. // - timestamp - Time the message was received; represented as nanoseconds // since unix epoch. // - lease - The number of nanoseconds that the message is valid for. // - roundId - The ID of the round that the message was received on. // - mType - the type of the message, always 1 for this call // - status - the [channels.SentStatus] of the message. // // Statuses will be enumerated as such: // Sent = 0 // Delivered = 1 // Failed = 2 // // Returns a non-negative unique uuid for the message by which it can be // referenced later with UpdateSentStatus ReceiveReaction(channelID, messageID, reactionTo []byte, nickname, reaction string, pubKey []byte, codeset int, timestamp, lease, roundId, mType, status int64) int64 // UpdateSentStatus is called whenever the sent status of a message has // changed. // // Parameters: // - messageID - The bytes of the [channel.MessageID] of the received // message. // - status - the [channels.SentStatus] of the message. // // Statuses will be enumerated as such: // Sent = 0 // Delivered = 1 // Failed = 2 UpdateSentStatus( uuid int64, messageID []byte, timestamp, roundID, status int64) // unimplemented // IgnoreMessage(ChannelID *id.ID, MessageID cryptoChannel.MessageID) // UnIgnoreMessage(ChannelID *id.ID, MessageID cryptoChannel.MessageID) // PinMessage(ChannelID *id.ID, MessageID cryptoChannel.MessageID, end time.Time) // UnPinMessage(ChannelID *id.ID, MessageID cryptoChannel.MessageID) } // toEventModel is a wrapper which wraps an existing channels.EventModel object. type toEventModel struct { em EventModel } // NewEventModel is a constructor for a toEventModel. This will take in an // EventModel and wraps it around the toEventModel. func NewEventModel(em EventModel) channels.EventModel { return &toEventModel{em: em} } // JoinChannel is called whenever a channel is joined locally. func (tem *toEventModel) JoinChannel(channel *cryptoBroadcast.Channel) { tem.em.JoinChannel(channel.PrettyPrint()) } // LeaveChannel is called whenever a channel is left locally. func (tem *toEventModel) LeaveChannel(channelID *id.ID) { tem.em.LeaveChannel(channelID[:]) } // ReceiveMessage is called whenever a message is received on a given channel. // It may be called multiple times on the same message. It is incumbent on the // user of the API to filter such called by message ID. func (tem *toEventModel) ReceiveMessage(channelID *id.ID, messageID cryptoChannel.MessageID, nickname, text string, pubKey ed25519.PublicKey, codeset uint8, timestamp time.Time, lease time.Duration, round rounds.Round, mType channels.MessageType, status channels.SentStatus) uint64 { return uint64(tem.em.ReceiveMessage(channelID[:], messageID[:], nickname, text, pubKey, int(codeset), timestamp.UnixNano(), int64(lease), int64(round.ID), int64(mType), int64(status))) } // ReceiveReply is called whenever a message is received that is a reply on a // given channel. It may be called multiple times on the same message. It is // incumbent on the user of the API to filter such called by message ID. // // Messages may arrive our of order, so a reply in theory can arrive before the // initial message. As a result, it may be important to buffer replies. func (tem *toEventModel) ReceiveReply(channelID *id.ID, messageID cryptoChannel.MessageID, reactionTo cryptoChannel.MessageID, nickname, text string, pubKey ed25519.PublicKey, codeset uint8, timestamp time.Time, lease time.Duration, round rounds.Round, mType channels.MessageType, status channels.SentStatus) uint64 { return uint64(tem.em.ReceiveReply(channelID[:], messageID[:], reactionTo[:], nickname, text, pubKey, int(codeset), timestamp.UnixNano(), int64(lease), int64(round.ID), int64(mType), int64(status))) } // ReceiveReaction is called whenever a reaction to a message is received on a // given channel. It may be called multiple times on the same reaction. It is // incumbent on the user of the API to filter such called by message ID. // // Messages may arrive our of order, so a reply in theory can arrive before the // initial message. As a result, it may be important to buffer reactions. func (tem *toEventModel) ReceiveReaction(channelID *id.ID, messageID cryptoChannel.MessageID, reactionTo cryptoChannel.MessageID, nickname, reaction string, pubKey ed25519.PublicKey, codeset uint8, timestamp time.Time, lease time.Duration, round rounds.Round, mType channels.MessageType, status channels.SentStatus) uint64 { return uint64(tem.em.ReceiveReaction(channelID[:], messageID[:], reactionTo[:], nickname, reaction, pubKey, int(codeset), timestamp.UnixNano(), int64(lease), int64(round.ID), int64(mType), int64(status))) } // UpdateSentStatus is called whenever the sent status of a message has changed. func (tem *toEventModel) UpdateSentStatus(uuid uint64, messageID cryptoChannel.MessageID, timestamp time.Time, round rounds.Round, status channels.SentStatus) { tem.em.UpdateSentStatus(int64(uuid), messageID[:], timestamp.UnixNano(), int64(round.ID), int64(status)) } //////////////////////////////////////////////////////////////////////////////// // Channel ChannelDbCipher // //////////////////////////////////////////////////////////////////////////////// // ChannelDbCipher is the bindings layer representation of the [channel.Cipher]. type ChannelDbCipher struct { api cryptoChannel.Cipher salt []byte id int } // channelDbCipherTrackerSingleton is used to track ChannelDbCipher objects // so that they can be referenced by ID back over the bindings. var channelDbCipherTrackerSingleton = &channelDbCipherTracker{ tracked: make(map[int]*ChannelDbCipher), count: 0, } // channelDbCipherTracker is a singleton used to keep track of extant // ChannelDbCipher objects, preventing race conditions created by passing it // over the bindings. type channelDbCipherTracker struct { tracked map[int]*ChannelDbCipher count int mux sync.RWMutex } // create creates a ChannelDbCipher from a [channel.Cipher], assigns it a unique // ID, and adds it to the channelDbCipherTracker. func (ct *channelDbCipherTracker) create(c cryptoChannel.Cipher) *ChannelDbCipher { ct.mux.Lock() defer ct.mux.Unlock() chID := ct.count ct.count++ ct.tracked[chID] = &ChannelDbCipher{ api: c, id: chID, } return ct.tracked[chID] } // get an ChannelDbCipher from the channelDbCipherTracker given its ID. func (ct *channelDbCipherTracker) get(id int) (*ChannelDbCipher, error) { ct.mux.RLock() defer ct.mux.RUnlock() c, exist := ct.tracked[id] if !exist { return nil, errors.Errorf( "Cannot get ChannelDbCipher for ID %d, does not exist", id) } return c, nil } // delete removes a ChannelDbCipher from the channelDbCipherTracker. func (ct *channelDbCipherTracker) delete(id int) { ct.mux.Lock() defer ct.mux.Unlock() delete(ct.tracked, id) } // GetChannelDbCipherTrackerFromID returns the ChannelDbCipher with the // corresponding ID in the tracker. func GetChannelDbCipherTrackerFromID(id int) (*ChannelDbCipher, error) { return channelDbCipherTrackerSingleton.get(id) } // NewChannelsDatabaseCipher constructs a ChannelDbCipher object. // // Parameters: // - cmixID - The tracked [Cmix] object ID. // - password - The password for storage. This should be the same password // passed into [NewCmix]. // - plaintTextBlockSize - The maximum size of a payload to be encrypted. // A payload passed into [ChannelDbCipher.Encrypt] that is larger than // plaintTextBlockSize will result in an error. func NewChannelsDatabaseCipher(cmixID int, password []byte, plaintTextBlockSize int) (*ChannelDbCipher, error) { // Get user from singleton user, err := cmixTrackerSingleton.get(cmixID) if err != nil { return nil, err } // Generate RNG stream := user.api.GetRng().GetStream() // Load or generate a salt salt, err := utility.NewOrLoadSalt( user.api.GetStorage().GetKV(), stream) if err != nil { return nil, err } // Construct a cipher c, err := cryptoChannel.NewCipher(password, salt, plaintTextBlockSize, stream) if err != nil { return nil, err } // Return a cipher return channelDbCipherTrackerSingleton.create(c), nil } // GetID returns the ID for this ChannelDbCipher in the channelDbCipherTracker. func (c *ChannelDbCipher) GetID() int { return c.id } // Encrypt will encrypt the raw data. It will return a ciphertext. Padding is // done on the plaintext so all encrypted data looks uniform at rest. // // Parameters: // - plaintext - The data to be encrypted. This must be smaller than the block // size passed into [NewChannelsDatabaseCipher]. If it is larger, this will // return an error. func (c *ChannelDbCipher) Encrypt(plaintext []byte) ([]byte, error) { return c.api.Encrypt(plaintext) } // Decrypt will decrypt the passed in encrypted value. The plaintext will // be returned by this function. Any padding will be discarded within // this function. // // Parameters: // - ciphertext - the encrypted data returned by [ChannelDbCipher.Encrypt]. func (c *ChannelDbCipher) Decrypt(ciphertext []byte) ([]byte, error) { return c.api.Decrypt(ciphertext) }