/* * Copyright (C) 2022-2023 Jordan Bancino <@jordan:bancino.net> * * Permission is hereby granted, free of charge, to any person * obtaining a copy of this software and associated documentation files * (the "Software"), to deal in the Software without restriction, * including without limitation the rights to use, copy, modify, merge, * publish, distribute, sublicense, and/or sell copies of the Software, * and to permit persons to whom the Software is furnished to do so, * subject to the following conditions: * * The above copyright notice and this permission notice shall be * included in all copies or portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE * SOFTWARE. */ #include #include #include #include #include #include #include #include #include static Array * RouteRegisterRegFlow(void) { Array *response = ArrayCreate(); if (!response) { return NULL; } ArrayAdd(response, UiaStageBuild("m.login.registration_token", NULL)); return response; } ROUTE_IMPL(RouteRegister, path, argp) { RouteArgs *args = argp; HashMap *request = NULL; HashMap *response = NULL; JsonValue *val; char *kind; char *username = NULL; char *password = NULL; char *initialDeviceDisplayName = NULL; int refreshToken = 0; int inhibitLogin = 0; char *deviceId = NULL; char *fullUsername; Db *db = args->matrixArgs->db; User *user = NULL; Array *uiaFlows = NULL; int uiaResult; char *session; DbRef *sessionRef; Config *config = ConfigLock(db); if (!config) { Log(LOG_ERR, "Registration endpoint failed to lock configuration."); HttpResponseStatus(args->context, HTTP_INTERNAL_SERVER_ERROR); return MatrixErrorCreate(M_UNKNOWN, NULL); } if (ArraySize(path) == 0) { if (HttpRequestMethodGet(args->context) != HTTP_POST) { HttpResponseStatus(args->context, HTTP_BAD_REQUEST); response = MatrixErrorCreate(M_UNRECOGNIZED, NULL); goto end; } request = JsonDecode(HttpServerStream(args->context)); if (!request) { HttpResponseStatus(args->context, HTTP_BAD_REQUEST); response = MatrixErrorCreate(M_NOT_JSON, NULL); goto end; } val = HashMapGet(request, "username"); if (val) { if (JsonValueType(val) != JSON_STRING) { HttpResponseStatus(args->context, HTTP_BAD_REQUEST); response = MatrixErrorCreate(M_BAD_JSON, NULL); goto finish; } username = StrDuplicate(JsonValueAsString(val)); if (!UserValidate(username, config->serverName)) { HttpResponseStatus(args->context, HTTP_BAD_REQUEST); response = MatrixErrorCreate(M_INVALID_USERNAME, NULL); goto finish; } if (UserExists(db, username)) { HttpResponseStatus(args->context, HTTP_BAD_REQUEST); response = MatrixErrorCreate(M_USER_IN_USE, NULL); goto finish; } } uiaFlows = ArrayCreate(); ArrayAdd(uiaFlows, RouteRegisterRegFlow()); if (config->flags & CONFIG_REGISTRATION) { ArrayAdd(uiaFlows, UiaDummyFlow()); } uiaResult = UiaComplete(uiaFlows, args->context, db, request, &response, config); if (uiaResult < 0) { HttpResponseStatus(args->context, HTTP_INTERNAL_SERVER_ERROR); response = MatrixErrorCreate(M_UNKNOWN, NULL); goto finish; } else if (!uiaResult) { /* UiaComplete() sets the response and status for us. */ goto finish; } kind = HashMapGet(HttpRequestParams(args->context), "kind"); /* We don't support guest accounts yet */ if (kind && !StrEquals(kind, "user")) { HttpResponseStatus(args->context, HTTP_FORBIDDEN); response = MatrixErrorCreate(M_INVALID_PARAM, NULL); goto finish; } val = HashMapGet(request, "password"); if (!val) { HttpResponseStatus(args->context, HTTP_BAD_REQUEST); response = MatrixErrorCreate(M_MISSING_PARAM, NULL); goto finish; } if (JsonValueType(val) != JSON_STRING) { HttpResponseStatus(args->context, HTTP_BAD_REQUEST); response = MatrixErrorCreate(M_BAD_JSON, NULL); goto finish; } password = StrDuplicate(JsonValueAsString(val)); val = HashMapGet(request, "device_id"); if (val) { if (JsonValueType(val) != JSON_STRING) { HttpResponseStatus(args->context, HTTP_BAD_REQUEST); response = MatrixErrorCreate(M_BAD_JSON, NULL); goto finish; } deviceId = StrDuplicate(JsonValueAsString(val)); } val = HashMapGet(request, "inhibit_login"); if (val) { if (JsonValueType(val) != JSON_BOOLEAN) { HttpResponseStatus(args->context, HTTP_BAD_REQUEST); response = MatrixErrorCreate(M_BAD_JSON, NULL); goto finish; } inhibitLogin = JsonValueAsBoolean(val); } val = HashMapGet(request, "initial_device_display_name"); if (val) { if (JsonValueType(val) != JSON_STRING) { HttpResponseStatus(args->context, HTTP_BAD_REQUEST); response = MatrixErrorCreate(M_BAD_JSON, NULL); goto finish; } initialDeviceDisplayName = StrDuplicate(JsonValueAsString(val)); } val = HashMapGet(request, "refresh_token"); if (val) { if (JsonValueType(val) != JSON_BOOLEAN) { HttpResponseStatus(args->context, HTTP_BAD_REQUEST); response = MatrixErrorCreate(M_BAD_JSON, NULL); goto finish; } refreshToken = JsonValueAsBoolean(val); } user = UserCreate(db, username, password); response = HashMapCreate(); fullUsername = StrConcat(4, "@", UserGetName(user), ":", config->serverName); HashMapSet(response, "user_id", JsonValueString(fullUsername)); Free(fullUsername); HttpResponseStatus(args->context, HTTP_OK); if (!inhibitLogin) { UserLoginInfo *loginInfo = UserLogin(user, password, deviceId, initialDeviceDisplayName, refreshToken); HashMapSet(response, "access_token", JsonValueString(loginInfo->accessToken->string)); HashMapSet(response, "device_id", JsonValueString(loginInfo->accessToken->deviceId)); if (refreshToken) { HashMapSet(response, "expires_in_ms", JsonValueInteger(loginInfo->accessToken->lifetime)); HashMapSet(response, "refresh_token", JsonValueString(loginInfo->refreshToken)); } UserAccessTokenFree(loginInfo->accessToken); Free(loginInfo->refreshToken); Free(loginInfo); } session = JsonValueAsString(JsonGet(request, 2, "auth", "session")); sessionRef = DbLock(db, 2, "user_interactive", session); if (sessionRef) { char *token = JsonValueAsString(HashMapGet(DbJson(sessionRef), "registration_token")); /* Grant the privileges specified by the given token */ if (token) { RegTokenInfo *info = RegTokenGetInfo(db, token); if (info) { UserSetPrivileges(user, UserDecodePrivileges(info->grants)); RegTokenClose(info); RegTokenFree(info); } } DbUnlock(db, sessionRef); } else { Log(LOG_WARNING, "Unable to lock UIA session reference to check"); Log(LOG_WARNING, "privileges for user registration."); } Log(LOG_INFO, "Registered user '%s'", UserGetName(user)); UserUnlock(user); finish: UiaFlowsFree(uiaFlows); Free(username); Free(password); Free(deviceId); Free(initialDeviceDisplayName); JsonFree(request); } else { if (HttpRequestMethodGet(args->context) == HTTP_GET && StrEquals(ArrayGet(path, 0), "available")) { username = HashMapGet( HttpRequestParams(args->context), "username"); if (!username) { HttpResponseStatus(args->context, HTTP_BAD_REQUEST); response = MatrixErrorCreate(M_MISSING_PARAM, NULL); } else if (!UserValidate(username, config->serverName)) { HttpResponseStatus(args->context, HTTP_BAD_REQUEST); response = MatrixErrorCreate(M_INVALID_USERNAME, NULL); } else if (!UserExists(db, username)) { response = HashMapCreate(); HashMapSet(response, "available", JsonValueBoolean(1)); } else { HttpResponseStatus(args->context, HTTP_BAD_REQUEST); response = MatrixErrorCreate(M_USER_IN_USE, NULL); } } else { HttpResponseStatus(args->context, HTTP_NOT_FOUND); response = MatrixErrorCreate(M_UNRECOGNIZED, NULL); } } end: ConfigUnlock(config); return response; }