235 lines
7.3 KiB
C
235 lines
7.3 KiB
C
|
#include "../plugin.h"
|
||
|
|
||
|
#include "xml.h"
|
||
|
|
||
|
|
||
|
//configuration
|
||
|
|
||
|
//#define NOICE //if defined, we only do simple raw udp connections.
|
||
|
//#define FILETRANSFERS //IBB only, speeds suck. autoaccept is forced on. no protection from mods stuffcmding sendfile commands. needs more extensive testing
|
||
|
#define QUAKECONNECT //including quake ICE connections (depends upon jingle)
|
||
|
#define VOIP //enables voice chat (depends upon jingle)
|
||
|
#define VOIP_LEGACY //enables google-only voice chat compat. google have not kept up with the standardisation of jingle (aka: gingle).
|
||
|
//#define VOIP_LEGACY_ONLY //disables jingle feature (advert+detection only)
|
||
|
#define JINGLE //enables jingle signalling
|
||
|
|
||
|
#ifdef JINGLE
|
||
|
#include "../../engine/common/netinc.h"
|
||
|
#else
|
||
|
#undef VOIP
|
||
|
#undef VOIP_LEGACY
|
||
|
#endif
|
||
|
|
||
|
|
||
|
#define JCL_BUILD "3"
|
||
|
#define DEFAULTDOMAIN ""
|
||
|
#define DEFAULTRESOURCE "Quake"
|
||
|
#define QUAKEMEDIATYPE "quake"
|
||
|
#define QUAKEMEDIAXMLNS "http://fteqw.com/protocol/quake"
|
||
|
#define DISCONODE "http://fteqw.com/ftexmpp" //some sort of client identifier
|
||
|
#define DEFAULTICEMODE ICEM_ICE
|
||
|
|
||
|
|
||
|
|
||
|
#define JCL_MAXMSGLEN 10000
|
||
|
|
||
|
//values are not on the wire or anything
|
||
|
#define CAP_QUERYING (1<<0) //we've sent a query and are waiting for the response.
|
||
|
#define CAP_QUERIED (1<<1) //feature capabilities are actually know.
|
||
|
#define CAP_QUERYFAILED (1<<2) //caps request failed due to bad hash or some such.
|
||
|
#define CAP_VOICE (1<<3) //supports voice
|
||
|
|
||
|
#define CAP_INVITE (1<<4) //supports game invites.
|
||
|
#define CAP_POKE (1<<5) //can be slapped.
|
||
|
#define CAP_SIFT (1<<6) //non-jingle file transfers
|
||
|
|
||
|
#define CAP_FUGOOG_VOICE (1<<7) //fuck you, google.
|
||
|
#define CAP_FUGOOG_SESSION (1<<8) //fuck you, google.
|
||
|
|
||
|
typedef struct bresource_s
|
||
|
{
|
||
|
char bstatus[128]; //basic status
|
||
|
char fstatus[128]; //full status
|
||
|
char server[256];
|
||
|
int servertype; //0=none, 1=already a client, 2=joinable
|
||
|
|
||
|
unsigned int caps;
|
||
|
char *client_node; //vendor name
|
||
|
char *client_ver; //cap hash value
|
||
|
char *client_hash; //cap hash method
|
||
|
char *client_ext; //deprecated. additionally queried
|
||
|
|
||
|
struct bresource_s *next;
|
||
|
|
||
|
char resource[1];
|
||
|
} bresource_t;
|
||
|
typedef struct buddy_s
|
||
|
{
|
||
|
bresource_t *resources;
|
||
|
bresource_t *defaultresource; //this is the one that last replied
|
||
|
int defaulttimestamp;
|
||
|
qboolean friended;
|
||
|
qboolean chatroom; //chatrooms are bizzare things that need special handling.
|
||
|
|
||
|
char name[256];
|
||
|
|
||
|
struct buddy_s *next;
|
||
|
char accountdomain[1]; //no resource on there
|
||
|
} buddy_t;
|
||
|
typedef struct jclient_s
|
||
|
{
|
||
|
int accountnum; //a private id to track which client links are associated with
|
||
|
|
||
|
char redirserveraddr[64]; //if empty, do an srv lookup.
|
||
|
|
||
|
enum
|
||
|
{
|
||
|
JCL_INACTIVE, //not trying to connect.
|
||
|
JCL_DEAD, //not connected. connection died or something.
|
||
|
JCL_AUTHING, //connected, but not able to send any info on it other than to auth
|
||
|
JCL_ACTIVE //we're connected, we got a buddy list and everything
|
||
|
} status;
|
||
|
unsigned int timeout; //reconnect/ping timer
|
||
|
|
||
|
qhandle_t socket;
|
||
|
|
||
|
//we buffer output for times when the outgoing socket is full.
|
||
|
//mostly this only happens at the start of the connection when the socket isn't actually open yet.
|
||
|
char *outbuf;
|
||
|
int outbufpos;
|
||
|
int outbuflen;
|
||
|
int outbufmax;
|
||
|
|
||
|
char bufferedinmessage[JCL_MAXMSGLEN+1]; //servers are required to be able to handle messages no shorter than a specific size.
|
||
|
//which means we need to be able to handle messages when they get to us.
|
||
|
//servers can still handle larger messages if they choose, so this might not be enough.
|
||
|
int bufferedinammount;
|
||
|
|
||
|
char defaultdest[256];
|
||
|
|
||
|
//config info
|
||
|
char serveraddr[64]; //if empty, do an srv lookup.
|
||
|
int serverport;
|
||
|
char domain[256];
|
||
|
char username[256];
|
||
|
char password[256];
|
||
|
char resource[256];
|
||
|
char certificatedomain[256];
|
||
|
int forcetls; //-1=off, 0=ifpossible, 1=fail if can't upgrade, 2=old-style tls
|
||
|
qboolean allowauth_plainnontls; //allow plain plain
|
||
|
qboolean allowauth_plaintls; //allow tls plain
|
||
|
qboolean allowauth_digestmd5; //allow digest-md5 auth
|
||
|
qboolean allowauth_scramsha1; //allow scram-sha-1 auth
|
||
|
qboolean allowauth_oauth2; //use oauth2 where possible
|
||
|
|
||
|
char jid[256]; //this is our full username@domain/resource string
|
||
|
char localalias[256];//this is what's shown infront of outgoing messages. >> by default until we can get our name.
|
||
|
|
||
|
char authnonce[256];
|
||
|
int authmode;
|
||
|
|
||
|
int tagdepth;
|
||
|
int openbracket;
|
||
|
int instreampos;
|
||
|
|
||
|
qboolean connected; //fully on server and authed and everything.
|
||
|
qboolean issecure; //tls enabled (either upgraded or initially)
|
||
|
qboolean streamdebug; //echo the stream to subconsoles
|
||
|
|
||
|
qboolean preapproval; //server supports presence preapproval
|
||
|
|
||
|
char curquakeserver[2048];
|
||
|
char defaultnamespace[2048]; //should be 'jabber:client' or blank (and spammy with all the extra xmlns attribs)
|
||
|
|
||
|
struct
|
||
|
{
|
||
|
char saslmethod[64];
|
||
|
char obtainurl[256];
|
||
|
char refreshurl[256];
|
||
|
char clientid[256];
|
||
|
char clientsecret[256];
|
||
|
char *useraccount;
|
||
|
char *scope;
|
||
|
char *accesstoken; //one-shot access token
|
||
|
char *refreshtoken; //long-term token that we can use to get new access tokens
|
||
|
char *authtoken; //short-term authorisation token, usable to get an access token (and a refresh token if we're lucky)
|
||
|
} oauth2;
|
||
|
|
||
|
struct iq_s
|
||
|
{
|
||
|
struct iq_s *next;
|
||
|
char id[64];
|
||
|
int timeout;
|
||
|
qboolean (*callback) (struct jclient_s *jcl, struct subtree_s *tree, struct iq_s *iq);
|
||
|
void *usrptr;
|
||
|
char to[1];
|
||
|
} *pendingiqs;
|
||
|
|
||
|
#ifdef JINGLE
|
||
|
struct c2c_s
|
||
|
{
|
||
|
struct c2c_s *next;
|
||
|
enum iceproto_e mediatype;
|
||
|
enum icemode_e method; //ICE_RAW or ICE_ICE. this is what the peer asked for. updated if we degrade it.
|
||
|
qboolean accepted; //connection is going
|
||
|
qboolean creator; //true if we're the creator.
|
||
|
unsigned int peercaps;
|
||
|
|
||
|
struct icestate_s *ice;
|
||
|
char *peeraddr;
|
||
|
int peerport;
|
||
|
|
||
|
char *with;
|
||
|
char sid[1];
|
||
|
} *c2c;
|
||
|
#endif
|
||
|
|
||
|
#ifdef FILETRANSFERS
|
||
|
struct ft_s
|
||
|
{
|
||
|
struct ft_s *next;
|
||
|
char fname[MAX_QPATH];
|
||
|
int size;
|
||
|
char *with;
|
||
|
char md5hash[16];
|
||
|
char sid[64];
|
||
|
int blocksize;
|
||
|
unsigned short seq;
|
||
|
qhandle_t file;
|
||
|
qboolean begun;
|
||
|
qboolean transmitting;
|
||
|
|
||
|
enum
|
||
|
{
|
||
|
FT_IBB, //in-band bytestreams
|
||
|
FT_BYTESTREAM //aka: relay
|
||
|
} method;
|
||
|
} *ft;
|
||
|
#endif
|
||
|
|
||
|
buddy_t *buddies;
|
||
|
} jclient_t;
|
||
|
|
||
|
|
||
|
#ifdef JINGLE
|
||
|
extern icefuncs_t *piceapi;
|
||
|
#endif
|
||
|
|
||
|
|
||
|
qboolean NET_DNSLookup_SRV(char *host, char *out, int outlen);
|
||
|
|
||
|
//xmpp functionality
|
||
|
struct iq_s *JCL_SendIQNode(jclient_t *jcl, qboolean (*callback) (jclient_t *jcl, xmltree_t *tree, struct iq_s *iq), char *iqtype, char *target, xmltree_t *node, qboolean destroynode);
|
||
|
void JCL_AddClientMessagef(jclient_t *jcl, char *fmt, ...);
|
||
|
qboolean JCL_FindBuddy(jclient_t *jcl, char *jid, buddy_t **buddy, bresource_t **bres);
|
||
|
|
||
|
//quake functionality
|
||
|
void JCL_GenLink(jclient_t *jcl, char *out, int outlen, char *action, char *context, char *contextres, char *sid, char *txtfmt, ...);
|
||
|
void Con_SubPrintf(char *subname, char *format, ...);
|
||
|
|
||
|
//jingle functions
|
||
|
void JCL_Join(jclient_t *jcl, char *target, char *sid, qboolean allow, int protocol);
|
||
|
void JCL_JingleTimeouts(jclient_t *jcl, qboolean killall);
|
||
|
//jingle iq message handlers
|
||
|
qboolean JCL_HandleGoogleSession(jclient_t *jcl, xmltree_t *tree, char *from, char *id);
|
||
|
qboolean JCL_ParseJingle(jclient_t *jcl, xmltree_t *tree, char *from, char *id);
|