seafile-server/server/notif-mgr.c
feiniks 77fa08566b
Add gen_jwt_token RPC (#674)
* Add gen_jwt_token RPC

* Set error when failed to generate jwt token

* Modify RPC name

---------

Co-authored-by: 杨赫然 <heran.yang@seafile.com>
2024-07-31 18:03:41 +08:00

162 lines
3.4 KiB
C

#include "common.h"
#include <pthread.h>
#include <jansson.h>
#include <timer.h>
#include <jwt.h>
#include "seafile-session.h"
#include "http-tx-mgr.h"
#include "notif-mgr.h"
#include "utils.h"
#include "seafile-error.h"
#include "log.h"
#define NOTIF_TIMEOUT_SEC 1
#define JWT_TOKEN_EXPIRE_TIME 300 /* 5 minutes */
struct _NotifPriv {
char *notif_url;
ConnectionPool *connection_pool;
};
typedef struct _NotifPriv NotifPriv;
typedef struct Event {
NotifPriv *priv;
char *msg;
} Event;
NotifManager *
seaf_notif_manager_new (struct _SeafileSession *seaf, char *url)
{
NotifManager *mgr = g_new0 (NotifManager, 1);
mgr->seaf = seaf;
NotifPriv *priv = g_new0 (NotifPriv, 1);
priv->connection_pool = connection_pool_new ();
if (!priv->connection_pool) {
g_free (priv);
g_free (mgr);
return NULL;
}
priv->notif_url = url;
mgr->priv = priv;
return mgr;
}
static char *
gen_jwt_token ()
{
char *jwt_token = NULL;
gint64 now = (gint64)time(NULL);
jwt_t *jwt = NULL;
if (!seaf->notif_server_private_key) {
seaf_warning ("No private key is configured for generating jwt token\n");
return NULL;
}
int ret = jwt_new (&jwt);
if (ret != 0 || jwt == NULL) {
seaf_warning ("Failed to create jwt\n");
goto out;
}
ret = jwt_add_grant_int (jwt, "exp", now + JWT_TOKEN_EXPIRE_TIME);
if (ret != 0) {
seaf_warning ("Failed to expire time to jwt\n");
goto out;
}
ret = jwt_set_alg (jwt, JWT_ALG_HS256, (unsigned char *)seaf->notif_server_private_key, strlen(seaf->notif_server_private_key));
if (ret != 0) {
seaf_warning ("Failed to set alg\n");
goto out;
}
jwt_token = jwt_encode_str (jwt);
out:
jwt_free (jwt);
return jwt_token;
}
static void*
send_event (void *data)
{
Event *event= data;
NotifPriv *priv = event->priv;
Connection *conn = NULL;
int rsp_status;
char *req_url = NULL;
char *jwt_token = NULL;
jwt_token = gen_jwt_token ();
if (!jwt_token) {
return event;
}
conn = connection_pool_get_connection (priv->connection_pool);
if (!conn) {
g_free (jwt_token);
seaf_warning ("Failed to get connection: out of memory.\n");
return event;
}
req_url = g_strdup_printf ("%s/events", priv->notif_url);
int ret;
ret = http_post (conn, req_url, jwt_token, event->msg, strlen (event->msg),
&rsp_status, NULL, NULL, TRUE, NOTIF_TIMEOUT_SEC);
if (ret < 0) {
goto out;
}
if (rsp_status != HTTP_OK) {
seaf_warning ("Failed to send event to notification server %s: %d.\n",
priv->notif_url, rsp_status);
}
out:
g_free (jwt_token);
g_free (req_url);
connection_pool_return_connection (priv->connection_pool, conn);
return event;
}
static void
free_send_event(void *data)
{
if (!data)
return;
Event *event= data;
if (event->msg)
g_free (event->msg);
g_free (event);
}
void
seaf_notif_manager_send_event (NotifManager *mgr, const char *msg)
{
Event *event = g_new0 (Event, 1);
event->priv = mgr->priv;
event->msg = g_strdup (msg);
ccnet_job_manager_schedule_job (seaf->job_mgr,
send_event,
free_send_event,
event);
}