pygomx: dm support

This commit is contained in:
saces 2026-04-13 22:08:04 +02:00
parent d657fea159
commit 4c9ba7dcdb
7 changed files with 121 additions and 3 deletions

View file

@ -87,6 +87,19 @@ func (mxc *MXClient) _storeDirectMap() error {
return nil
}
func (mxc *MXClient) GetUserDM(mxid string) []string {
var res = make([]string, 0)
for room, uids := range mxc._directMap {
for _, uid := range uids {
if uid.String() == mxid {
res = append(res, room.String())
}
}
}
return res
}
func (mxc *MXClient) _onEventMember(ctx context.Context, evt *event.Event) {
if evt.GetStateKey() == mxc.UserID.String() && evt.Content.AsMember().Membership == event.MembershipInvite {
if evt.Content.AsMember().IsDirect {
@ -108,8 +121,18 @@ func (mxc *MXClient) _onEventMember(ctx context.Context, evt *event.Event) {
Str("inviter", evt.Sender.String()).
Msg("Failed to join room after invite")
}
} else if evt.Content.AsMember().Membership == event.MembershipJoin {
out, err := json.Marshal(evt)
if err != nil {
log.Error().Err(err).
Str("id", evt.ID.String()).
Str("joiner", evt.Sender.String()).
Msg("Marshalling error")
return
}
mxc.OnEvent(string(out))
} else {
fmt.Printf("\nGot member event: %#v\n", evt)
fmt.Printf("\nGot member event: %s\n%#v\n", evt.GetStateKey(), evt)
}
}
@ -161,6 +184,23 @@ func (mxc *MXClient) LeaveRoomAndForget(ctx context.Context, room id.RoomID) err
return nil
}
func (mxc *MXClient) CreateDM(ctx context.Context, uid id.UserID) (resp *mautrix.RespCreateRoom, err error) {
req := mautrix.ReqCreateRoom{
IsDirect: true,
Preset: "trusted_private_chat",
Invite: []id.UserID{uid},
}
resp, err = mxc.CreateRoom(context.Background(), &req)
if err != nil {
return
}
mxc.AddDirectRoom(uid, resp.RoomID)
err = mxc._storeDirectMap()
return
}
type sendmessage_data struct {
RoomId id.RoomID `json:"roomid"`
Type event.Type `json:"type"`

View file

@ -435,7 +435,54 @@ func apiv0_createroom(cid C.int, data *C.char) *C.char {
return C.CString(fmt.Sprintf("ERR: %v", err))
}
out, err := json.Marshal(resp)
type roomResult struct {
RoomId id.RoomID `json:"roomid"`
IsDirect bool `json:"is_direct"`
}
out, err := json.Marshal(roomResult{RoomId: resp.RoomID, IsDirect: false})
if err != nil {
return C.CString(fmt.Sprintf("ERR: %v", err))
}
s := string(out)
return C.CString(s)
}
//export apiv0_createdm
func apiv0_createdm(cid C.int, uid *C.char) *C.char {
cli, err := getClient(int(cid))
if err != nil {
return C.CString(fmt.Sprintf("ERR: %v", err))
}
mxid := C.GoString(uid)
resp, err := cli.CreateDM(context.Background(), id.UserID(mxid))
if err != nil {
return C.CString(fmt.Sprintf("ERR: %v", err))
}
type roomResult struct {
RoomId id.RoomID `json:"roomid"`
IsDirect bool `json:"is_direct"`
}
out, err := json.Marshal(roomResult{RoomId: resp.RoomID, IsDirect: true})
if err != nil {
return C.CString(fmt.Sprintf("ERR: %v", err))
}
s := string(out)
return C.CString(s)
}
//export apiv0_getuserdm
func apiv0_getuserdm(cid C.int, userid *C.char) *C.char {
cli, err := getClient(int(cid))
if err != nil {
return C.CString(fmt.Sprintf("ERR: %v", err))
}
list := cli.GetUserDM(C.GoString(userid))
out, err := json.Marshal(list)
if err != nil {
return C.CString(fmt.Sprintf("ERR: %v", err))
}

View file

@ -63,7 +63,9 @@ ffibuilder.cdef(
extern char* apiv0_leaveroom(int cid, char* roomid);
extern char* apiv0_joinedrooms(int cid);
extern char* apiv0_createroom(int cid, char* data);
extern char* apiv0_createdm(int cid, char* uid);
extern char* apiv0_genericrequest(int cid, char* method, char* path, char* data);
extern char* apiv0_getuserdm(int cid, char* userid);
extern int apiv0_removeclient(int cid);
extern char* apiv0_listclients();
extern char* apiv0_getoptions(int cid);

View file

@ -3,10 +3,12 @@ FreeCString
apiv0_createclient
apiv0_createclient_pass
apiv0_createroom
apiv0_createdm
apiv0_deinitialize
apiv0_discover
apiv0_genericrequest
apiv0_getoptions
apiv0_getuserdm
apiv0_initialize
apiv0_joinedrooms
apiv0_leaveroom

View file

@ -4,7 +4,7 @@ import logging
from _pygomx import lib
from .util import _stringresult, _autostring, _autodict
from .util import _stringresult, _autostring, _autodict, _autolist
from .errors import CheckApiError, CheckApiErrorOnly, CheckApiResult
logger = logging.getLogger(__name__)
@ -57,6 +57,14 @@ class ApiV0Api:
)
)
@staticmethod
def createdm(cid, uid):
return _stringresult(lib.apiv0_createdm(cid, _autostring(uid)))
@staticmethod
def getuserdm(cid, userid):
return _stringresult(lib.apiv0_getuserdm(cid, _autostring(userid)))
class ApiV0:
"""ApiV0"""

View file

@ -96,6 +96,13 @@ class _AsyncClient:
r = ApiV0Api.generic(self.client_id, method, path, data)
return CheckApiErrorOnly(r)
async def createdm(self, uid):
r = ApiV0Api.createdm(self.client_id, uid)
return CheckApiResult(r)
async def getuserdm(self, userid):
r = ApiV0Api.getuserdm(self.client_id, userid)
return CheckApiResult(r)
def process_event(self, evt):
if hasattr(self, "on_event") and callable(self.on_event):

View file

@ -30,3 +30,15 @@ def _autodict(xdict):
return json.dumps(xdict).encode(encoding="utf-8")
case _:
raise TypeError("only str or bytes or dict allowed")
def _autolist(xlist):
match xlist:
case bytes():
return xlist
case str():
return xlist.encode(encoding="utf-8")
case list():
return json.dumps(xlist).encode(encoding="utf-8")
case _:
raise TypeError("only str or bytes or list allowed")