0c8ad17f7c
Added sv_guidkey cvar, allowing cross-server guid key generation (although it lacks auth). Support .ico, because we can. preliminary support for sdl 2.0.6's vulkan stuff. will wait till its actually released before its properly used. Fix capturedemo. videomap should typically use premultiplied alpha, apparently. Updated sound drivers. No more old drivers. Better cvar registration. More drivers optionally support float output. Added certificate log for dtls connections. Rewrote font char cache, now supports full unicode char range, not just ucs-2. Attempt to support FreeType 2.5+ rgba fonts. XMPP now supports carbons, and shows avatars in conversations. Updated xmpp's scram auth to be more strict, including the plus variation (hopefully), to block evil tls proxies. ffmpeg plugin now uses the decoupled api for decoding too. Cef plugin updated to support fte-scheme post data properly, as well as request/response headers (like cross-origin). git-svn-id: https://svn.code.sf.net/p/fteqw/code/trunk@5148 fc73d0e0-1445-4013-8a0c-d673dee63da5
344 lines
11 KiB
C
344 lines
11 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 "5"
|
|
//#define DEFAULTDOMAIN "triptohell.info"
|
|
#define DEFAULTRESOURCE "Quake"
|
|
#define QUAKEMEDIAXMLNS "http://fteqw.com/protocol/quake"
|
|
#define DISCONODE "http://fteqw.com/ftexmpp" //some sort of client identifier
|
|
#define DEFAULTICEMODE ICEM_ICE
|
|
|
|
#define MEDIATYPE_QUAKE "quake"
|
|
#define MEDIATYPE_VIDEO "video"
|
|
#define MEDIATYPE_AUDIO "audio"
|
|
|
|
|
|
|
|
#define JCL_MAXMSGLEN 0x10000
|
|
|
|
//values are not on the wire or anything
|
|
#define CAP_VOICE (1u<<0) //supports voice
|
|
#define CAP_GOOGLE_VOICE (1u<<1) //google supports some old non-standard version of jingle.
|
|
#define CAP_VIDEO (1u<<2) //supports video calls
|
|
|
|
#define CAP_GAMEINVITE (1u<<3) //supports game invites. custom/private protocol
|
|
#define CAP_POKE (1u<<4) //can be slapped.
|
|
#define CAP_SIFT (1u<<5) //non-jingle file transfers
|
|
#define CAP_AVATARS (1u<<6) //can enable querying for user avatars, but cannot disable advertising our own.
|
|
|
|
//not actually capabilities, but just down to how we handle querying them.
|
|
#define CAP_QUERYING (1u<<29) //we've sent a query and are waiting for the response.
|
|
#define CAP_QUERIED (1u<<30) //feature capabilities are actually know.
|
|
#define CAP_QUERYFAILED (1u<<31) //caps request failed due to bad hash or some such.
|
|
|
|
//features that default to disabled.
|
|
#define CAP_DEFAULTENABLEDCAPS (CAP_VOICE/*|CAP_VIDEO*/|CAP_GAMEINVITE|CAP_POKE|CAP_AVATARS/*|CAP_SIFT*/|CAP_GOOGLE_VOICE)
|
|
|
|
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
|
|
int priority;
|
|
|
|
unsigned int buggycaps;
|
|
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 askfriend;
|
|
qboolean friended;
|
|
qboolean chatroom; //chatrooms are bizzare things that need special handling.
|
|
qboolean vcardphotochanged;
|
|
|
|
char name[256];
|
|
char vcardphotohash[41];
|
|
qhandle_t image;
|
|
|
|
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
|
|
char errormsg[256];
|
|
|
|
unsigned int enabledcapabilities;
|
|
|
|
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 resource[256];
|
|
char certificatedomain[256];
|
|
int forcetls; //-1=off, 0=ifpossible, 1=fail if can't upgrade, 2=old-style tls
|
|
qboolean savepassword;
|
|
|
|
char fulljid[256]; //this is our full username@domain/resource string
|
|
char barejid[256]; //this is our bare username@domain string
|
|
char localalias[256];//this is what's shown infront of outgoing messages. >> by default until we can get our name.
|
|
char vcardphotohash[20]; //20-byte sha1 hash.
|
|
enum
|
|
{
|
|
VCP_UNKNOWN,
|
|
VCP_NONE,
|
|
VCP_KNOWN
|
|
} vcardphotohashstatus;
|
|
qboolean vcardphotohashchanged; //forces a presence send.
|
|
|
|
int instreampos;
|
|
|
|
qboolean connecting; //still waiting for intial stream tag
|
|
qboolean connected; //fully on server and authed and everything.
|
|
qboolean issecure; //tls enabled (either upgraded or initially)
|
|
int 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 sasl_ctx_s
|
|
{
|
|
char *username; //might be different from the account name, but probably isn't.
|
|
char *domain; //might be different from the account domain, but probably isn't.
|
|
qboolean issecure; //tls enabled (either upgraded or initially)
|
|
int socket;
|
|
|
|
//this stuff should be saved
|
|
char password_plain[256]; //plain password. scrubbed if we auth using a hashed auth.
|
|
qbyte password_hash[256]; //safer password, not encrypted, but stored hashed.
|
|
size_t password_hash_size;
|
|
char password_validity[256]; //internal string used to check that the salt was unchanged
|
|
|
|
struct saslmethod_s *authmethod; //null name = oauth2->saslmethod
|
|
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
|
|
|
|
struct
|
|
{
|
|
char authnonce[256];
|
|
} digest;
|
|
|
|
struct
|
|
{
|
|
qboolean plus;
|
|
char authnonce[256];
|
|
char authvhash[20];
|
|
char authcbindtype[20];
|
|
char authcbinding[256];
|
|
hashfunc_t *hashfunc;
|
|
size_t hashsize;
|
|
} scram;
|
|
|
|
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;
|
|
} sasl;
|
|
|
|
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;
|
|
qboolean accepted; //connection is going
|
|
qboolean creator; //true if we're the creator.
|
|
unsigned int peercaps;
|
|
|
|
struct
|
|
{
|
|
char name[64]; //uniquely identifies the content within the session.
|
|
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.
|
|
struct icestate_s *ice;
|
|
char *peeraddr;
|
|
int peerport;
|
|
} content[3];
|
|
int contents;
|
|
|
|
char *with;
|
|
char sid[1];
|
|
} *c2c;
|
|
#endif
|
|
|
|
#ifdef FILETRANSFERS
|
|
struct ft_s
|
|
{
|
|
struct ft_s *next;
|
|
char fname[MAX_QPATH];
|
|
int size;
|
|
int sizedone;
|
|
char *with;
|
|
char md5hash[16];
|
|
int privateid;
|
|
char iqid[64];
|
|
char sid[64];
|
|
int blocksize;
|
|
unsigned short seq;
|
|
qhandle_t file;
|
|
qhandle_t stream;
|
|
qboolean begun; //handshake
|
|
qboolean eof;
|
|
qboolean transmitting; //we're offering
|
|
qboolean allowed; //if false, don't handshake the transfer
|
|
|
|
struct
|
|
{
|
|
char jid[128];
|
|
char host[40];
|
|
int port;
|
|
} streamhosts[16];
|
|
int nexthost;
|
|
enum
|
|
{
|
|
STRM_IDLE,
|
|
STRM_AUTH,
|
|
STRM_AUTHED,
|
|
STRM_ACTIVE
|
|
} streamstatus;
|
|
char indata[64]; //only for handshake data
|
|
int inlen;
|
|
|
|
enum
|
|
{
|
|
FT_NOTSTARTED,
|
|
FT_IBB, //in-band bytestreams
|
|
FT_BYTESTREAM //aka: relay
|
|
} method;
|
|
} *ft;
|
|
int privateidseq;
|
|
#endif
|
|
|
|
|
|
//persistant achived info about buddies, to avoid spamming the server every time we connect
|
|
//such things might result in lag mid game!
|
|
struct buddyinfo_s
|
|
{
|
|
struct buddyinfo_s *next;
|
|
char *image;
|
|
char *imagehash;
|
|
char *imagemime;
|
|
char accountdomain[1];
|
|
} *buddyinfo;
|
|
|
|
buddy_t *buddies;
|
|
struct iq_s *avatarupdate; //we only grab one buddy's photo at a time, this is to avoid too much spam.
|
|
} 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, ...);
|
|
void JCL_AddClientMessageString(jclient_t *jcl, char *msg);
|
|
qboolean JCL_FindBuddy(jclient_t *jcl, char *jid, buddy_t **buddy, bresource_t **bres, qboolean create);
|
|
|
|
//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(const char *subname, char *format, ...);
|
|
void XMPP_ConversationPrintf(const char *context, const char *title, 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);
|
|
|
|
void XMPP_FT_AcceptFile(jclient_t *jcl, int fileid, qboolean accept);
|
|
qboolean XMPP_FT_OfferAcked(jclient_t *jcl, xmltree_t *x, struct iq_s *iq);
|
|
qboolean XMPP_FT_ParseIQSet(jclient_t *jcl, char *iqfrom, char *iqid, xmltree_t *tree);
|
|
void XMPP_FT_SendFile(jclient_t *jcl, char *console, char *to, char *fname);
|
|
void XMPP_FT_Frame(jclient_t *jcl);
|
|
|
|
void Base64_Add(char *s, int len);
|
|
char *Base64_Finish(void);
|
|
int Base64_Decode(char *out, int outlen, char *src, int srclen);
|