server.ts 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108
  1. import md5 from "spark-md5";
  2. import { DEFAULT_MODELS } from "../constant";
  3. declare global {
  4. namespace NodeJS {
  5. interface ProcessEnv {
  6. PROXY_URL?: string; // docker only
  7. OPENAI_API_KEY?: string;
  8. CODE?: string;
  9. BASE_URL?: string;
  10. OPENAI_ORG_ID?: string; // openai only
  11. VERCEL?: string;
  12. BUILD_MODE?: "standalone" | "export";
  13. BUILD_APP?: string; // is building desktop app
  14. HIDE_USER_API_KEY?: string; // disable user's api key input
  15. DISABLE_GPT4?: string; // allow user to use gpt-4 or not
  16. ENABLE_BALANCE_QUERY?: string; // allow user to query balance or not
  17. DISABLE_FAST_LINK?: string; // disallow parse settings from url or not
  18. CUSTOM_MODELS?: string; // to control custom models
  19. // azure only
  20. AZURE_URL?: string; // https://{azure-url}/openai/deployments/{deploy-name}
  21. AZURE_API_KEY?: string;
  22. AZURE_API_VERSION?: string;
  23. // oauth2
  24. OAUTH_CLIENT_ID?: string;
  25. OAUTH_CLIENT_SECRET?: string;
  26. OAUTH_REDIRECT_URI?: string;
  27. OAUTH_AUTHORIZE_ENDPOINT?: string;
  28. OAUTH_USERINFO?: string;
  29. }
  30. }
  31. }
  32. const ACCESS_CODES = (function getAccessCodes(): Set<string> {
  33. const code = process.env.CODE;
  34. try {
  35. const codes = (code?.split(",") ?? [])
  36. .filter((v) => !!v)
  37. .map((v) => md5.hash(v.trim()));
  38. return new Set(codes);
  39. } catch (e) {
  40. return new Set();
  41. }
  42. })();
  43. export const getServerSideConfig = () => {
  44. if (typeof process === "undefined") {
  45. throw Error(
  46. "[Server Config] you are importing a nodejs-only module outside of nodejs",
  47. );
  48. }
  49. const disableGPT4 = !!process.env.DISABLE_GPT4;
  50. let customModels = process.env.CUSTOM_MODELS ?? "";
  51. if (disableGPT4) {
  52. if (customModels) customModels += ",";
  53. customModels += DEFAULT_MODELS.filter((m) => m.name.startsWith("gpt-4"))
  54. .map((m) => "-" + m.name)
  55. .join(",");
  56. }
  57. const isAzure = !!process.env.AZURE_URL;
  58. const apiKeyEnvVar = process.env.OPENAI_API_KEY ?? "";
  59. const apiKeys = apiKeyEnvVar.split(",").map((v) => v.trim());
  60. const randomIndex = Math.floor(Math.random() * apiKeys.length);
  61. const apiKey = apiKeys[randomIndex];
  62. console.log(
  63. `[Server Config] using ${randomIndex + 1} of ${apiKeys.length} api key`,
  64. );
  65. return {
  66. baseUrl: process.env.BASE_URL,
  67. apiKey,
  68. openaiOrgId: process.env.OPENAI_ORG_ID,
  69. isAzure,
  70. azureUrl: process.env.AZURE_URL,
  71. azureApiKey: process.env.AZURE_API_KEY,
  72. azureApiVersion: process.env.AZURE_API_VERSION,
  73. needCode: ACCESS_CODES.size > 0,
  74. code: process.env.CODE,
  75. codes: ACCESS_CODES,
  76. proxyUrl: process.env.PROXY_URL,
  77. isVercel: !!process.env.VERCEL,
  78. hideUserApiKey: !!process.env.HIDE_USER_API_KEY,
  79. disableGPT4,
  80. hideBalanceQuery: !process.env.ENABLE_BALANCE_QUERY,
  81. disableFastLink: !!process.env.DISABLE_FAST_LINK,
  82. customModels,
  83. // oauth2
  84. OAUTH_CLIENT_ID: process.env.OAUTH_CLIENT_ID,
  85. OAUTH_CLIENT_SECRET: process.env.OAUTH_CLIENT_SECRET,
  86. OAUTH_REDIRECT_URI: process.env.OAUTH_REDIRECT_URI,
  87. OAUTH_AUTHORIZE_ENDPOINT: process.env.OAUTH_AUTHORIZE_ENDPOINT,
  88. OAUTH_USERINFO: process.env.OAUTH_USERINFO,
  89. };
  90. };