diff --git a/src/packages/server/accounts/create-account.ts b/src/packages/server/accounts/create-account.ts index 0768be410c..f2bf486908 100644 --- a/src/packages/server/accounts/create-account.ts +++ b/src/packages/server/accounts/create-account.ts @@ -85,3 +85,4 @@ export default async function createAccount({ throw error; // re-throw to bubble up to higher layers if needed } } + diff --git a/src/packages/server/messages/send.ts b/src/packages/server/messages/send.ts index a118de7593..f4b53886f5 100644 --- a/src/packages/server/messages/send.ts +++ b/src/packages/server/messages/send.ts @@ -6,11 +6,11 @@ import isValidAccount from "@cocalc/server/accounts/is-valid-account"; import getPool from "@cocalc/database/pool"; import { getServerSettings } from "@cocalc/database/settings/server-settings"; import { updateUnreadMessageCount } from "@cocalc/database/postgres/messages"; -import getAdmins from "@cocalc/server/accounts/admins"; import basePath from "@cocalc/backend/base-path"; import { join } from "path"; import type { Message } from "@cocalc/util/db-schema/messages"; import { getUser } from "@cocalc/server/purchases/statements/email-statement"; +import { getSupportAccountId } from "./support-account"; export async function name(account_id: string) { const { name: name0, email_address } = await getUser(account_id); @@ -49,12 +49,10 @@ export default async function send({ // validate sender -- if not given, assumed internal and tries to send // from support or an admin if (!from_id) { - const { support_account_id } = await getServerSettings(); - from_id = support_account_id ? support_account_id : (await getAdmins())[0]; + from_id = await getSupportAccountId(); } if (!from_id) { - // if support not configured, just make message be **from the user.** - // this is better than nothing... + // this should be impossible, but just in case. from_id = to_ids[0]; } if (!(await isValidAccount(from_id))) { diff --git a/src/packages/server/messages/support-account.ts b/src/packages/server/messages/support-account.ts new file mode 100644 index 0000000000..b24c9ce82f --- /dev/null +++ b/src/packages/server/messages/support-account.ts @@ -0,0 +1,56 @@ +import { getServerSettings } from "@cocalc/database/settings/server-settings"; +import createAccount from "@cocalc/server/accounts/create-account"; +import { uuid } from "@cocalc/util/misc"; +import { randomBytes } from "crypto"; +import getPool from "@cocalc/database/pool"; +import { callback2 } from "@cocalc/util/async-utils"; +import { db } from "@cocalc/database"; + +export async function getSupportAccountId() { + const { support_account_id } = await getServerSettings(); + if (support_account_id) { + return support_account_id; + } + return await createSupportAccount(); +} + +async function createSupportAccount() { + let email = ""; + const account_id = uuid(); + const password = randomBytes(20).toString("hex"); + const pool = getPool(); + + const { help_email } = await getServerSettings(); + if (help_email) { + const { rows } = await pool.query( + "SELECT COUNT(*) AS count FROM accounts WHERE email_address=$1", + [help_email], + ); + if (rows[0].count == 0) { + email = help_email; + } + } + await createAccount({ + email, + password, + firstName: "CoCalc", + lastName: "Support", + account_id, + noFirstProject: true, + }); + await callback2(db().set_server_setting, { + name: "support_account_id", + value: account_id, + }); + await pool.query("UPDATE accounts SET profile=$1 WHERE account_id=$2", [ + PROFILE, + account_id, + ]); + return account_id; +} + +// this is the cocalc logo as a profile image... +const PROFILE = { + image: + "", +};