console.ts 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209
  1. import { domain } from "./stage"
  2. import { EMAILOCTOPUS_API_KEY } from "./app"
  3. ////////////////
  4. // DATABASE
  5. ////////////////
  6. const cluster = planetscale.getDatabaseOutput({
  7. name: "opencode",
  8. organization: "anomalyco",
  9. })
  10. const branch =
  11. $app.stage === "production"
  12. ? planetscale.getBranchOutput({
  13. name: "production",
  14. organization: cluster.organization,
  15. database: cluster.name,
  16. })
  17. : new planetscale.Branch("DatabaseBranch", {
  18. database: cluster.name,
  19. organization: cluster.organization,
  20. name: $app.stage,
  21. parentBranch: "production",
  22. })
  23. const password = new planetscale.Password("DatabasePassword", {
  24. name: $app.stage,
  25. database: cluster.name,
  26. organization: cluster.organization,
  27. branch: branch.name,
  28. })
  29. export const database = new sst.Linkable("Database", {
  30. properties: {
  31. host: password.accessHostUrl,
  32. database: cluster.name,
  33. username: password.username,
  34. password: password.plaintext,
  35. port: 3306,
  36. },
  37. })
  38. new sst.x.DevCommand("Studio", {
  39. link: [database],
  40. dev: {
  41. command: "bun db studio",
  42. directory: "packages/console/core",
  43. autostart: true,
  44. },
  45. })
  46. ////////////////
  47. // AUTH
  48. ////////////////
  49. const GITHUB_CLIENT_ID_CONSOLE = new sst.Secret("GITHUB_CLIENT_ID_CONSOLE")
  50. const GITHUB_CLIENT_SECRET_CONSOLE = new sst.Secret("GITHUB_CLIENT_SECRET_CONSOLE")
  51. const GOOGLE_CLIENT_ID = new sst.Secret("GOOGLE_CLIENT_ID")
  52. const authStorage = new sst.cloudflare.Kv("AuthStorage")
  53. export const auth = new sst.cloudflare.Worker("AuthApi", {
  54. domain: `auth.${domain}`,
  55. handler: "packages/console/function/src/auth.ts",
  56. url: true,
  57. link: [database, authStorage, GITHUB_CLIENT_ID_CONSOLE, GITHUB_CLIENT_SECRET_CONSOLE, GOOGLE_CLIENT_ID],
  58. })
  59. ////////////////
  60. // GATEWAY
  61. ////////////////
  62. export const stripeWebhook = new stripe.WebhookEndpoint("StripeWebhookEndpoint", {
  63. url: $interpolate`https://${domain}/stripe/webhook`,
  64. enabledEvents: [
  65. "checkout.session.async_payment_failed",
  66. "checkout.session.async_payment_succeeded",
  67. "checkout.session.completed",
  68. "checkout.session.expired",
  69. "charge.refunded",
  70. "invoice.payment_succeeded",
  71. "invoice.payment_failed",
  72. "invoice.payment_action_required",
  73. "customer.created",
  74. "customer.deleted",
  75. "customer.updated",
  76. "customer.discount.created",
  77. "customer.discount.deleted",
  78. "customer.discount.updated",
  79. "customer.source.created",
  80. "customer.source.deleted",
  81. "customer.source.expiring",
  82. "customer.source.updated",
  83. "customer.subscription.created",
  84. "customer.subscription.deleted",
  85. "customer.subscription.paused",
  86. "customer.subscription.pending_update_applied",
  87. "customer.subscription.pending_update_expired",
  88. "customer.subscription.resumed",
  89. "customer.subscription.trial_will_end",
  90. "customer.subscription.updated",
  91. ],
  92. })
  93. const zenProduct = new stripe.Product("ZenBlack", {
  94. name: "OpenCode Black",
  95. })
  96. const zenPriceProps = {
  97. product: zenProduct.id,
  98. currency: "usd",
  99. recurring: {
  100. interval: "month",
  101. intervalCount: 1,
  102. },
  103. }
  104. const zenPrice200 = new stripe.Price("ZenBlackPrice", { ...zenPriceProps, unitAmount: 20000 })
  105. const zenPrice100 = new stripe.Price("ZenBlack100Price", { ...zenPriceProps, unitAmount: 10000 })
  106. const zenPrice20 = new stripe.Price("ZenBlack20Price", { ...zenPriceProps, unitAmount: 2000 })
  107. const ZEN_BLACK_PRICE = new sst.Linkable("ZEN_BLACK_PRICE", {
  108. properties: {
  109. product: zenProduct.id,
  110. plan200: zenPrice200.id,
  111. plan100: zenPrice100.id,
  112. plan20: zenPrice20.id,
  113. },
  114. })
  115. const ZEN_BLACK_LIMITS = new sst.Secret("ZEN_BLACK_LIMITS")
  116. const ZEN_MODELS = [
  117. new sst.Secret("ZEN_MODELS1"),
  118. new sst.Secret("ZEN_MODELS2"),
  119. new sst.Secret("ZEN_MODELS3"),
  120. new sst.Secret("ZEN_MODELS4"),
  121. new sst.Secret("ZEN_MODELS5"),
  122. new sst.Secret("ZEN_MODELS6"),
  123. new sst.Secret("ZEN_MODELS7"),
  124. new sst.Secret("ZEN_MODELS8"),
  125. new sst.Secret("ZEN_MODELS9"),
  126. new sst.Secret("ZEN_MODELS10"),
  127. ]
  128. const STRIPE_SECRET_KEY = new sst.Secret("STRIPE_SECRET_KEY")
  129. const STRIPE_PUBLISHABLE_KEY = new sst.Secret("STRIPE_PUBLISHABLE_KEY")
  130. const AUTH_API_URL = new sst.Linkable("AUTH_API_URL", {
  131. properties: { value: auth.url.apply((url) => url!) },
  132. })
  133. const STRIPE_WEBHOOK_SECRET = new sst.Linkable("STRIPE_WEBHOOK_SECRET", {
  134. properties: { value: stripeWebhook.secret },
  135. })
  136. const gatewayKv = new sst.cloudflare.Kv("GatewayKv")
  137. ////////////////
  138. // CONSOLE
  139. ////////////////
  140. const bucket = new sst.cloudflare.Bucket("ZenData")
  141. const bucketNew = new sst.cloudflare.Bucket("ZenDataNew")
  142. const AWS_SES_ACCESS_KEY_ID = new sst.Secret("AWS_SES_ACCESS_KEY_ID")
  143. const AWS_SES_SECRET_ACCESS_KEY = new sst.Secret("AWS_SES_SECRET_ACCESS_KEY")
  144. let logProcessor
  145. if ($app.stage === "production" || $app.stage === "frank") {
  146. const HONEYCOMB_API_KEY = new sst.Secret("HONEYCOMB_API_KEY")
  147. logProcessor = new sst.cloudflare.Worker("LogProcessor", {
  148. handler: "packages/console/function/src/log-processor.ts",
  149. link: [HONEYCOMB_API_KEY],
  150. })
  151. }
  152. new sst.cloudflare.x.SolidStart("Console", {
  153. domain,
  154. path: "packages/console/app",
  155. link: [
  156. bucket,
  157. bucketNew,
  158. database,
  159. AUTH_API_URL,
  160. STRIPE_WEBHOOK_SECRET,
  161. STRIPE_SECRET_KEY,
  162. EMAILOCTOPUS_API_KEY,
  163. AWS_SES_ACCESS_KEY_ID,
  164. AWS_SES_SECRET_ACCESS_KEY,
  165. ZEN_BLACK_PRICE,
  166. ZEN_BLACK_LIMITS,
  167. new sst.Secret("ZEN_SESSION_SECRET"),
  168. ...ZEN_MODELS,
  169. ...($dev
  170. ? [
  171. new sst.Secret("CLOUDFLARE_DEFAULT_ACCOUNT_ID", process.env.CLOUDFLARE_DEFAULT_ACCOUNT_ID!),
  172. new sst.Secret("CLOUDFLARE_API_TOKEN", process.env.CLOUDFLARE_API_TOKEN!),
  173. ]
  174. : []),
  175. gatewayKv,
  176. ],
  177. environment: {
  178. //VITE_DOCS_URL: web.url.apply((url) => url!),
  179. //VITE_API_URL: gateway.url.apply((url) => url!),
  180. VITE_AUTH_URL: auth.url.apply((url) => url!),
  181. VITE_STRIPE_PUBLISHABLE_KEY: STRIPE_PUBLISHABLE_KEY.value,
  182. },
  183. transform: {
  184. server: {
  185. transform: {
  186. worker: {
  187. placement: { mode: "smart" },
  188. tailConsumers: logProcessor ? [{ service: logProcessor.nodes.worker.scriptName }] : [],
  189. },
  190. },
  191. },
  192. },
  193. })