mirror of
https://github.com/tulir/mautrix-whatsapp
synced 2024-12-14 09:23:51 +01:00
a948ea0146
See d578d1a610
Database upgrades from before v0.4.0 were squashed, users must update
to at least v0.4.0 before updating beyond this commit.
220 lines
6.3 KiB
Go
220 lines
6.3 KiB
Go
// mautrix-whatsapp - A Matrix-WhatsApp puppeting bridge.
|
|
// Copyright (C) 2021 Tulir Asokan
|
|
//
|
|
// This program is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU Affero General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
//
|
|
// This program is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU Affero General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU Affero General Public License
|
|
// along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
package database
|
|
|
|
import (
|
|
"database/sql"
|
|
|
|
log "maunium.net/go/maulogger/v2"
|
|
|
|
"maunium.net/go/mautrix/id"
|
|
"maunium.net/go/mautrix/util/dbutil"
|
|
|
|
"go.mau.fi/whatsmeow/types"
|
|
)
|
|
|
|
type PortalKey struct {
|
|
JID types.JID
|
|
Receiver types.JID
|
|
}
|
|
|
|
func NewPortalKey(jid, receiver types.JID) PortalKey {
|
|
if jid.Server == types.GroupServer {
|
|
receiver = jid
|
|
} else if jid.Server == types.LegacyUserServer {
|
|
jid.Server = types.DefaultUserServer
|
|
}
|
|
return PortalKey{
|
|
JID: jid.ToNonAD(),
|
|
Receiver: receiver.ToNonAD(),
|
|
}
|
|
}
|
|
|
|
func (key PortalKey) String() string {
|
|
if key.Receiver == key.JID {
|
|
return key.JID.String()
|
|
}
|
|
return key.JID.String() + "-" + key.Receiver.String()
|
|
}
|
|
|
|
type PortalQuery struct {
|
|
db *Database
|
|
log log.Logger
|
|
}
|
|
|
|
func (pq *PortalQuery) New() *Portal {
|
|
return &Portal{
|
|
db: pq.db,
|
|
log: pq.log,
|
|
}
|
|
}
|
|
|
|
func (pq *PortalQuery) GetAll() []*Portal {
|
|
return pq.getAll("SELECT * FROM portal")
|
|
}
|
|
|
|
func (pq *PortalQuery) GetAllForUser(userID id.UserID) []*Portal {
|
|
return pq.getAll(`
|
|
SELECT p.* FROM portal p
|
|
LEFT JOIN user_portal up ON p.jid=up.portal_jid AND p.receiver=up.portal_receiver
|
|
WHERE mxid<>'' AND up.user_mxid=$1
|
|
`, userID)
|
|
}
|
|
|
|
func (pq *PortalQuery) GetByJID(key PortalKey) *Portal {
|
|
return pq.get("SELECT * FROM portal WHERE jid=$1 AND receiver=$2", key.JID, key.Receiver)
|
|
}
|
|
|
|
func (pq *PortalQuery) GetByMXID(mxid id.RoomID) *Portal {
|
|
return pq.get("SELECT * FROM portal WHERE mxid=$1", mxid)
|
|
}
|
|
|
|
func (pq *PortalQuery) GetAllByJID(jid types.JID) []*Portal {
|
|
return pq.getAll("SELECT * FROM portal WHERE jid=$1", jid.ToNonAD())
|
|
}
|
|
|
|
func (pq *PortalQuery) FindPrivateChats(receiver types.JID) []*Portal {
|
|
return pq.getAll("SELECT * FROM portal WHERE receiver=$1 AND jid LIKE '%@s.whatsapp.net'", receiver.ToNonAD())
|
|
}
|
|
|
|
func (pq *PortalQuery) FindPrivateChatsNotInSpace(receiver types.JID) (keys []PortalKey) {
|
|
receiver = receiver.ToNonAD()
|
|
rows, err := pq.db.Query(`
|
|
SELECT jid FROM portal
|
|
LEFT JOIN user_portal ON portal.jid=user_portal.portal_jid AND portal.receiver=user_portal.portal_receiver
|
|
WHERE mxid<>'' AND receiver=$1 AND (in_space=false OR in_space IS NULL)
|
|
`, receiver)
|
|
if err != nil || rows == nil {
|
|
return
|
|
}
|
|
for rows.Next() {
|
|
var key PortalKey
|
|
key.Receiver = receiver
|
|
err = rows.Scan(&key.JID)
|
|
if err == nil {
|
|
keys = append(keys, key)
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
func (pq *PortalQuery) getAll(query string, args ...interface{}) (portals []*Portal) {
|
|
rows, err := pq.db.Query(query, args...)
|
|
if err != nil || rows == nil {
|
|
return nil
|
|
}
|
|
defer rows.Close()
|
|
for rows.Next() {
|
|
portals = append(portals, pq.New().Scan(rows))
|
|
}
|
|
return
|
|
}
|
|
|
|
func (pq *PortalQuery) get(query string, args ...interface{}) *Portal {
|
|
row := pq.db.QueryRow(query, args...)
|
|
if row == nil {
|
|
return nil
|
|
}
|
|
return pq.New().Scan(row)
|
|
}
|
|
|
|
type Portal struct {
|
|
db *Database
|
|
log log.Logger
|
|
|
|
Key PortalKey
|
|
MXID id.RoomID
|
|
|
|
Name string
|
|
Topic string
|
|
Avatar string
|
|
AvatarURL id.ContentURI
|
|
Encrypted bool
|
|
|
|
FirstEventID id.EventID
|
|
NextBatchID id.BatchID
|
|
|
|
RelayUserID id.UserID
|
|
|
|
ExpirationTime uint32
|
|
}
|
|
|
|
func (portal *Portal) Scan(row dbutil.Scannable) *Portal {
|
|
var mxid, avatarURL, firstEventID, nextBatchID, relayUserID sql.NullString
|
|
err := row.Scan(&portal.Key.JID, &portal.Key.Receiver, &mxid, &portal.Name, &portal.Topic, &portal.Avatar, &avatarURL, &portal.Encrypted, &firstEventID, &nextBatchID, &relayUserID, &portal.ExpirationTime)
|
|
if err != nil {
|
|
if err != sql.ErrNoRows {
|
|
portal.log.Errorln("Database scan failed:", err)
|
|
}
|
|
return nil
|
|
}
|
|
portal.MXID = id.RoomID(mxid.String)
|
|
portal.AvatarURL, _ = id.ParseContentURI(avatarURL.String)
|
|
portal.FirstEventID = id.EventID(firstEventID.String)
|
|
portal.NextBatchID = id.BatchID(nextBatchID.String)
|
|
portal.RelayUserID = id.UserID(relayUserID.String)
|
|
return portal
|
|
}
|
|
|
|
func (portal *Portal) mxidPtr() *id.RoomID {
|
|
if len(portal.MXID) > 0 {
|
|
return &portal.MXID
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (portal *Portal) relayUserPtr() *id.UserID {
|
|
if len(portal.RelayUserID) > 0 {
|
|
return &portal.RelayUserID
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (portal *Portal) Insert() {
|
|
_, err := portal.db.Exec("INSERT INTO portal (jid, receiver, mxid, name, topic, avatar, avatar_url, encrypted, first_event_id, next_batch_id, relay_user_id, expiration_time) VALUES ($1, $2, $3, $4, $5, $6, $7, $8, $9, $10, $11, $12)",
|
|
portal.Key.JID, portal.Key.Receiver, portal.mxidPtr(), portal.Name, portal.Topic, portal.Avatar, portal.AvatarURL.String(), portal.Encrypted, portal.FirstEventID.String(), portal.NextBatchID.String(), portal.relayUserPtr(), portal.ExpirationTime)
|
|
if err != nil {
|
|
portal.log.Warnfln("Failed to insert %s: %v", portal.Key, err)
|
|
}
|
|
}
|
|
|
|
func (portal *Portal) Update(txn *sql.Tx) {
|
|
query := `
|
|
UPDATE portal
|
|
SET mxid=$1, name=$2, topic=$3, avatar=$4, avatar_url=$5, encrypted=$6, first_event_id=$7, next_batch_id=$8, relay_user_id=$9, expiration_time=$10
|
|
WHERE jid=$11 AND receiver=$12
|
|
`
|
|
args := []interface{}{
|
|
portal.mxidPtr(), portal.Name, portal.Topic, portal.Avatar, portal.AvatarURL.String(), portal.Encrypted, portal.FirstEventID.String(), portal.NextBatchID.String(), portal.relayUserPtr(), portal.ExpirationTime, portal.Key.JID, portal.Key.Receiver,
|
|
}
|
|
var err error
|
|
if txn != nil {
|
|
_, err = txn.Exec(query, args...)
|
|
} else {
|
|
_, err = portal.db.Exec(query, args...)
|
|
}
|
|
if err != nil {
|
|
portal.log.Warnfln("Failed to update %s: %v", portal.Key, err)
|
|
}
|
|
}
|
|
|
|
func (portal *Portal) Delete() {
|
|
_, err := portal.db.Exec("DELETE FROM portal WHERE jid=$1 AND receiver=$2", portal.Key.JID, portal.Key.Receiver)
|
|
if err != nil {
|
|
portal.log.Warnfln("Failed to delete %s: %v", portal.Key, err)
|
|
}
|
|
}
|