2020-11-18 10:16:51 +01:00
package communities
import (
2021-01-11 11:32:51 +01:00
"context"
2020-11-18 10:16:51 +01:00
"crypto/ecdsa"
"database/sql"
2021-01-11 11:32:51 +01:00
"errors"
2022-03-21 15:18:36 +01:00
"fmt"
2023-06-21 13:20:43 +02:00
"math/big"
2023-06-06 20:33:09 +02:00
"strconv"
"strings"
2023-06-17 11:19:05 +03:00
"time"
2020-11-18 10:16:51 +01:00
"github.com/golang/protobuf/proto"
"go.uber.org/zap"
"github.com/status-im/status-go/eth-node/crypto"
2022-03-08 16:25:00 +01:00
"github.com/status-im/status-go/eth-node/types"
2021-01-11 11:32:51 +01:00
"github.com/status-im/status-go/protocol/common"
2023-07-07 15:03:37 +02:00
"github.com/status-im/status-go/protocol/communities/token"
2020-11-18 10:16:51 +01:00
"github.com/status-im/status-go/protocol/protobuf"
2023-06-21 13:20:43 +02:00
"github.com/status-im/status-go/services/wallet/bigint"
2020-11-18 10:16:51 +01:00
)
type Persistence struct {
2023-09-28 17:37:03 +02:00
db * sql . DB
logger * zap . Logger
timesource common . TimeSource
2020-11-18 10:16:51 +01:00
}
2021-11-11 16:37:04 +00:00
var ErrOldRequestToJoin = errors . New ( "old request to join" )
2022-08-22 12:10:31 +02:00
var ErrOldRequestToLeave = errors . New ( "old request to leave" )
2021-11-11 16:37:04 +00:00
2022-03-21 15:18:36 +01:00
const OR = " OR "
2023-07-18 17:06:12 +02:00
const communitiesBaseQuery = `
2023-10-12 15:21:49 -04:00
SELECT c . id , c . private_key , c . description , c . joined , c . spectated , c . verified , c . muted , c . muted_till , r . clock , ae . raw_events , ae . raw_description , c . shard_cluster , c . shard_index
2023-07-18 17:06:12 +02:00
FROM communities_communities c
LEFT JOIN communities_requests_to_join r ON c . id = r . community_id AND r . public_key = ?
LEFT JOIN communities_events ae ON c . id = ae . id `
2021-01-11 11:32:51 +01:00
2020-11-18 10:16:51 +01:00
func ( p * Persistence ) SaveCommunity ( community * Community ) error {
id := community . ID ( )
privateKey := community . PrivateKey ( )
2023-07-10 17:35:15 +02:00
wrappedCommunity , err := community . ToProtocolMessageBytes ( )
2020-11-18 10:16:51 +01:00
if err != nil {
return err
}
2023-10-12 15:21:49 -04:00
var shardIndex , shardCluster * uint
if community . Shard ( ) != nil {
index := uint ( community . Shard ( ) . Index )
shardIndex = & index
cluster := uint ( community . Shard ( ) . Cluster )
shardCluster = & cluster
}
2023-07-18 17:06:12 +02:00
_ , err = p . db . Exec ( `
2023-10-12 15:21:49 -04:00
INSERT INTO communities_communities ( id , private_key , description , joined , spectated , verified , muted , muted_till , shard_cluster , shard_index ) VALUES ( ? , ? , ? , ? , ? , ? , ? , ? , ? , ? ) ` ,
id , crypto . FromECDSA ( privateKey ) , wrappedCommunity , community . config . Joined , community . config . Spectated , community . config . Verified , community . config . Muted , community . config . MuteTill , shardCluster , shardIndex )
2023-07-18 17:06:12 +02:00
return err
}
func ( p * Persistence ) DeleteCommunityEvents ( id types . HexBytes ) error {
_ , err := p . db . Exec ( ` DELETE FROM communities_events WHERE id = ?; ` , id )
return err
}
func ( p * Persistence ) SaveCommunityEvents ( community * Community ) error {
id := community . ID ( )
if community . config . EventsData == nil {
return nil
}
rawEvents , err := communityEventsToJSONEncodedBytes ( community . config . EventsData . Events )
if err != nil {
return err
}
_ , err = p . db . Exec ( `
INSERT INTO communities_events ( id , raw_events , raw_description ) VALUES ( ? , ? , ? ) ; ` ,
id , rawEvents , community . config . EventsData . EventsBaseCommunityDescription )
2020-11-18 10:16:51 +01:00
return err
}
2022-09-29 13:50:23 +02:00
func ( p * Persistence ) DeleteCommunity ( id types . HexBytes ) error {
2023-07-18 17:06:12 +02:00
_ , err := p . db . Exec ( ` DELETE FROM communities_communities WHERE id = ? ;
DELETE FROM communities_events WHERE id = ? ; ` , id , id )
2022-09-29 13:50:23 +02:00
return err
}
2022-06-01 09:55:48 +02:00
func ( p * Persistence ) ShouldHandleSyncCommunitySettings ( settings * protobuf . SyncCommunitySettings ) ( bool , error ) {
qr := p . db . QueryRow ( ` SELECT * FROM communities_settings WHERE community_id = ? AND clock > ? ` , settings . CommunityId , settings . Clock )
_ , err := p . scanRowToStruct ( qr . Scan )
switch err {
case sql . ErrNoRows :
// Query does not match, therefore clock value is not older than the new clock value or id was not found
return true , nil
case nil :
// Error is nil, therefore query matched and clock is older than the new clock
return false , nil
default :
// Error is not nil and is not sql.ErrNoRows, therefore pass out the error
return false , err
}
}
2023-08-18 12:39:59 +01:00
func ( p * Persistence ) ShouldHandleSyncCommunity ( community * protobuf . SyncInstallationCommunity ) ( bool , error ) {
2021-08-06 16:40:23 +01:00
// TODO see if there is a way to make this more elegant
// When the test for this function fails because the table has changed we should update sync functionality
2023-10-12 15:21:49 -04:00
qr := p . db . QueryRow ( ` SELECT id, private_key, description, joined, verified, spectated, muted, muted_till, synced_at FROM communities_communities WHERE id = ? AND synced_at > ? ` , community . Id , community . Clock )
2021-08-06 16:40:23 +01:00
_ , err := p . scanRowToStruct ( qr . Scan )
switch err {
case sql . ErrNoRows :
// Query does not match, therefore synced_at value is not older than the new clock value or id was not found
return true , nil
case nil :
// Error is nil, therefore query matched and synced_at is older than the new clock
return false , nil
default :
// Error is not nil and is not sql.ErrNoRows, therefore pass out the error
return false , err
}
}
2021-01-11 11:32:51 +01:00
func ( p * Persistence ) queryCommunities ( memberIdentity * ecdsa . PublicKey , query string ) ( response [ ] * Community , err error ) {
2020-11-18 10:16:51 +01:00
2021-01-11 11:32:51 +01:00
rows , err := p . db . Query ( query , common . PubkeyToHex ( memberIdentity ) )
2020-11-18 10:16:51 +01:00
if err != nil {
return nil , err
}
2020-12-22 12:00:13 +01:00
defer func ( ) {
if err != nil {
// Don't shadow original error
_ = rows . Close ( )
return
}
err = rows . Close ( )
} ( )
2020-11-18 10:16:51 +01:00
for rows . Next ( ) {
var publicKeyBytes , privateKeyBytes , descriptionBytes [ ] byte
2022-09-20 21:57:39 +02:00
var joined , spectated , verified , muted bool
2023-06-17 11:19:05 +03:00
var muteTill sql . NullTime
2023-10-12 15:21:49 -04:00
var cluster , index , requestedToJoinAt sql . NullInt64
2023-07-18 17:06:12 +02:00
// Community events specific fields
var eventsBytes , eventsDescriptionBytes [ ] byte
2023-10-12 15:21:49 -04:00
err := rows . Scan ( & publicKeyBytes , & privateKeyBytes , & descriptionBytes , & joined , & spectated , & verified , & muted , & muteTill , & requestedToJoinAt , & eventsBytes , & eventsDescriptionBytes , & cluster , & index )
2020-11-18 10:16:51 +01:00
if err != nil {
return nil , err
}
2023-10-12 15:21:49 -04:00
var clusterValue * uint
if cluster . Valid {
v := uint ( cluster . Int64 )
clusterValue = & v
}
var indexValue * uint
if index . Valid {
v := uint ( index . Int64 )
indexValue = & v
}
org , err := p . unmarshalCommunityFromDB ( memberIdentity , publicKeyBytes , privateKeyBytes , descriptionBytes , joined , spectated , verified , muted , muteTill . Time , uint64 ( requestedToJoinAt . Int64 ) , eventsBytes , eventsDescriptionBytes , clusterValue , indexValue , p . logger )
2020-11-18 10:16:51 +01:00
if err != nil {
return nil , err
}
response = append ( response , org )
}
return response , nil
}
2021-01-11 11:32:51 +01:00
func ( p * Persistence ) AllCommunities ( memberIdentity * ecdsa . PublicKey ) ( [ ] * Community , error ) {
return p . queryCommunities ( memberIdentity , communitiesBaseQuery )
2020-11-18 10:16:51 +01:00
}
2021-01-11 11:32:51 +01:00
func ( p * Persistence ) JoinedCommunities ( memberIdentity * ecdsa . PublicKey ) ( [ ] * Community , error ) {
query := communitiesBaseQuery + ` WHERE c.joined `
return p . queryCommunities ( memberIdentity , query )
2020-11-18 10:16:51 +01:00
}
2022-09-20 21:57:39 +02:00
func ( p * Persistence ) SpectatedCommunities ( memberIdentity * ecdsa . PublicKey ) ( [ ] * Community , error ) {
query := communitiesBaseQuery + ` WHERE c.spectated `
return p . queryCommunities ( memberIdentity , query )
}
2022-04-11 18:14:08 +02:00
func ( p * Persistence ) rowsToCommunities ( memberIdentity * ecdsa . PublicKey , rows * sql . Rows ) ( comms [ ] * Community , err error ) {
2021-08-06 16:40:23 +01:00
defer func ( ) {
if err != nil {
// Don't shadow original error
_ = rows . Close ( )
return
}
err = rows . Close ( )
} ( )
for rows . Next ( ) {
var comm * Community
// Community specific fields
var publicKeyBytes , privateKeyBytes , descriptionBytes [ ] byte
2022-09-20 21:57:39 +02:00
var joined , spectated , verified , muted bool
2023-06-28 11:53:46 +01:00
var muteTill sql . NullTime
2023-10-12 15:21:49 -04:00
var cluster , index sql . NullInt64
2021-08-06 16:40:23 +01:00
// Request to join specific fields
var rtjID , rtjCommunityID [ ] byte
var rtjPublicKey , rtjENSName , rtjChatID sql . NullString
var rtjClock , rtjState sql . NullInt64
2023-07-18 17:06:12 +02:00
// Community events specific fields
var eventsBytes , eventsDescriptionBytes [ ] byte
2021-08-06 16:40:23 +01:00
err = rows . Scan (
2023-06-28 11:53:46 +01:00
& publicKeyBytes , & privateKeyBytes , & descriptionBytes , & joined , & spectated , & verified , & muted , & muteTill ,
2023-10-12 15:21:49 -04:00
& rtjID , & rtjPublicKey , & rtjClock , & rtjENSName , & rtjChatID , & rtjCommunityID , & rtjState , & eventsBytes ,
& eventsDescriptionBytes , & cluster , & index )
2021-08-06 16:40:23 +01:00
if err != nil {
return nil , err
}
2023-10-12 15:21:49 -04:00
var clusterValue * uint
if cluster . Valid {
v := uint ( cluster . Int64 )
clusterValue = & v
}
var indexValue * uint
if index . Valid {
v := uint ( index . Int64 )
indexValue = & v
}
comm , err = p . unmarshalCommunityFromDB ( memberIdentity , publicKeyBytes , privateKeyBytes , descriptionBytes , joined , spectated , verified , muted , muteTill . Time , uint64 ( rtjClock . Int64 ) , eventsBytes , eventsDescriptionBytes , clusterValue , indexValue , p . logger )
2021-08-06 16:40:23 +01:00
if err != nil {
return nil , err
}
rtj := unmarshalRequestToJoinFromDB ( rtjID , rtjCommunityID , rtjPublicKey , rtjENSName , rtjChatID , rtjClock , rtjState )
if ! rtj . Empty ( ) {
comm . AddRequestToJoin ( rtj )
}
comms = append ( comms , comm )
}
return comms , nil
}
2022-04-11 18:14:08 +02:00
func ( p * Persistence ) JoinedAndPendingCommunitiesWithRequests ( memberIdentity * ecdsa . PublicKey ) ( comms [ ] * Community , err error ) {
query := ` SELECT
2023-06-28 11:53:46 +01:00
c . id , c . private_key , c . description , c . joined , c . spectated , c . verified , c . muted , c . muted_till ,
2023-10-12 15:21:49 -04:00
r . id , r . public_key , r . clock , r . ens_name , r . chat_id , r . community_id , r . state , ae . raw_events , ae . raw_description ,
c . shard_cluster , c . shard_index
2022-04-11 18:14:08 +02:00
FROM communities_communities c
LEFT JOIN communities_requests_to_join r ON c . id = r . community_id AND r . public_key = ?
2023-07-18 17:06:12 +02:00
LEFT JOIN communities_events ae ON c . id = ae . id
2022-04-11 18:14:08 +02:00
WHERE c . Joined OR r . state = ? `
rows , err := p . db . Query ( query , common . PubkeyToHex ( memberIdentity ) , RequestToJoinStatePending )
if err != nil {
return nil , err
}
return p . rowsToCommunities ( memberIdentity , rows )
}
func ( p * Persistence ) DeletedCommunities ( memberIdentity * ecdsa . PublicKey ) ( comms [ ] * Community , err error ) {
query := ` SELECT
2023-06-28 11:53:46 +01:00
c . id , c . private_key , c . description , c . joined , c . spectated , c . verified , c . muted , c . muted_till ,
2023-10-12 15:21:49 -04:00
r . id , r . public_key , r . clock , r . ens_name , r . chat_id , r . community_id , r . state , ae . raw_events , ae . raw_description ,
c . shard_cluster , c . shard_index
2022-04-11 18:14:08 +02:00
FROM communities_communities c
LEFT JOIN communities_requests_to_join r ON c . id = r . community_id AND r . public_key = ?
2023-07-18 17:06:12 +02:00
LEFT JOIN communities_events ae ON c . id = ae . id
2022-04-11 18:14:08 +02:00
WHERE NOT c . Joined AND ( r . community_id IS NULL or r . state != ? ) `
rows , err := p . db . Query ( query , common . PubkeyToHex ( memberIdentity ) , RequestToJoinStatePending )
if err != nil {
return nil , err
}
return p . rowsToCommunities ( memberIdentity , rows )
}
2023-07-06 19:44:31 +02:00
func ( p * Persistence ) CommunitiesWithPrivateKey ( memberIdentity * ecdsa . PublicKey ) ( [ ] * Community , error ) {
2021-01-11 11:32:51 +01:00
query := communitiesBaseQuery + ` WHERE c.private_key IS NOT NULL `
return p . queryCommunities ( memberIdentity , query )
2020-11-18 10:16:51 +01:00
}
2021-01-11 11:32:51 +01:00
func ( p * Persistence ) GetByID ( memberIdentity * ecdsa . PublicKey , id [ ] byte ) ( * Community , error ) {
2020-11-18 10:16:51 +01:00
var publicKeyBytes , privateKeyBytes , descriptionBytes [ ] byte
var joined bool
2022-09-20 21:57:39 +02:00
var spectated bool
2020-11-18 10:16:51 +01:00
var verified bool
2021-06-30 09:29:43 -04:00
var muted bool
2023-06-17 11:19:05 +03:00
var muteTill sql . NullTime
2023-10-12 15:21:49 -04:00
var requestedToJoinAt , cluster , index sql . NullInt64
2020-11-18 10:16:51 +01:00
2023-07-18 17:06:12 +02:00
// Community events specific fields
var eventsBytes , eventsDescriptionBytes [ ] byte
2023-10-12 15:21:49 -04:00
err := p . db . QueryRow ( communitiesBaseQuery + ` WHERE c.id = ? ` , common . PubkeyToHex ( memberIdentity ) , id ) . Scan ( & publicKeyBytes , & privateKeyBytes , & descriptionBytes , & joined , & spectated , & verified , & muted , & muteTill , & requestedToJoinAt , & eventsBytes , & eventsDescriptionBytes , & cluster , & index )
2020-11-18 10:16:51 +01:00
if err == sql . ErrNoRows {
return nil , nil
} else if err != nil {
return nil , err
}
2023-10-12 15:21:49 -04:00
var clusterValue * uint
if cluster . Valid {
v := uint ( cluster . Int64 )
clusterValue = & v
}
var indexValue * uint
if index . Valid {
v := uint ( index . Int64 )
indexValue = & v
}
return p . unmarshalCommunityFromDB ( memberIdentity , publicKeyBytes , privateKeyBytes , descriptionBytes , joined , spectated , verified , muted , muteTill . Time , uint64 ( requestedToJoinAt . Int64 ) , eventsBytes , eventsDescriptionBytes , clusterValue , indexValue , p . logger )
2020-11-18 10:16:51 +01:00
}
2023-09-28 17:37:03 +02:00
func ( p * Persistence ) unmarshalCommunityFromDB ( memberIdentity * ecdsa . PublicKey , publicKeyBytes , privateKeyBytes , wrappedCommunity [ ] byte , joined ,
2023-06-28 11:53:46 +01:00
spectated , verified , muted bool , muteTill time . Time , requestedToJoinAt uint64 , eventsBytes [ ] byte ,
2023-10-12 15:21:49 -04:00
eventsDescriptionBytes [ ] byte , cluster * uint , index * uint , logger * zap . Logger ) ( * Community , error ) {
2020-11-18 10:16:51 +01:00
var privateKey * ecdsa . PrivateKey
var err error
if privateKeyBytes != nil {
privateKey , err = crypto . ToECDSA ( privateKeyBytes )
if err != nil {
return nil , err
}
}
2023-07-10 17:35:15 +02:00
description , err := decodeWrappedCommunityDescription ( wrappedCommunity )
2020-11-18 10:16:51 +01:00
if err != nil {
return nil , err
}
2023-07-18 17:06:12 +02:00
id , err := crypto . DecompressPubkey ( publicKeyBytes )
2020-11-18 10:16:51 +01:00
if err != nil {
return nil , err
}
2023-07-18 17:06:12 +02:00
eventsData , err := decodeEventsData ( eventsBytes , eventsDescriptionBytes )
2020-11-18 10:16:51 +01:00
if err != nil {
return nil , err
}
2023-10-12 15:21:49 -04:00
var shard * common . Shard = nil
if cluster != nil && index != nil {
shard = & common . Shard {
Cluster : uint16 ( * cluster ) ,
Index : uint16 ( * index ) ,
}
}
2020-11-18 10:16:51 +01:00
config := Config {
2023-07-10 17:35:15 +02:00
PrivateKey : privateKey ,
CommunityDescription : description ,
MemberIdentity : memberIdentity ,
CommunityDescriptionProtocolMessage : wrappedCommunity ,
Logger : logger ,
ID : id ,
Verified : verified ,
Muted : muted ,
MuteTill : muteTill ,
RequestedToJoinAt : requestedToJoinAt ,
Joined : joined ,
Spectated : spectated ,
EventsData : eventsData ,
2023-10-12 15:21:49 -04:00
Shard : shard ,
2023-07-18 17:06:12 +02:00
}
2023-09-28 17:37:03 +02:00
community , err := New ( config , p . timesource )
2023-07-18 17:06:12 +02:00
if err != nil {
return nil , err
2020-11-18 10:16:51 +01:00
}
2023-07-18 17:06:12 +02:00
return community , nil
2020-11-18 10:16:51 +01:00
}
2021-01-11 11:32:51 +01:00
2021-08-06 16:40:23 +01:00
func unmarshalRequestToJoinFromDB ( ID , communityID [ ] byte , publicKey , ensName , chatID sql . NullString , clock , state sql . NullInt64 ) * RequestToJoin {
return & RequestToJoin {
ID : ID ,
PublicKey : publicKey . String ,
Clock : uint64 ( clock . Int64 ) ,
ENSName : ensName . String ,
ChatID : chatID . String ,
CommunityID : communityID ,
State : RequestToJoinState ( state . Int64 ) ,
}
}
2021-01-11 11:32:51 +01:00
func ( p * Persistence ) SaveRequestToJoin ( request * RequestToJoin ) ( err error ) {
tx , err := p . db . BeginTx ( context . Background ( ) , & sql . TxOptions { } )
if err != nil {
return err
}
defer func ( ) {
if err == nil {
err = tx . Commit ( )
return
}
// don't shadow original error
_ = tx . Rollback ( )
} ( )
var clock uint64
// Fetch any existing request to join
err = tx . QueryRow ( ` SELECT clock FROM communities_requests_to_join WHERE state = ? AND public_key = ? AND community_id = ? ` , RequestToJoinStatePending , request . PublicKey , request . CommunityID ) . Scan ( & clock )
if err != nil && err != sql . ErrNoRows {
return err
}
// This is already processed
if clock >= request . Clock {
2021-11-11 16:37:04 +00:00
return ErrOldRequestToJoin
2021-01-11 11:32:51 +01:00
}
2023-08-18 15:52:13 -04:00
_ , err = tx . Exec ( ` INSERT OR REPLACE INTO communities_requests_to_join(id,public_key,clock,ens_name,chat_id,community_id,state) VALUES (?, ?, ?, ?, ?, ?, ?) ` , request . ID , request . PublicKey , request . Clock , request . ENSName , request . ChatID , request . CommunityID , request . State )
2021-01-11 11:32:51 +01:00
return err
}
2023-08-29 14:56:30 -04:00
func ( p * Persistence ) SaveRequestToJoinRevealedAddresses ( requestID types . HexBytes , revealedAccounts [ ] * protobuf . RevealedAccount ) ( err error ) {
feat: add verified wallet accounts to community requests
This commit extends the `CommunityRequestToJoin` with `RevealedAddresses` which represent wallet addresses and signatures provided by the sender, to proof a community owner ownership of those wallet addresses.
**Note: This only works with keystore files maanged by status-go**
At high level, the follwing happens:
1. User instructs Status to send a request to join to a community. By adding a password hash to the instruction, Status will try to unlock the users keystore and verify each wallet account.
2. For every verified wallet account, a signature is created for the following payload, using each wallet's private key
``` keccak256(chatkey + communityID + requestToJoinID) ``` A map of walletAddress->signature is then attached to the community request to join, which will be sent to the community owner
3. The owner node receives the request, and if the community requires users to hold tokens to become a member, it will check and verify whether the given wallet addresses are indeed owned by the sender. If any signature provided by the request cannot be recovered, the request is immediately declined by the owner.
4. The verified addresses are then added to the owner node's database such that, once the request should be accepted, the addresses can be used to check on chain whether they own the necessary funds to fulfill the community's permissions
The checking of required funds is **not** part of this commit. It will be added in a follow-up commit.
2023-03-17 10:19:40 +01:00
tx , err := p . db . BeginTx ( context . Background ( ) , & sql . TxOptions { } )
if err != nil {
return
}
defer func ( ) {
if err == nil {
err = tx . Commit ( )
return
}
// don't shadow original error
_ = tx . Rollback ( )
} ( )
2023-07-14 13:33:47 -04:00
query := ` INSERT OR REPLACE INTO communities_requests_to_join_revealed_addresses (request_id, address, chain_ids, is_airdrop_address) VALUES (?, ?, ?, ?) `
feat: add verified wallet accounts to community requests
This commit extends the `CommunityRequestToJoin` with `RevealedAddresses` which represent wallet addresses and signatures provided by the sender, to proof a community owner ownership of those wallet addresses.
**Note: This only works with keystore files maanged by status-go**
At high level, the follwing happens:
1. User instructs Status to send a request to join to a community. By adding a password hash to the instruction, Status will try to unlock the users keystore and verify each wallet account.
2. For every verified wallet account, a signature is created for the following payload, using each wallet's private key
``` keccak256(chatkey + communityID + requestToJoinID) ``` A map of walletAddress->signature is then attached to the community request to join, which will be sent to the community owner
3. The owner node receives the request, and if the community requires users to hold tokens to become a member, it will check and verify whether the given wallet addresses are indeed owned by the sender. If any signature provided by the request cannot be recovered, the request is immediately declined by the owner.
4. The verified addresses are then added to the owner node's database such that, once the request should be accepted, the addresses can be used to check on chain whether they own the necessary funds to fulfill the community's permissions
The checking of required funds is **not** part of this commit. It will be added in a follow-up commit.
2023-03-17 10:19:40 +01:00
stmt , err := tx . Prepare ( query )
if err != nil {
return
}
defer stmt . Close ( )
2023-08-29 14:56:30 -04:00
for _ , account := range revealedAccounts {
2023-06-06 20:33:09 +02:00
var chainIDs [ ] string
for _ , ID := range account . ChainIds {
chainIDs = append ( chainIDs , strconv . Itoa ( int ( ID ) ) )
}
feat: add verified wallet accounts to community requests
This commit extends the `CommunityRequestToJoin` with `RevealedAddresses` which represent wallet addresses and signatures provided by the sender, to proof a community owner ownership of those wallet addresses.
**Note: This only works with keystore files maanged by status-go**
At high level, the follwing happens:
1. User instructs Status to send a request to join to a community. By adding a password hash to the instruction, Status will try to unlock the users keystore and verify each wallet account.
2. For every verified wallet account, a signature is created for the following payload, using each wallet's private key
``` keccak256(chatkey + communityID + requestToJoinID) ``` A map of walletAddress->signature is then attached to the community request to join, which will be sent to the community owner
3. The owner node receives the request, and if the community requires users to hold tokens to become a member, it will check and verify whether the given wallet addresses are indeed owned by the sender. If any signature provided by the request cannot be recovered, the request is immediately declined by the owner.
4. The verified addresses are then added to the owner node's database such that, once the request should be accepted, the addresses can be used to check on chain whether they own the necessary funds to fulfill the community's permissions
The checking of required funds is **not** part of this commit. It will be added in a follow-up commit.
2023-03-17 10:19:40 +01:00
_ , err = stmt . Exec (
2023-08-29 14:56:30 -04:00
requestID ,
2023-06-06 20:33:09 +02:00
account . Address ,
strings . Join ( chainIDs , "," ) ,
2023-07-14 13:33:47 -04:00
account . IsAirdropAddress ,
feat: add verified wallet accounts to community requests
This commit extends the `CommunityRequestToJoin` with `RevealedAddresses` which represent wallet addresses and signatures provided by the sender, to proof a community owner ownership of those wallet addresses.
**Note: This only works with keystore files maanged by status-go**
At high level, the follwing happens:
1. User instructs Status to send a request to join to a community. By adding a password hash to the instruction, Status will try to unlock the users keystore and verify each wallet account.
2. For every verified wallet account, a signature is created for the following payload, using each wallet's private key
``` keccak256(chatkey + communityID + requestToJoinID) ``` A map of walletAddress->signature is then attached to the community request to join, which will be sent to the community owner
3. The owner node receives the request, and if the community requires users to hold tokens to become a member, it will check and verify whether the given wallet addresses are indeed owned by the sender. If any signature provided by the request cannot be recovered, the request is immediately declined by the owner.
4. The verified addresses are then added to the owner node's database such that, once the request should be accepted, the addresses can be used to check on chain whether they own the necessary funds to fulfill the community's permissions
The checking of required funds is **not** part of this commit. It will be added in a follow-up commit.
2023-03-17 10:19:40 +01:00
)
if err != nil {
return
}
}
return
}
2023-06-22 08:54:58 +02:00
func ( p * Persistence ) SaveCheckChannelPermissionResponse ( communityID string , chatID string , response * CheckChannelPermissionsResponse ) error {
tx , err := p . db . BeginTx ( context . Background ( ) , & sql . TxOptions { } )
if err != nil {
return err
}
defer func ( ) {
if err == nil {
err = tx . Commit ( )
return
}
// don't shadow original error
_ = tx . Rollback ( )
} ( )
viewOnlyPermissionIDs := make ( [ ] string , 0 )
viewAndPostPermissionIDs := make ( [ ] string , 0 )
for permissionID := range response . ViewOnlyPermissions . Permissions {
viewOnlyPermissionIDs = append ( viewOnlyPermissionIDs , permissionID )
}
for permissionID := range response . ViewAndPostPermissions . Permissions {
viewAndPostPermissionIDs = append ( viewAndPostPermissionIDs , permissionID )
}
_ , err = tx . Exec ( ` INSERT INTO communities_check_channel_permission_responses (community_id,chat_id,view_only_permissions_satisfied,view_and_post_permissions_satisfied, view_only_permission_ids, view_and_post_permission_ids) VALUES (?, ?, ?, ?, ?, ?) ` , communityID , chatID , response . ViewOnlyPermissions . Satisfied , response . ViewAndPostPermissions . Satisfied , strings . Join ( viewOnlyPermissionIDs [ : ] , "," ) , strings . Join ( viewAndPostPermissionIDs [ : ] , "," ) )
if err != nil {
return err
}
saveCriteriaResults := func ( permissions map [ string ] * PermissionTokenCriteriaResult ) error {
for permissionID , criteriaResult := range permissions {
criteria := make ( [ ] string , 0 )
for _ , val := range criteriaResult . Criteria {
criteria = append ( criteria , strconv . FormatBool ( val ) )
}
_ , err = tx . Exec ( ` INSERT INTO communities_permission_token_criteria_results (permission_id,community_id, chat_id, criteria) VALUES (?, ?, ?, ?) ` , permissionID , communityID , chatID , strings . Join ( criteria [ : ] , "," ) )
if err != nil {
return err
}
}
return nil
}
err = saveCriteriaResults ( response . ViewOnlyPermissions . Permissions )
if err != nil {
return err
}
return saveCriteriaResults ( response . ViewAndPostPermissions . Permissions )
}
func ( p * Persistence ) GetCheckChannelPermissionResponses ( communityID string ) ( map [ string ] * CheckChannelPermissionsResponse , error ) {
rows , err := p . db . Query ( ` SELECT chat_id, view_only_permissions_satisfied, view_and_post_permissions_satisfied, view_only_permission_ids, view_and_post_permission_ids FROM communities_check_channel_permission_responses WHERE community_id = ? ` , communityID )
if err != nil {
return nil , err
}
defer rows . Close ( )
checkChannelPermissionResponses := make ( map [ string ] * CheckChannelPermissionsResponse , 0 )
for rows . Next ( ) {
permissionResponse := & CheckChannelPermissionsResponse {
ViewOnlyPermissions : & CheckChannelViewOnlyPermissionsResult {
Satisfied : false ,
Permissions : make ( map [ string ] * PermissionTokenCriteriaResult ) ,
} ,
ViewAndPostPermissions : & CheckChannelViewAndPostPermissionsResult {
Satisfied : false ,
Permissions : make ( map [ string ] * PermissionTokenCriteriaResult ) ,
} ,
}
var chatID string
var viewOnlyPermissionIDsString string
var viewAndPostPermissionIDsString string
err := rows . Scan ( & chatID , & permissionResponse . ViewOnlyPermissions . Satisfied , & permissionResponse . ViewAndPostPermissions . Satisfied , & viewOnlyPermissionIDsString , & viewAndPostPermissionIDsString )
if err != nil {
return nil , err
}
for _ , permissionID := range strings . Split ( viewOnlyPermissionIDsString , "," ) {
if permissionID != "" {
permissionResponse . ViewOnlyPermissions . Permissions [ permissionID ] = & PermissionTokenCriteriaResult { Criteria : make ( [ ] bool , 0 ) }
}
}
for _ , permissionID := range strings . Split ( viewAndPostPermissionIDsString , "," ) {
if permissionID != "" {
permissionResponse . ViewAndPostPermissions . Permissions [ permissionID ] = & PermissionTokenCriteriaResult { Criteria : make ( [ ] bool , 0 ) }
}
}
checkChannelPermissionResponses [ chatID ] = permissionResponse
}
addCriteriaResult := func ( channelResponses map [ string ] * CheckChannelPermissionsResponse , permissions map [ string ] * PermissionTokenCriteriaResult , chatID string , viewOnly bool ) error {
for permissionID := range permissions {
criteria , err := p . GetPermissionTokenCriteriaResult ( permissionID , communityID , chatID )
if err != nil {
return err
}
if viewOnly {
channelResponses [ chatID ] . ViewOnlyPermissions . Permissions [ permissionID ] = criteria
} else {
channelResponses [ chatID ] . ViewAndPostPermissions . Permissions [ permissionID ] = criteria
}
}
return nil
}
for chatID , response := range checkChannelPermissionResponses {
err := addCriteriaResult ( checkChannelPermissionResponses , response . ViewOnlyPermissions . Permissions , chatID , true )
if err != nil {
return nil , err
}
err = addCriteriaResult ( checkChannelPermissionResponses , response . ViewAndPostPermissions . Permissions , chatID , false )
if err != nil {
return nil , err
}
}
return checkChannelPermissionResponses , nil
}
func ( p * Persistence ) GetPermissionTokenCriteriaResult ( permissionID string , communityID string , chatID string ) ( * PermissionTokenCriteriaResult , error ) {
tx , err := p . db . BeginTx ( context . Background ( ) , & sql . TxOptions { } )
if err != nil {
return nil , err
}
defer func ( ) {
if err == nil {
err = tx . Commit ( )
return
}
// don't shadow original error
_ = tx . Rollback ( )
} ( )
criteriaString := ""
err = tx . QueryRow ( ` SELECT criteria FROM communities_permission_token_criteria_results WHERE permission_id = ? AND community_id = ? AND chat_id = ? ` , permissionID , communityID , chatID ) . Scan ( & criteriaString )
if err != nil {
return nil , err
}
criteria := make ( [ ] bool , 0 )
for _ , r := range strings . Split ( criteriaString , "," ) {
val , err := strconv . ParseBool ( r )
if err != nil {
return nil , err
}
criteria = append ( criteria , val )
}
return & PermissionTokenCriteriaResult { Criteria : criteria } , nil
}
2023-08-04 11:49:11 +02:00
func ( p * Persistence ) RemoveRequestToJoinRevealedAddresses ( requestID [ ] byte ) error {
_ , err := p . db . Exec ( ` DELETE FROM communities_requests_to_join_revealed_addresses WHERE request_id = ? ` , requestID )
return err
}
2023-06-06 20:33:09 +02:00
func ( p * Persistence ) GetRequestToJoinRevealedAddresses ( requestID [ ] byte ) ( [ ] * protobuf . RevealedAccount , error ) {
revealedAccounts := make ( [ ] * protobuf . RevealedAccount , 0 )
2023-07-14 13:33:47 -04:00
rows , err := p . db . Query ( ` SELECT address, chain_ids, is_airdrop_address FROM communities_requests_to_join_revealed_addresses WHERE request_id = ? ` , requestID )
feat: add verified wallet accounts to community requests
This commit extends the `CommunityRequestToJoin` with `RevealedAddresses` which represent wallet addresses and signatures provided by the sender, to proof a community owner ownership of those wallet addresses.
**Note: This only works with keystore files maanged by status-go**
At high level, the follwing happens:
1. User instructs Status to send a request to join to a community. By adding a password hash to the instruction, Status will try to unlock the users keystore and verify each wallet account.
2. For every verified wallet account, a signature is created for the following payload, using each wallet's private key
``` keccak256(chatkey + communityID + requestToJoinID) ``` A map of walletAddress->signature is then attached to the community request to join, which will be sent to the community owner
3. The owner node receives the request, and if the community requires users to hold tokens to become a member, it will check and verify whether the given wallet addresses are indeed owned by the sender. If any signature provided by the request cannot be recovered, the request is immediately declined by the owner.
4. The verified addresses are then added to the owner node's database such that, once the request should be accepted, the addresses can be used to check on chain whether they own the necessary funds to fulfill the community's permissions
The checking of required funds is **not** part of this commit. It will be added in a follow-up commit.
2023-03-17 10:19:40 +01:00
if err != nil {
return nil , err
}
defer rows . Close ( )
for rows . Next ( ) {
2023-09-20 10:37:46 +02:00
var address sql . NullString
var chainIDsStr sql . NullString
2023-07-19 15:23:00 -04:00
var isAirdropAddress sql . NullBool
err := rows . Scan ( & address , & chainIDsStr , & isAirdropAddress )
feat: add verified wallet accounts to community requests
This commit extends the `CommunityRequestToJoin` with `RevealedAddresses` which represent wallet addresses and signatures provided by the sender, to proof a community owner ownership of those wallet addresses.
**Note: This only works with keystore files maanged by status-go**
At high level, the follwing happens:
1. User instructs Status to send a request to join to a community. By adding a password hash to the instruction, Status will try to unlock the users keystore and verify each wallet account.
2. For every verified wallet account, a signature is created for the following payload, using each wallet's private key
``` keccak256(chatkey + communityID + requestToJoinID) ``` A map of walletAddress->signature is then attached to the community request to join, which will be sent to the community owner
3. The owner node receives the request, and if the community requires users to hold tokens to become a member, it will check and verify whether the given wallet addresses are indeed owned by the sender. If any signature provided by the request cannot be recovered, the request is immediately declined by the owner.
4. The verified addresses are then added to the owner node's database such that, once the request should be accepted, the addresses can be used to check on chain whether they own the necessary funds to fulfill the community's permissions
The checking of required funds is **not** part of this commit. It will be added in a follow-up commit.
2023-03-17 10:19:40 +01:00
if err != nil {
return nil , err
}
2023-06-06 20:33:09 +02:00
2023-09-20 10:37:46 +02:00
revealedAccount , err := toRevealedAccount ( address , chainIDsStr , isAirdropAddress )
if err != nil {
return nil , err
2023-06-06 20:33:09 +02:00
}
2023-09-20 10:37:46 +02:00
if revealedAccount == nil {
return nil , errors . New ( "invalid RequestToJoin RevealedAddresses data" )
2023-06-06 20:33:09 +02:00
}
revealedAccounts = append ( revealedAccounts , revealedAccount )
feat: add verified wallet accounts to community requests
This commit extends the `CommunityRequestToJoin` with `RevealedAddresses` which represent wallet addresses and signatures provided by the sender, to proof a community owner ownership of those wallet addresses.
**Note: This only works with keystore files maanged by status-go**
At high level, the follwing happens:
1. User instructs Status to send a request to join to a community. By adding a password hash to the instruction, Status will try to unlock the users keystore and verify each wallet account.
2. For every verified wallet account, a signature is created for the following payload, using each wallet's private key
``` keccak256(chatkey + communityID + requestToJoinID) ``` A map of walletAddress->signature is then attached to the community request to join, which will be sent to the community owner
3. The owner node receives the request, and if the community requires users to hold tokens to become a member, it will check and verify whether the given wallet addresses are indeed owned by the sender. If any signature provided by the request cannot be recovered, the request is immediately declined by the owner.
4. The verified addresses are then added to the owner node's database such that, once the request should be accepted, the addresses can be used to check on chain whether they own the necessary funds to fulfill the community's permissions
The checking of required funds is **not** part of this commit. It will be added in a follow-up commit.
2023-03-17 10:19:40 +01:00
}
2023-06-06 20:33:09 +02:00
return revealedAccounts , nil
feat: add verified wallet accounts to community requests
This commit extends the `CommunityRequestToJoin` with `RevealedAddresses` which represent wallet addresses and signatures provided by the sender, to proof a community owner ownership of those wallet addresses.
**Note: This only works with keystore files maanged by status-go**
At high level, the follwing happens:
1. User instructs Status to send a request to join to a community. By adding a password hash to the instruction, Status will try to unlock the users keystore and verify each wallet account.
2. For every verified wallet account, a signature is created for the following payload, using each wallet's private key
``` keccak256(chatkey + communityID + requestToJoinID) ``` A map of walletAddress->signature is then attached to the community request to join, which will be sent to the community owner
3. The owner node receives the request, and if the community requires users to hold tokens to become a member, it will check and verify whether the given wallet addresses are indeed owned by the sender. If any signature provided by the request cannot be recovered, the request is immediately declined by the owner.
4. The verified addresses are then added to the owner node's database such that, once the request should be accepted, the addresses can be used to check on chain whether they own the necessary funds to fulfill the community's permissions
The checking of required funds is **not** part of this commit. It will be added in a follow-up commit.
2023-03-17 10:19:40 +01:00
}
2022-08-22 12:10:31 +02:00
func ( p * Persistence ) SaveRequestToLeave ( request * RequestToLeave ) error {
tx , err := p . db . BeginTx ( context . Background ( ) , & sql . TxOptions { } )
if err != nil {
return err
}
defer func ( ) {
if err == nil {
err = tx . Commit ( )
return
}
// don't shadow original error
_ = tx . Rollback ( )
} ( )
var clock uint64
// Fetch any existing request to leave
err = tx . QueryRow ( ` SELECT clock FROM communities_requests_to_leave WHERE public_key = ? AND community_id = ? ` , request . PublicKey , request . CommunityID ) . Scan ( & clock )
if err != nil && err != sql . ErrNoRows {
return err
}
// This is already processed
if clock >= request . Clock {
return ErrOldRequestToLeave
}
_ , err = tx . Exec ( ` INSERT INTO communities_requests_to_leave(id,public_key,clock,community_id) VALUES (?, ?, ?, ?) ` , request . ID , request . PublicKey , request . Clock , request . CommunityID )
return err
}
2022-10-28 11:41:20 +03:00
func ( p * Persistence ) CanceledRequestsToJoinForUser ( pk string ) ( [ ] * RequestToJoin , error ) {
var requests [ ] * RequestToJoin
rows , err := p . db . Query ( ` SELECT id,public_key,clock,ens_name,chat_id,community_id,state FROM communities_requests_to_join WHERE state = ? AND public_key = ? ` , RequestToJoinStateCanceled , pk )
if err != nil {
return nil , err
}
defer rows . Close ( )
for rows . Next ( ) {
request := & RequestToJoin { }
err := rows . Scan ( & request . ID , & request . PublicKey , & request . Clock , & request . ENSName , & request . ChatID , & request . CommunityID , & request . State )
if err != nil {
return nil , err
}
requests = append ( requests , request )
}
return requests , nil
}
2021-01-11 11:32:51 +01:00
func ( p * Persistence ) PendingRequestsToJoinForUser ( pk string ) ( [ ] * RequestToJoin , error ) {
var requests [ ] * RequestToJoin
rows , err := p . db . Query ( ` SELECT id,public_key,clock,ens_name,chat_id,community_id,state FROM communities_requests_to_join WHERE state = ? AND public_key = ? ` , RequestToJoinStatePending , pk )
if err != nil {
return nil , err
}
defer rows . Close ( )
for rows . Next ( ) {
request := & RequestToJoin { }
err := rows . Scan ( & request . ID , & request . PublicKey , & request . Clock , & request . ENSName , & request . ChatID , & request . CommunityID , & request . State )
if err != nil {
return nil , err
}
requests = append ( requests , request )
}
return requests , nil
}
func ( p * Persistence ) HasPendingRequestsToJoinForUserAndCommunity ( userPk string , communityID [ ] byte ) ( bool , error ) {
var count int
err := p . db . QueryRow ( ` SELECT count(1) FROM communities_requests_to_join WHERE state = ? AND public_key = ? AND community_id = ? ` , RequestToJoinStatePending , userPk , communityID ) . Scan ( & count )
if err != nil {
return false , err
}
return count > 0 , nil
}
2022-08-04 09:44:35 +02:00
func ( p * Persistence ) RequestsToJoinForCommunityWithState ( id [ ] byte , state RequestToJoinState ) ( [ ] * RequestToJoin , error ) {
2021-01-11 11:32:51 +01:00
var requests [ ] * RequestToJoin
2022-08-04 09:44:35 +02:00
rows , err := p . db . Query ( ` SELECT id,public_key,clock,ens_name,chat_id,community_id,state FROM communities_requests_to_join WHERE state = ? AND community_id = ? ` , state , id )
2021-01-11 11:32:51 +01:00
if err != nil {
return nil , err
}
defer rows . Close ( )
for rows . Next ( ) {
request := & RequestToJoin { }
err := rows . Scan ( & request . ID , & request . PublicKey , & request . Clock , & request . ENSName , & request . ChatID , & request . CommunityID , & request . State )
if err != nil {
return nil , err
}
requests = append ( requests , request )
}
return requests , nil
}
2023-04-21 17:18:47 +08:00
func ( p * Persistence ) PendingRequestsToJoin ( ) ( [ ] * RequestToJoin , error ) {
var requests [ ] * RequestToJoin
rows , err := p . db . Query ( ` SELECT id,public_key,clock,ens_name,chat_id,community_id,state FROM communities_requests_to_join WHERE state = ? ` , RequestToJoinStatePending )
if err != nil {
return nil , err
}
defer rows . Close ( )
for rows . Next ( ) {
request := & RequestToJoin { }
err := rows . Scan ( & request . ID , & request . PublicKey , & request . Clock , & request . ENSName , & request . ChatID , & request . CommunityID , & request . State )
if err != nil {
return nil , err
}
requests = append ( requests , request )
}
return requests , nil
}
2022-08-04 09:44:35 +02:00
func ( p * Persistence ) PendingRequestsToJoinForCommunity ( id [ ] byte ) ( [ ] * RequestToJoin , error ) {
return p . RequestsToJoinForCommunityWithState ( id , RequestToJoinStatePending )
}
func ( p * Persistence ) DeclinedRequestsToJoinForCommunity ( id [ ] byte ) ( [ ] * RequestToJoin , error ) {
return p . RequestsToJoinForCommunityWithState ( id , RequestToJoinStateDeclined )
}
2022-10-28 11:41:20 +03:00
func ( p * Persistence ) CanceledRequestsToJoinForCommunity ( id [ ] byte ) ( [ ] * RequestToJoin , error ) {
return p . RequestsToJoinForCommunityWithState ( id , RequestToJoinStateCanceled )
}
2023-06-14 16:15:46 +02:00
func ( p * Persistence ) AcceptedRequestsToJoinForCommunity ( id [ ] byte ) ( [ ] * RequestToJoin , error ) {
return p . RequestsToJoinForCommunityWithState ( id , RequestToJoinStateAccepted )
}
refactor: EventSenders forward RequestToJoin decision to control node
This is a bigger change in how community membership requests are handled
among admins, token masters, owners, and control nodes.
Prior to this commit, all privileged users, also known as
`EventSenders`, were able to accept and reject community membership
requests and those changes would be applied by all users.
This commit changes this behaviour such that:
1. EventSenders can make a decision (accept, reject), but merely forward
their decision to the control node, which ultimately has to confirm
it
2. EventSenders are no longer removing or adding members to and from
communities
3. When an eventsender signaled a decision, the membership request will
enter a pending state (acceptedPending or rejectedPending)
4. Once a decision was made by one eventsender, no other eventsender can
override that decision
This implementation is covered with a bunch of tests:
- Ensure that decision made by event sender is shared with other event
senders
- `testAcceptMemberRequestToJoinResponseSharedWithOtherEventSenders()`
- `testRejectMemberRequestToJoinResponseSharedWithOtherEventSenders()`
- Ensure memebrship request stays pending, until control node has
confirmed decision by event senders
- `testAcceptMemberRequestToJoinNotConfirmedByControlNode()`
- `testRejectMemberRequestToJoinNotConfirmedByControlNode()`
- Ensure that decision made by event sender cannot be overriden by other
event senders
- `testEventSenderCannotOverrideRequestToJoinState()`
These test cases live in three test suites for different event sender
types respectively
- `OwnerWithoutCommunityKeyCommunityEventsSuite`
- `TokenMasterCommunityEventsSuite`
- `AdminCommunityEventsSuite`
In addition to the changes mentioned above, there's also a smaller
changes that ensures membership requests to *not* attached revealed wallet
addresses when the requests are sent to event senders (in addition to
control nodes).
Requests send to a control node will still include revealed addresses as
the control node needs them to verify token permissions.
This commit does not yet handle the case of event senders attempting to
kick and ban members.
Similar to accepting and rejecting membership requests, kicking and
banning need a new pending state. However, we don't track such state in
local databases yet so those two cases will be handled in future commit
to not have this commit grow larger.
2023-08-02 14:04:47 +02:00
func ( p * Persistence ) AcceptedPendingRequestsToJoinForCommunity ( id [ ] byte ) ( [ ] * RequestToJoin , error ) {
return p . RequestsToJoinForCommunityWithState ( id , RequestToJoinStateAcceptedPending )
}
func ( p * Persistence ) DeclinedPendingRequestsToJoinForCommunity ( id [ ] byte ) ( [ ] * RequestToJoin , error ) {
return p . RequestsToJoinForCommunityWithState ( id , RequestToJoinStateDeclinedPending )
}
2021-08-06 16:40:23 +01:00
func ( p * Persistence ) SetRequestToJoinState ( pk string , communityID [ ] byte , state RequestToJoinState ) error {
2021-01-11 11:32:51 +01:00
_ , err := p . db . Exec ( ` UPDATE communities_requests_to_join SET state = ? WHERE community_id = ? AND public_key = ? ` , state , communityID , pk )
return err
}
2023-04-21 17:18:47 +08:00
func ( p * Persistence ) DeletePendingRequestToJoin ( id [ ] byte ) error {
_ , err := p . db . Exec ( ` DELETE FROM communities_requests_to_join WHERE id = ? ` , id )
return err
}
// UpdateClockInRequestToJoin method is used for testing
func ( p * Persistence ) UpdateClockInRequestToJoin ( id [ ] byte , clock uint64 ) error {
_ , err := p . db . Exec ( ` UPDATE communities_requests_to_join SET clock = ? WHERE id = ? ` , clock , id )
return err
}
2023-07-19 15:14:42 +03:00
func ( p * Persistence ) SetMuted ( communityID [ ] byte , muted bool ) error {
_ , err := p . db . Exec ( ` UPDATE communities_communities SET muted = ? WHERE id = ? ` , muted , communityID )
return err
}
func ( p * Persistence ) MuteCommunityTill ( communityID [ ] byte , mutedTill time . Time ) error {
2023-06-17 11:19:05 +03:00
mutedTillFormatted := mutedTill . Format ( time . RFC3339 )
2023-07-19 15:14:42 +03:00
_ , err := p . db . Exec ( ` UPDATE communities_communities SET muted_till = ? WHERE id = ? ` , mutedTillFormatted , communityID )
2021-06-30 09:29:43 -04:00
return err
}
2021-01-11 11:32:51 +01:00
func ( p * Persistence ) GetRequestToJoin ( id [ ] byte ) ( * RequestToJoin , error ) {
request := & RequestToJoin { }
err := p . db . QueryRow ( ` SELECT id,public_key,clock,ens_name,chat_id,community_id,state FROM communities_requests_to_join WHERE id = ? ` , id ) . Scan ( & request . ID , & request . PublicKey , & request . Clock , & request . ENSName , & request . ChatID , & request . CommunityID , & request . State )
if err != nil {
return nil , err
}
return request , nil
}
2021-08-06 16:40:23 +01:00
2023-02-03 16:33:16 +00:00
func ( p * Persistence ) GetRequestToJoinByPkAndCommunityID ( pk string , communityID [ ] byte ) ( * RequestToJoin , error ) {
request := & RequestToJoin { }
err := p . db . QueryRow ( ` SELECT id,public_key,clock,ens_name,chat_id,community_id,state FROM communities_requests_to_join WHERE public_key = ? AND community_id = ? ` , pk , communityID ) . Scan ( & request . ID , & request . PublicKey , & request . Clock , & request . ENSName , & request . ChatID , & request . CommunityID , & request . State )
if err != nil {
return nil , err
}
return request , nil
}
2022-10-26 01:06:20 +03:00
func ( p * Persistence ) GetRequestToJoinIDByPkAndCommunityID ( pk string , communityID [ ] byte ) ( [ ] byte , error ) {
var id [ ] byte
err := p . db . QueryRow ( ` SELECT id FROM communities_requests_to_join WHERE community_id = ? AND public_key = ? ` , communityID , pk ) . Scan ( & id )
if err != nil && err != sql . ErrNoRows {
return nil , err
}
return id , nil
}
2022-10-28 11:41:20 +03:00
func ( p * Persistence ) GetRequestToJoinByPk ( pk string , communityID [ ] byte , state RequestToJoinState ) ( * RequestToJoin , error ) {
request := & RequestToJoin { }
err := p . db . QueryRow ( ` SELECT id,public_key,clock,ens_name,chat_id,community_id,state FROM communities_requests_to_join WHERE public_key = ? AND community_id = ? AND state = ? ` , pk , communityID , state ) . Scan ( & request . ID , & request . PublicKey , & request . Clock , & request . ENSName , & request . ChatID , & request . CommunityID , & request . State )
if err != nil {
return nil , err
}
return request , nil
}
2021-08-06 16:40:23 +01:00
func ( p * Persistence ) SetSyncClock ( id [ ] byte , clock uint64 ) error {
_ , err := p . db . Exec ( ` UPDATE communities_communities SET synced_at = ? WHERE id = ? AND synced_at < ? ` , clock , id , clock )
return err
}
func ( p * Persistence ) SetPrivateKey ( id [ ] byte , privKey * ecdsa . PrivateKey ) error {
_ , err := p . db . Exec ( ` UPDATE communities_communities SET private_key = ? WHERE id = ? ` , crypto . FromECDSA ( privKey ) , id )
return err
}
2022-03-08 16:25:00 +01:00
2022-09-28 14:45:34 +02:00
func ( p * Persistence ) SaveWakuMessages ( messages [ ] * types . Message ) ( err error ) {
tx , err := p . db . BeginTx ( context . Background ( ) , & sql . TxOptions { } )
if err != nil {
return
}
defer func ( ) {
if err == nil {
err = tx . Commit ( )
return
}
// don't shadow original error
_ = tx . Rollback ( )
} ( )
query := ` INSERT OR REPLACE INTO waku_messages (sig, timestamp, topic, payload, padding, hash, third_party_id) VALUES (?, ?, ?, ?, ?, ?, ?) `
stmt , err := tx . Prepare ( query )
if err != nil {
return
}
defer stmt . Close ( )
for _ , msg := range messages {
_ , err = stmt . Exec (
msg . Sig ,
msg . Timestamp ,
msg . Topic . String ( ) ,
msg . Payload ,
msg . Padding ,
types . Bytes2Hex ( msg . Hash ) ,
msg . ThirdPartyID ,
)
if err != nil {
return
}
}
return
}
2022-03-09 10:58:05 +01:00
func ( p * Persistence ) SaveWakuMessage ( message * types . Message ) error {
2022-09-28 14:45:34 +02:00
_ , err := p . db . Exec ( ` INSERT OR REPLACE INTO waku_messages (sig, timestamp, topic, payload, padding, hash, third_party_id) VALUES (?, ?, ?, ?, ?, ?, ?) ` ,
2022-03-09 10:58:05 +01:00
message . Sig ,
message . Timestamp ,
message . Topic . String ( ) ,
message . Payload ,
message . Padding ,
types . Bytes2Hex ( message . Hash ) ,
2022-09-28 14:45:34 +02:00
message . ThirdPartyID ,
2022-03-09 10:58:05 +01:00
)
return err
}
2022-03-21 15:18:36 +01:00
func wakuMessageTimestampQuery ( topics [ ] types . TopicType ) string {
query := " FROM waku_messages WHERE "
for i , topic := range topics {
query += ` topic = " ` + topic . String ( ) + ` " `
if i < len ( topics ) - 1 {
query += OR
}
}
return query
}
func ( p * Persistence ) GetOldestWakuMessageTimestamp ( topics [ ] types . TopicType ) ( uint64 , error ) {
var timestamp sql . NullInt64
query := "SELECT MIN(timestamp)"
query += wakuMessageTimestampQuery ( topics )
err := p . db . QueryRow ( query ) . Scan ( & timestamp )
return uint64 ( timestamp . Int64 ) , err
}
func ( p * Persistence ) GetLatestWakuMessageTimestamp ( topics [ ] types . TopicType ) ( uint64 , error ) {
var timestamp sql . NullInt64
query := "SELECT MAX(timestamp)"
query += wakuMessageTimestampQuery ( topics )
err := p . db . QueryRow ( query ) . Scan ( & timestamp )
return uint64 ( timestamp . Int64 ) , err
}
func ( p * Persistence ) GetWakuMessagesByFilterTopic ( topics [ ] types . TopicType , from uint64 , to uint64 ) ( [ ] types . Message , error ) {
2022-09-28 14:45:34 +02:00
query := "SELECT sig, timestamp, topic, payload, padding, hash, third_party_id FROM waku_messages WHERE timestamp >= " + fmt . Sprint ( from ) + " AND timestamp < " + fmt . Sprint ( to ) + " AND ("
2022-03-21 15:18:36 +01:00
for i , topic := range topics {
query += ` topic = " ` + topic . String ( ) + ` " `
if i < len ( topics ) - 1 {
query += OR
}
}
query += ")"
rows , err := p . db . Query ( query )
if err != nil {
return nil , err
}
defer rows . Close ( )
messages := [ ] types . Message { }
for rows . Next ( ) {
msg := types . Message { }
var topicStr string
var hashStr string
2022-09-28 14:45:34 +02:00
err := rows . Scan ( & msg . Sig , & msg . Timestamp , & topicStr , & msg . Payload , & msg . Padding , & hashStr , & msg . ThirdPartyID )
2022-03-21 15:18:36 +01:00
if err != nil {
return nil , err
}
msg . Topic = types . StringToTopic ( topicStr )
msg . Hash = types . Hex2Bytes ( hashStr )
messages = append ( messages , msg )
}
return messages , nil
}
2022-04-22 09:42:22 +02:00
func ( p * Persistence ) HasCommunityArchiveInfo ( communityID types . HexBytes ) ( exists bool , err error ) {
err = p . db . QueryRow ( ` SELECT EXISTS(SELECT 1 FROM communities_archive_info WHERE community_id = ?) ` , communityID . String ( ) ) . Scan ( & exists )
return exists , err
}
2022-12-19 09:34:37 +01:00
func ( p * Persistence ) GetLastSeenMagnetlink ( communityID types . HexBytes ) ( string , error ) {
var magnetlinkURI string
err := p . db . QueryRow ( ` SELECT last_magnetlink_uri FROM communities_archive_info WHERE community_id = ? ` , communityID . String ( ) ) . Scan ( & magnetlinkURI )
if err == sql . ErrNoRows {
return "" , nil
}
return magnetlinkURI , err
}
2022-03-21 15:18:36 +01:00
func ( p * Persistence ) GetMagnetlinkMessageClock ( communityID types . HexBytes ) ( uint64 , error ) {
var magnetlinkClock uint64
err := p . db . QueryRow ( ` SELECT magnetlink_clock FROM communities_archive_info WHERE community_id = ? ` , communityID . String ( ) ) . Scan ( & magnetlinkClock )
if err == sql . ErrNoRows {
return 0 , nil
}
return magnetlinkClock , err
}
2022-04-22 09:42:22 +02:00
func ( p * Persistence ) SaveCommunityArchiveInfo ( communityID types . HexBytes , clock uint64 , lastArchiveEndDate uint64 ) error {
_ , err := p . db . Exec ( ` INSERT INTO communities_archive_info (magnetlink_clock, last_message_archive_end_date, community_id) VALUES (?, ?, ?) ` ,
clock ,
lastArchiveEndDate ,
communityID . String ( ) )
return err
}
2022-03-21 15:18:36 +01:00
func ( p * Persistence ) UpdateMagnetlinkMessageClock ( communityID types . HexBytes , clock uint64 ) error {
_ , err := p . db . Exec ( ` UPDATE communities_archive_info SET
magnetlink_clock = ?
WHERE community_id = ? ` ,
clock ,
communityID . String ( ) )
return err
}
2022-12-19 09:34:37 +01:00
func ( p * Persistence ) UpdateLastSeenMagnetlink ( communityID types . HexBytes , magnetlinkURI string ) error {
_ , err := p . db . Exec ( ` UPDATE communities_archive_info SET
last_magnetlink_uri = ?
WHERE community_id = ? ` ,
magnetlinkURI ,
communityID . String ( ) )
return err
}
2022-03-21 15:18:36 +01:00
func ( p * Persistence ) SaveLastMessageArchiveEndDate ( communityID types . HexBytes , endDate uint64 ) error {
_ , err := p . db . Exec ( ` INSERT INTO communities_archive_info (last_message_archive_end_date, community_id) VALUES (?, ?) ` ,
endDate ,
communityID . String ( ) )
return err
}
func ( p * Persistence ) UpdateLastMessageArchiveEndDate ( communityID types . HexBytes , endDate uint64 ) error {
_ , err := p . db . Exec ( ` UPDATE communities_archive_info SET
last_message_archive_end_date = ?
WHERE community_id = ? ` ,
endDate ,
communityID . String ( ) )
return err
}
func ( p * Persistence ) GetLastMessageArchiveEndDate ( communityID types . HexBytes ) ( uint64 , error ) {
var lastMessageArchiveEndDate uint64
err := p . db . QueryRow ( ` SELECT last_message_archive_end_date FROM communities_archive_info WHERE community_id = ? ` , communityID . String ( ) ) . Scan ( & lastMessageArchiveEndDate )
if err == sql . ErrNoRows {
return 0 , nil
} else if err != nil {
return 0 , err
}
return lastMessageArchiveEndDate , nil
}
2022-12-12 10:22:37 +01:00
func ( p * Persistence ) GetMessageArchiveIDsToImport ( communityID types . HexBytes ) ( [ ] string , error ) {
rows , err := p . db . Query ( "SELECT hash FROM community_message_archive_hashes WHERE community_id = ? AND NOT(imported)" , communityID . String ( ) )
if err != nil {
return nil , err
}
defer rows . Close ( )
ids := [ ] string { }
for rows . Next ( ) {
id := ""
err := rows . Scan ( & id )
if err != nil {
return nil , err
}
ids = append ( ids , id )
}
return ids , err
}
func ( p * Persistence ) GetDownloadedMessageArchiveIDs ( communityID types . HexBytes ) ( [ ] string , error ) {
rows , err := p . db . Query ( "SELECT hash FROM community_message_archive_hashes WHERE community_id = ?" , communityID . String ( ) )
if err != nil {
return nil , err
}
defer rows . Close ( )
ids := [ ] string { }
for rows . Next ( ) {
id := ""
err := rows . Scan ( & id )
if err != nil {
return nil , err
}
ids = append ( ids , id )
}
return ids , err
}
func ( p * Persistence ) SetMessageArchiveIDImported ( communityID types . HexBytes , hash string , imported bool ) error {
_ , err := p . db . Exec ( ` UPDATE community_message_archive_hashes SET imported = ? WHERE hash = ? AND community_id = ? ` , imported , hash , communityID . String ( ) )
return err
}
2022-04-22 09:42:22 +02:00
func ( p * Persistence ) HasMessageArchiveID ( communityID types . HexBytes , hash string ) ( exists bool , err error ) {
err = p . db . QueryRow ( ` SELECT EXISTS (SELECT 1 FROM community_message_archive_hashes WHERE community_id = ? AND hash = ?) ` ,
communityID . String ( ) ,
hash ,
) . Scan ( & exists )
return exists , err
}
func ( p * Persistence ) SaveMessageArchiveID ( communityID types . HexBytes , hash string ) error {
_ , err := p . db . Exec ( ` INSERT INTO community_message_archive_hashes (community_id, hash) VALUES (?, ?) ` ,
communityID . String ( ) ,
hash ,
)
return err
}
2022-03-08 16:25:00 +01:00
func ( p * Persistence ) GetCommunitiesSettings ( ) ( [ ] CommunitySettings , error ) {
2022-06-01 09:55:48 +02:00
rows , err := p . db . Query ( "SELECT community_id, message_archive_seeding_enabled, message_archive_fetching_enabled, clock FROM communities_settings" )
2022-03-08 16:25:00 +01:00
if err != nil {
return nil , err
}
defer rows . Close ( )
communitiesSettings := [ ] CommunitySettings { }
for rows . Next ( ) {
settings := CommunitySettings { }
2022-06-01 09:55:48 +02:00
err := rows . Scan ( & settings . CommunityID , & settings . HistoryArchiveSupportEnabled , & settings . HistoryArchiveSupportEnabled , & settings . Clock )
2022-03-08 16:25:00 +01:00
if err != nil {
return nil , err
}
communitiesSettings = append ( communitiesSettings , settings )
}
return communitiesSettings , err
}
func ( p * Persistence ) CommunitySettingsExist ( communityID types . HexBytes ) ( bool , error ) {
var count int
err := p . db . QueryRow ( ` SELECT count(1) FROM communities_settings WHERE community_id = ? ` , communityID . String ( ) ) . Scan ( & count )
if err != nil {
return false , err
}
return count > 0 , nil
}
func ( p * Persistence ) GetCommunitySettingsByID ( communityID types . HexBytes ) ( * CommunitySettings , error ) {
settings := CommunitySettings { }
2022-06-01 09:55:48 +02:00
err := p . db . QueryRow ( ` SELECT community_id, message_archive_seeding_enabled, message_archive_fetching_enabled, clock FROM communities_settings WHERE community_id = ? ` , communityID . String ( ) ) . Scan ( & settings . CommunityID , & settings . HistoryArchiveSupportEnabled , & settings . HistoryArchiveSupportEnabled , & settings . Clock )
2022-03-08 16:25:00 +01:00
if err == sql . ErrNoRows {
return nil , nil
} else if err != nil {
return nil , err
}
return & settings , nil
}
func ( p * Persistence ) DeleteCommunitySettings ( communityID types . HexBytes ) error {
_ , err := p . db . Exec ( "DELETE FROM communities_settings WHERE community_id = ?" , communityID . String ( ) )
return err
}
func ( p * Persistence ) SaveCommunitySettings ( communitySettings CommunitySettings ) error {
_ , err := p . db . Exec ( ` INSERT INTO communities_settings (
community_id ,
message_archive_seeding_enabled ,
2022-06-01 09:55:48 +02:00
message_archive_fetching_enabled ,
clock
) VALUES ( ? , ? , ? , ? ) ` ,
2022-03-08 16:25:00 +01:00
communitySettings . CommunityID ,
communitySettings . HistoryArchiveSupportEnabled ,
communitySettings . HistoryArchiveSupportEnabled ,
2022-06-01 09:55:48 +02:00
communitySettings . Clock ,
2022-03-08 16:25:00 +01:00
)
return err
}
func ( p * Persistence ) UpdateCommunitySettings ( communitySettings CommunitySettings ) error {
_ , err := p . db . Exec ( ` UPDATE communities_settings SET
message_archive_seeding_enabled = ? ,
2022-06-01 09:55:48 +02:00
message_archive_fetching_enabled = ? ,
clock = ?
2022-03-08 16:25:00 +01:00
WHERE community_id = ? ` ,
communitySettings . HistoryArchiveSupportEnabled ,
communitySettings . HistoryArchiveSupportEnabled ,
2022-06-01 09:55:48 +02:00
communitySettings . Clock ,
2022-10-24 10:45:41 +02:00
communitySettings . CommunityID ,
2022-03-08 16:25:00 +01:00
)
return err
}
2022-03-21 15:18:36 +01:00
func ( p * Persistence ) GetCommunityChatIDs ( communityID types . HexBytes ) ( [ ] string , error ) {
rows , err := p . db . Query ( ` SELECT id FROM chats WHERE community_id = ? ` , communityID . String ( ) )
if err != nil {
return nil , err
}
defer rows . Close ( )
ids := [ ] string { }
for rows . Next ( ) {
id := ""
err := rows . Scan ( & id )
if err != nil {
return nil , err
}
ids = append ( ids , id )
}
return ids , nil
}
2023-01-27 14:27:24 +01:00
2023-07-07 15:03:37 +02:00
func ( p * Persistence ) GetAllCommunityTokens ( ) ( [ ] * token . CommunityToken , error ) {
2023-06-21 13:20:43 +02:00
rows , err := p . db . Query ( ` SELECT community_id , address , type , name , symbol , description , supply_str ,
2023-07-18 10:33:45 +02:00
infinite_supply , transferable , remote_self_destruct , chain_id , deploy_state , image_base64 , decimals ,
deployer , privileges_level FROM community_tokens ` )
2023-04-26 10:48:10 +02:00
if err != nil {
return nil , err
}
defer rows . Close ( )
return p . getCommunityTokensInternal ( rows )
}
2023-07-07 15:03:37 +02:00
func ( p * Persistence ) GetCommunityTokens ( communityID string ) ( [ ] * token . CommunityToken , error ) {
2023-06-21 13:20:43 +02:00
rows , err := p . db . Query ( ` SELECT community_id , address , type , name , symbol , description , supply_str ,
2023-07-18 10:33:45 +02:00
infinite_supply , transferable , remote_self_destruct , chain_id , deploy_state , image_base64 , decimals ,
deployer , privileges_level
2023-03-02 18:33:30 +01:00
FROM community_tokens WHERE community_id = ? ` , communityID )
2023-01-27 14:27:24 +01:00
if err != nil {
return nil , err
}
defer rows . Close ( )
2023-04-26 10:48:10 +02:00
return p . getCommunityTokensInternal ( rows )
}
2023-07-07 15:03:37 +02:00
func ( p * Persistence ) GetCommunityToken ( communityID string , chainID int , address string ) ( * token . CommunityToken , error ) {
token := token . CommunityToken { }
!refactor: introduce `SaveCommunityToken()` and change `AddCommunityToken()`
**This is a breaking change!**
Prior to this commit we had `AddCommunityToken(token *communities,
croppedImage CroppedImage)` that we used to
1. add a `CommunityToken` to the user's database and
2. to create a `CommunityTokenMetadata` from it which is then added to
the community's `CommunityDescription` and published to its members
However, I've then discovered that we need to separate these two things,
such that we can deploy a community token, then add it to the database
only for tracking purposes, **then** add it to the community description
(and propagate to members) once we know that the deploy tx indeed went
through.
To implement this, this commit introduces a new API
`SaveCommunityToken(token *communities.CommunityToken, croppedImage
CroppedImage)` which adds the token to the database only and doesn't
touch the community description.
The `AddCommunityToken` API is then changed that it's exclusively used
for adding an already saved `CommunityToken` to the community
description so it can be published to members. Hence, the signature is
now `AddCommunityToken(communityID string, chainID int, address
string)`, which makes this a breaking change.
Clients that used `AddCommunityToken()` before now need to ensure that
they first call `SaveCommunityToken()` as `AddCommunityToken()` will
fail otherwise.
2023-07-25 13:35:17 +02:00
var supplyStr string
2023-07-18 10:33:45 +02:00
err := p . db . QueryRow ( ` SELECT community_id , address , type , name , symbol , description , supply_str , infinite_supply ,
transferable , remote_self_destruct , chain_id , deploy_state , image_base64 , decimals , deployer , privileges_level
FROM community_tokens WHERE community_id = ? AND chain_id = ? AND address = ? ` , communityID , chainID , address ) . Scan ( & token . CommunityID , & token . Address , & token . TokenType , & token . Name ,
!refactor: introduce `SaveCommunityToken()` and change `AddCommunityToken()`
**This is a breaking change!**
Prior to this commit we had `AddCommunityToken(token *communities,
croppedImage CroppedImage)` that we used to
1. add a `CommunityToken` to the user's database and
2. to create a `CommunityTokenMetadata` from it which is then added to
the community's `CommunityDescription` and published to its members
However, I've then discovered that we need to separate these two things,
such that we can deploy a community token, then add it to the database
only for tracking purposes, **then** add it to the community description
(and propagate to members) once we know that the deploy tx indeed went
through.
To implement this, this commit introduces a new API
`SaveCommunityToken(token *communities.CommunityToken, croppedImage
CroppedImage)` which adds the token to the database only and doesn't
touch the community description.
The `AddCommunityToken` API is then changed that it's exclusively used
for adding an already saved `CommunityToken` to the community
description so it can be published to members. Hence, the signature is
now `AddCommunityToken(communityID string, chainID int, address
string)`, which makes this a breaking change.
Clients that used `AddCommunityToken()` before now need to ensure that
they first call `SaveCommunityToken()` as `AddCommunityToken()` will
fail otherwise.
2023-07-25 13:35:17 +02:00
& token . Symbol , & token . Description , & supplyStr , & token . InfiniteSupply , & token . Transferable ,
2023-07-18 10:33:45 +02:00
& token . RemoteSelfDestruct , & token . ChainID , & token . DeployState , & token . Base64Image , & token . Decimals ,
& token . Deployer , & token . PrivilegesLevel )
!refactor: introduce `SaveCommunityToken()` and change `AddCommunityToken()`
**This is a breaking change!**
Prior to this commit we had `AddCommunityToken(token *communities,
croppedImage CroppedImage)` that we used to
1. add a `CommunityToken` to the user's database and
2. to create a `CommunityTokenMetadata` from it which is then added to
the community's `CommunityDescription` and published to its members
However, I've then discovered that we need to separate these two things,
such that we can deploy a community token, then add it to the database
only for tracking purposes, **then** add it to the community description
(and propagate to members) once we know that the deploy tx indeed went
through.
To implement this, this commit introduces a new API
`SaveCommunityToken(token *communities.CommunityToken, croppedImage
CroppedImage)` which adds the token to the database only and doesn't
touch the community description.
The `AddCommunityToken` API is then changed that it's exclusively used
for adding an already saved `CommunityToken` to the community
description so it can be published to members. Hence, the signature is
now `AddCommunityToken(communityID string, chainID int, address
string)`, which makes this a breaking change.
Clients that used `AddCommunityToken()` before now need to ensure that
they first call `SaveCommunityToken()` as `AddCommunityToken()` will
fail otherwise.
2023-07-25 13:35:17 +02:00
if err == sql . ErrNoRows {
return nil , nil
} else if err != nil {
return nil , err
}
supplyBigInt , ok := new ( big . Int ) . SetString ( supplyStr , 10 )
if ok {
token . Supply = & bigint . BigInt { Int : supplyBigInt }
} else {
token . Supply = & bigint . BigInt { Int : big . NewInt ( 0 ) }
}
return & token , nil
}
2023-07-07 15:03:37 +02:00
func ( p * Persistence ) getCommunityTokensInternal ( rows * sql . Rows ) ( [ ] * token . CommunityToken , error ) {
tokens := [ ] * token . CommunityToken { }
2023-01-27 14:27:24 +01:00
for rows . Next ( ) {
2023-07-07 15:03:37 +02:00
token := token . CommunityToken { }
2023-06-21 13:20:43 +02:00
var supplyStr string
2023-01-27 14:27:24 +01:00
err := rows . Scan ( & token . CommunityID , & token . Address , & token . TokenType , & token . Name ,
2023-06-21 13:20:43 +02:00
& token . Symbol , & token . Description , & supplyStr , & token . InfiniteSupply , & token . Transferable ,
2023-07-18 10:33:45 +02:00
& token . RemoteSelfDestruct , & token . ChainID , & token . DeployState , & token . Base64Image , & token . Decimals ,
& token . Deployer , & token . PrivilegesLevel )
2023-01-27 14:27:24 +01:00
if err != nil {
return nil , err
}
2023-06-21 13:20:43 +02:00
supplyBigInt , ok := new ( big . Int ) . SetString ( supplyStr , 10 )
if ok {
token . Supply = & bigint . BigInt { Int : supplyBigInt }
} else {
token . Supply = & bigint . BigInt { Int : big . NewInt ( 0 ) }
p . logger . Error ( "can't create bigInt from string" )
}
2023-01-27 14:27:24 +01:00
tokens = append ( tokens , & token )
}
return tokens , nil
}
2023-07-07 15:03:37 +02:00
func ( p * Persistence ) HasCommunityToken ( communityID string , address string , chainID int ) ( bool , error ) {
var count int
err := p . db . QueryRow ( ` SELECT count(1) FROM community_tokens WHERE community_id = ? AND address = ? AND chain_id = ? ` , communityID , address , chainID ) . Scan ( & count )
if err != nil {
return false , err
}
return count > 0 , nil
}
func ( p * Persistence ) AddCommunityToken ( token * token . CommunityToken ) error {
2023-06-21 13:20:43 +02:00
_ , err := p . db . Exec ( ` INSERT INTO community_tokens ( community_id , address , type , name , symbol , description , supply_str ,
2023-07-18 10:33:45 +02:00
infinite_supply , transferable , remote_self_destruct , chain_id , deploy_state , image_base64 , decimals , deployer , privileges_level )
VALUES ( ? , ? , ? , ? , ? , ? , ? , ? , ? , ? , ? , ? , ? , ? , ? , ? ) ` , token . CommunityID , token . Address , token . TokenType , token . Name ,
2023-06-21 13:20:43 +02:00
token . Symbol , token . Description , token . Supply . String ( ) , token . InfiniteSupply , token . Transferable , token . RemoteSelfDestruct ,
2023-07-18 10:33:45 +02:00
token . ChainID , token . DeployState , token . Base64Image , token . Decimals , token . Deployer , token . PrivilegesLevel )
2023-01-27 14:27:24 +01:00
return err
}
2023-07-07 15:03:37 +02:00
func ( p * Persistence ) UpdateCommunityTokenState ( chainID int , contractAddress string , deployState token . DeployState ) error {
2023-06-02 10:07:00 +02:00
_ , err := p . db . Exec ( ` UPDATE community_tokens SET deploy_state = ? WHERE address = ? AND chain_id = ? ` , deployState , contractAddress , chainID )
return err
}
2023-07-18 10:33:45 +02:00
func ( p * Persistence ) UpdateCommunityTokenAddress ( chainID int , oldContractAddress string , newContractAddress string ) error {
_ , err := p . db . Exec ( ` UPDATE community_tokens SET address = ? WHERE address = ? AND chain_id = ? ` , newContractAddress , oldContractAddress , chainID )
return err
}
2023-06-21 13:20:43 +02:00
func ( p * Persistence ) UpdateCommunityTokenSupply ( chainID int , contractAddress string , supply * bigint . BigInt ) error {
_ , err := p . db . Exec ( ` UPDATE community_tokens SET supply_str = ? WHERE address = ? AND chain_id = ? ` , supply . String ( ) , contractAddress , chainID )
2023-01-27 14:27:24 +01:00
return err
}
2023-07-18 17:06:12 +02:00
2023-07-24 15:04:11 +02:00
func ( p * Persistence ) RemoveCommunityToken ( chainID int , contractAddress string ) error {
_ , err := p . db . Exec ( ` DELETE FROM community_tokens WHERE chain_id = ? AND address = ? ` , chainID , contractAddress )
return err
}
2023-07-10 17:35:15 +02:00
func decodeWrappedCommunityDescription ( wrappedDescriptionBytes [ ] byte ) ( * protobuf . CommunityDescription , error ) {
2023-07-18 17:06:12 +02:00
metadata := & protobuf . ApplicationMetadataMessage { }
2023-07-10 17:35:15 +02:00
err := proto . Unmarshal ( wrappedDescriptionBytes , metadata )
2023-07-18 17:06:12 +02:00
if err != nil {
return nil , err
}
description := & protobuf . CommunityDescription { }
err = proto . Unmarshal ( metadata . Payload , description )
if err != nil {
return nil , err
}
return description , nil
}
func decodeEventsData ( eventsBytes [ ] byte , eventsDescriptionBytes [ ] byte ) ( * EventsData , error ) {
if len ( eventsDescriptionBytes ) == 0 {
return nil , nil
}
var events [ ] CommunityEvent
if eventsBytes != nil {
var err error
events , err = communityEventsFromJSONEncodedBytes ( eventsBytes )
if err != nil {
return nil , err
}
}
return & EventsData {
EventsBaseCommunityDescription : eventsDescriptionBytes ,
Events : events ,
} , nil
}
2023-09-20 10:37:46 +02:00
func ( p * Persistence ) GetCommunityRequestsToJoinWithRevealedAddresses ( communityID [ ] byte ) ( [ ] * RequestToJoin , error ) {
requests := [ ] * RequestToJoin { }
rows , err := p . db . Query ( `
SELECT r . id , r . public_key , r . clock , r . ens_name , r . chat_id , r . state , r . community_id , a . address , a . chain_ids , a . is_airdrop_address
FROM communities_requests_to_join r
LEFT JOIN communities_requests_to_join_revealed_addresses a ON r . id = a . request_id
WHERE community_id = ? ` , communityID )
if err != nil {
return nil , err
}
defer rows . Close ( )
prevRequest := & RequestToJoin { }
for rows . Next ( ) {
request := & RequestToJoin { }
var address sql . NullString
var chainIDsStr sql . NullString
var isAirdropAddress sql . NullBool
err = rows . Scan ( & request . ID , & request . PublicKey , & request . Clock , & request . ENSName , & request . ChatID , & request . State , & request . CommunityID ,
& address , & chainIDsStr , & isAirdropAddress )
if err != nil {
return nil , err
}
revealedAccount , err := toRevealedAccount ( address , chainIDsStr , isAirdropAddress )
if err != nil {
return nil , err
}
if types . EncodeHex ( prevRequest . ID ) == types . EncodeHex ( request . ID ) {
if revealedAccount != nil {
prevRequest . RevealedAccounts = append ( prevRequest . RevealedAccounts , revealedAccount )
}
} else {
if revealedAccount != nil {
request . RevealedAccounts = [ ] * protobuf . RevealedAccount {
revealedAccount ,
}
}
requests = append ( requests , request )
prevRequest = request
}
}
return requests , nil
}
func toRevealedAccount ( rawAddress sql . NullString , rawChainIDsStr sql . NullString , isAirdropAddress sql . NullBool ) ( * protobuf . RevealedAccount , error ) {
if ! rawAddress . Valid {
return nil , nil
}
address := rawAddress . String
if address == "" {
return nil , nil
}
chainIDsStr := ""
if rawChainIDsStr . Valid {
chainIDsStr = rawChainIDsStr . String
}
chainIDs := make ( [ ] uint64 , 0 )
for _ , chainIDstr := range strings . Split ( chainIDsStr , "," ) {
if chainIDstr != "" {
chainID , err := strconv . Atoi ( chainIDstr )
if err != nil {
return nil , err
}
chainIDs = append ( chainIDs , uint64 ( chainID ) )
}
}
revealedAccount := & protobuf . RevealedAccount {
Address : address ,
ChainIds : chainIDs ,
IsAirdropAddress : false ,
}
if isAirdropAddress . Valid {
revealedAccount . IsAirdropAddress = isAirdropAddress . Bool
}
return revealedAccount , nil
}