[{"data":1,"prerenderedAt":587},["ShallowReactive",2],{"repo-tree":3,"repo-\u002Finfra\u002Fturborepo-cache\u002Freadme":283},[4,7,10,13,16,19,22,25,28,31,34,37,40,43,46,49,52,55,58,61,64,67,69,72,75,78,81,84,86,88,90,93,96,99,102,105,108,111,114,117,120,123,125,127,129,131,133,135,138,141,143,146,149,152,155,158,161,164,167,169,172,175,178,180,183,186,189,192,195,198,201,203,206,209,212,215,218,221,224,227,230,233,236,239,242,245,248,251,254,257,260,263,266,269,272,275,278,281],{"path":5,"title":6},"\u002Fagents\u002Fbackend-code-style","Backend Conventions",{"path":8,"title":9},"\u002Fagents\u002Fdatabase","Database",{"path":11,"title":12},"\u002Fagents\u002Fportal-code-style","Portal Conventions",{"path":14,"title":15},"\u002Fagents\u002Ftranslation","Translation",{"path":17,"title":18},"\u002Fconventions\u002Fbackend-coding","Backend coding conventions",{"path":20,"title":21},"\u002Fconventions\u002Ffrontend-coding","Frontend coding conventions",{"path":23,"title":24},"\u002Fdevelopment-process","Development process",{"path":26,"title":27},"\u002Flearning-api-preview-hetzner-setup","Learning API Preview on Hetzner + Cloudflare",{"path":29,"title":30},"\u002Flearning-api-preview-vm-plan","Learning API Preview VM Plan",{"path":32,"title":33},"\u002Fmonorepo-structure","Monorepo structure",{"path":35,"title":36},"\u002Foperations","Operations — bugs and support",{"path":38,"title":39},"\u002Fpostmortems\u002F2026-03-16_onboarding-currency-regression","Onboarding Zod transform silently broken — web signups assigned wrong checkout currency",{"path":41,"title":42},"\u002Fpostmortems\u002Freadme","Postmortems",{"path":44,"title":45},"\u002Fpostmortems\u002F_template","TEMPLATE",{"path":47,"title":48},"\u002Fpostmortems\u002Fposthog-comparison","Postmortem practice — comparison with PostHog",{"path":50,"title":51},"\u002Fpreview-environment-plan","Preview Environment Plan",{"path":53,"title":54},"\u002Fprinciples","Engineering principles",{"path":56,"title":57},"\u002Fworking-with-ai","Working with AI",{"path":59,"title":60},"\u002F.claude\u002Fskills\u002Feval-playground\u002Fskill","Eval Playground — Co-development Skill",{"path":62,"title":63},"\u002F.claude\u002Fskills\u002Ffigma-diff-section\u002Fskill","Figma Diff Section Pipeline",{"path":65,"title":66},"\u002Fagents","AGENTS.md",{"path":68,"title":66},"\u002Fclaude",{"path":70,"title":71},"\u002Freadme","Studyflash",{"path":73,"title":74},"\u002Fapps\u002Fcore-api\u002Fagents","Core API (apps\u002Fcore-api)",{"path":76,"title":77},"\u002Fapps\u002Fcore-api\u002Freadme","README",{"path":79,"title":80},"\u002Fapps\u002Femail-previews\u002Fagents","Email Previews (apps\u002Femail-previews)",{"path":82,"title":83},"\u002Fapps\u002Flanding-page\u002Fagents","Landing Page (apps\u002Flanding-page)",{"path":85,"title":83},"\u002Fapps\u002Flanding-page\u002Fclaude",{"path":87,"title":66},"\u002Fapps\u002Flearning-api\u002Fagents",{"path":89,"title":77},"\u002Fapps\u002Flearning-api\u002Freadme",{"path":91,"title":92},"\u002Fapps\u002Flearning-api\u002Fevals-playground\u002Feval_metrics_design","Surface-Specific Eval Metrics Design",{"path":94,"title":95},"\u002Fapps\u002Flearning-api\u002Fevals-playground\u002Ftest_set","Quiz Eval Test Set",{"path":97,"title":98},"\u002Fapps\u002Flearning-api\u002Fevals-playground\u002Ffrontend\u002Freadme","React + TypeScript + Vite",{"path":100,"title":101},"\u002Fapps\u002Flearning-api\u002Fevals-playground\u002Fknown-issues\u002Fcontent-pillar-shallow-coverage\u002Freadme","Content pillar misses subtopics in dense documents",{"path":103,"title":104},"\u002Fapps\u002Flearning-api\u002Fevals-playground\u002Fknown-issues\u002Fdocling-empty-section-headers\u002Freadme","Empty section headers dropped by docling chunker",{"path":106,"title":107},"\u002Fapps\u002Flearning-api\u002Fevals-playground\u002Fknown-issues\u002Fdocling-table-reading-order\u002Freadme","Table\u002Fbox layout causes wrong reading order",{"path":109,"title":110},"\u002Fapps\u002Flearning-api\u002Fevals-playground\u002Fmetrics\u002Freadme","Quiz eval metrics — canonical rubrics",{"path":112,"title":113},"\u002Fapps\u002Flearning-api\u002Fevals-playground\u002Freports\u002F2026-04-12-quiz-summary-feedback-current-state","Quiz and Summary Feedback Current State",{"path":115,"title":116},"\u002Fapps\u002Flearning-api\u002Fevals-playground\u002Freports\u002F2026-04-24-quiz-eval-metrics","Quiz Evaluation Metrics",{"path":118,"title":119},"\u002Fapps\u002Flearning-api\u002Fevals-playground\u002Freports\u002F2026-05-01-quiz-eval-current-state","Quiz Eval Current State",{"path":121,"title":122},"\u002Fapps\u002Flearning-api\u002Fmonitoring\u002Freadme","Monitoring Stack",{"path":124,"title":77},"\u002Fapps\u002Flearning-api\u002Fshared\u002Freadme",{"path":126,"title":77},"\u002Fapps\u002Flearning-api\u002Fworkers\u002Flearning_agents\u002Fflashcard_agent\u002Freadme",{"path":128,"title":77},"\u002Fapps\u002Flearning-api\u002Fworkers\u002Flearning_agents\u002Fingestion_agent\u002Freadme",{"path":130,"title":77},"\u002Fapps\u002Flearning-api\u002Fworkers\u002Flearning_agents\u002Fquiz_agent\u002Freadme",{"path":132,"title":77},"\u002Fapps\u002Flearning-api\u002Fworkers\u002Flearning_agents\u002Fsummary_agent\u002Freadme",{"path":134,"title":77},"\u002Fapps\u002Flearning-api\u002Fworkers\u002Fparser\u002Freadme",{"path":136,"title":137},"\u002Fapps\u002Fmarketing-emails-preview\u002Fagents","Marketing Emails Preview (apps\u002Fmarketing-emails-preview)",{"path":139,"title":140},"\u002Fapps\u002Fmobile-app\u002Fagents","StudyFlash Mobile App - Claude Code Configuration",{"path":142,"title":140},"\u002Fapps\u002Fmobile-app\u002Fclaude",{"path":144,"title":145},"\u002Fapps\u002Fmountain-max\u002Fagents","Mountain Max (apps\u002Fmountain-max)",{"path":147,"title":148},"\u002Fapps\u002Fmountain-max\u002Fgame\u002Freadme","Mountain Max Game",{"path":150,"title":151},"\u002Fapps\u002Fportal\u002Fagents","Portal (apps\u002Fportal)",{"path":153,"title":154},"\u002Fapps\u002Fportal\u002Freadme","Nuxt Minimal Starter",{"path":156,"title":157},"\u002Fapps\u002Fportal\u002Fapp\u002Fcomposables\u002Ffiles\u002Freadme","File Upload Composables",{"path":159,"title":160},"\u002Fapps\u002Fportal\u002Fdocs\u002Flibrary-routing","Library Routing Documentation",{"path":162,"title":163},"\u002Fapps\u002Fsupabase\u002Fagents","Supabase (apps\u002Fsupabase)",{"path":165,"title":166},"\u002Fapps\u002Fwrapped\u002Fagents","Wrapped (apps\u002Fwrapped)",{"path":168,"title":98},"\u002Fapps\u002Fwrapped\u002Freadme",{"path":170,"title":171},"\u002Finfra\u002Freadme","infra\u002F",{"path":173,"title":174},"\u002Finfra\u002Fdns\u002Freadme","DNS Infrastructure",{"path":176,"title":177},"\u002Finfra\u002Fdokploy\u002Freadme","studyflash-dokploy",{"path":179,"title":77},"\u002Finfra\u002Fdokploy\u002Fsdk\u002Fnodejs\u002Freadme",{"path":181,"title":182},"\u002Finfra\u002Finfisical\u002Freadme","Infisical Infrastructure",{"path":184,"title":185},"\u002Finfra\u002Flearning-api\u002Freadme","Pulumi GCP TypeScript Template",{"path":187,"title":188},"\u002Finfra\u002Fopenreplay\u002Freadme","OpenReplay on Hetzner",{"path":190,"title":191},"\u002Finfra\u002Fscripts\u002Freadme","infra\u002Fscripts\u002F",{"path":193,"title":194},"\u002Finfra\u002Fturborepo-cache\u002Freadme","Turborepo Remote Cache Infrastructure",{"path":196,"title":197},"\u002Finternal\u002Fchatwoot\u002Freadme","Chatwoot Infrastructure",{"path":199,"title":200},"\u002Finternal\u002Fchatwoot\u002Fprovider\u002Freadme","studyflash-chatwoot-provider",{"path":202,"title":77},"\u002Finternal\u002Fchatwoot\u002Fprovider\u002Fsdk\u002Fnodejs\u002Freadme",{"path":204,"title":205},"\u002Finternal\u002Fdocs\u002Freadme","internal\u002Fdocs",{"path":207,"title":208},"\u002Finternal\u002Fsupport-bot\u002Fclaude","Support Bot (Maximilian)",{"path":210,"title":211},"\u002Finternal\u002Fsupport-bot\u002Freadme","Studyflash Customer Support Bot (Maximilian)",{"path":213,"title":214},"\u002Finternal\u002Fsupport-bot\u002Fkb\u002Faccount_issues","Account Issues",{"path":216,"title":217},"\u002Finternal\u002Fsupport-bot\u002Fkb\u002Fbilling_invoice","Billing Invoice",{"path":219,"title":220},"\u002Finternal\u002Fsupport-bot\u002Fkb\u002Fcontent_upload","Content Upload",{"path":222,"title":223},"\u002Finternal\u002Fsupport-bot\u002Fkb\u002Fdata_loss","Data Loss",{"path":225,"title":226},"\u002Finternal\u002Fsupport-bot\u002Fkb\u002Fflashcard_issues","Flashcard Issues",{"path":228,"title":229},"\u002Finternal\u002Fsupport-bot\u002Fkb\u002Fgarbage","Garbage",{"path":231,"title":232},"\u002Finternal\u002Fsupport-bot\u002Fkb\u002Fgeneral_how_to","General How To",{"path":234,"title":235},"\u002Finternal\u002Fsupport-bot\u002Fkb","Knowledge Base Index",{"path":237,"title":238},"\u002Finternal\u002Fsupport-bot\u002Fkb\u002Flanguage_issues","Language Issues",{"path":240,"title":241},"\u002Finternal\u002Fsupport-bot\u002Fkb\u002Fmindmap_issues","Mindmap Issues",{"path":243,"title":244},"\u002Finternal\u002Fsupport-bot\u002Fkb\u002Fmisunderstanding","Misunderstanding",{"path":246,"title":247},"\u002Finternal\u002Fsupport-bot\u002Fkb\u002Fmock_exam_issues","Mock Exam Issues",{"path":249,"title":250},"\u002Finternal\u002Fsupport-bot\u002Fkb\u002Fpodcast_issues","Podcast Issues",{"path":252,"title":253},"\u002Finternal\u002Fsupport-bot\u002Fkb\u002Fquiz_issues","Quiz Issues",{"path":255,"title":256},"\u002Finternal\u002Fsupport-bot\u002Fkb\u002Frefund_request","Refund Request",{"path":258,"title":259},"\u002Finternal\u002Fsupport-bot\u002Fkb\u002Fsubscription_cancellation","Subscription Cancellation",{"path":261,"title":262},"\u002Finternal\u002Fsupport-bot\u002Fkb\u002Fsubscription_info","Subscription Info",{"path":264,"title":265},"\u002Finternal\u002Fsupport-bot\u002Fkb\u002Fsummary_issues","Summary Issues",{"path":267,"title":268},"\u002Finternal\u002Fsupport-bot\u002Fkb\u002Ftechnical_errors","Technical Errors",{"path":270,"title":271},"\u002Finternal\u002Fsupport-bot\u002Fkb\u002Fvideo_issues","Video Issues",{"path":273,"title":274},"\u002Fpackages\u002Fcommon\u002Fdocs\u002Fearly-access-features","Declarative Early Access Features",{"path":276,"title":277},"\u002Fpackages\u002Fcommon\u002Fscripts\u002Freadme","Common Package Scripts",{"path":279,"title":280},"\u002Fpackages\u002Fdevtools\u002Ffigma-plugins\u002Freadme","Figma plugins",{"path":282,"title":77},"\u002Fpackages\u002Fpulumi-infisical\u002Freadme",{"id":284,"title":194,"body":285,"description":580,"extension":581,"lastReviewed":582,"meta":583,"navigation":422,"owner":582,"path":193,"seo":584,"status":582,"stem":585,"tags":582,"__hash__":586},"repo\u002Finfra\u002Fturborepo-cache\u002FREADME.md",{"type":286,"value":287,"toc":572},"minimark",[288,292,313,318,350,354,382,386,467,471,474,510,514,543,547,554,568],[289,290,194],"h1",{"id":291},"turborepo-remote-cache-infrastructure",[293,294,295,296,303,304,308,309,312],"p",{},"Pulumi project managing the ",[297,298,302],"a",{"href":299,"rel":300},"https:\u002F\u002Fturborepo.dev\u002Fdocs\u002Fcore-concepts\u002Fremote-caching",[301],"nofollow","Turborepo remote cache"," on GCP (",[305,306,307],"code",{},"studyflash-project-1",", ",[305,310,311],{},"europe-west6",").",[314,315,317],"h2",{"id":316},"resources","Resources",[319,320,321,333],"ul",{},[322,323,324,328,329,332],"li",{},[325,326,327],"strong",{},"GCS bucket"," ",[305,330,331],{},"studyflash-turborepo"," — cache artifact storage",[322,334,335,328,338,341,342,349],{},[325,336,337],{},"Cloud Run v2 service",[305,339,340],{},"turborepo-remote-cache"," — runs ",[297,343,346],{"href":344,"rel":345},"https:\u002F\u002Fgithub.com\u002Fducktors\u002Fturborepo-remote-cache",[301],[305,347,348],{},"ducktors\u002Fturborepo-remote-cache"," with a GCS FUSE mount",[314,351,353],{"id":352},"prerequisites","Prerequisites",[319,355,356,368,379],{},[322,357,358,363,364,367],{},[297,359,362],{"href":360,"rel":361},"https:\u002F\u002Fwww.pulumi.com\u002Fdocs\u002Finstall\u002F",[301],"Pulumi CLI"," (",[305,365,366],{},"brew install pulumi\u002Ftap\u002Fpulumi",")",[322,369,370,375,376],{},[297,371,374],{"href":372,"rel":373},"https:\u002F\u002Fcloud.google.com\u002Fsdk\u002Fdocs\u002Finstall",[301],"gcloud CLI"," authenticated as ",[305,377,378],{},"rajiv@studyflash.ch",[322,380,381],{},"pnpm",[314,383,385],{"id":384},"setup","Setup",[387,388,393],"pre",{"className":389,"code":390,"language":391,"meta":392,"style":392},"language-bash shiki shiki-themes github-light github-dark","cd infra\u002Fturborepo-cache\npnpm install\n\npulumi login --local\npulumi stack init prod\ngcloud auth application-default login --account=rajiv@studyflash.ch\n","bash","",[305,394,395,408,417,424,436,450],{"__ignoreMap":392},[396,397,400,404],"span",{"class":398,"line":399},"line",1,[396,401,403],{"class":402},"sj4cs","cd",[396,405,407],{"class":406},"sZZnC"," infra\u002Fturborepo-cache\n",[396,409,411,414],{"class":398,"line":410},2,[396,412,381],{"class":413},"sScJk",[396,415,416],{"class":406}," install\n",[396,418,420],{"class":398,"line":419},3,[396,421,423],{"emptyLinePlaceholder":422},true,"\n",[396,425,427,430,433],{"class":398,"line":426},4,[396,428,429],{"class":413},"pulumi",[396,431,432],{"class":406}," login",[396,434,435],{"class":402}," --local\n",[396,437,439,441,444,447],{"class":398,"line":438},5,[396,440,429],{"class":413},[396,442,443],{"class":406}," stack",[396,445,446],{"class":406}," init",[396,448,449],{"class":406}," prod\n",[396,451,453,456,459,462,464],{"class":398,"line":452},6,[396,454,455],{"class":413},"gcloud",[396,457,458],{"class":406}," auth",[396,460,461],{"class":406}," application-default",[396,463,432],{"class":406},[396,465,466],{"class":402}," --account=rajiv@studyflash.ch\n",[314,468,470],{"id":469},"import-existing-resources","Import existing resources",[293,472,473],{},"Since these resources already exist in GCP, import them so Pulumi adopts them without recreating:",[387,475,477],{"className":389,"code":476,"language":391,"meta":392,"style":392},"pulumi import gcp:storage\u002Fbucket:Bucket turborepo-cache studyflash-turborepo\npulumi import gcp:cloudrunv2\u002Fservice:Service turborepo-remote-cache projects\u002Fstudyflash-project-1\u002Flocations\u002Feurope-west6\u002Fservices\u002Fturborepo-remote-cache\n",[305,478,479,495],{"__ignoreMap":392},[396,480,481,483,486,489,492],{"class":398,"line":399},[396,482,429],{"class":413},[396,484,485],{"class":406}," import",[396,487,488],{"class":406}," gcp:storage\u002Fbucket:Bucket",[396,490,491],{"class":406}," turborepo-cache",[396,493,494],{"class":406}," studyflash-turborepo\n",[396,496,497,499,501,504,507],{"class":398,"line":410},[396,498,429],{"class":413},[396,500,485],{"class":406},[396,502,503],{"class":406}," gcp:cloudrunv2\u002Fservice:Service",[396,505,506],{"class":406}," turborepo-remote-cache",[396,508,509],{"class":406}," projects\u002Fstudyflash-project-1\u002Flocations\u002Feurope-west6\u002Fservices\u002Fturborepo-remote-cache\n",[314,511,513],{"id":512},"usage","Usage",[387,515,517],{"className":389,"code":516,"language":391,"meta":392,"style":392},"pnpm preview   # check for drift\npnpm run pulumi:up # apply changes\n",[305,518,519,530],{"__ignoreMap":392},[396,520,521,523,526],{"class":398,"line":399},[396,522,381],{"class":413},[396,524,525],{"class":406}," preview",[396,527,529],{"class":528},"sJ8bj","   # check for drift\n",[396,531,532,534,537,540],{"class":398,"line":410},[396,533,381],{"class":413},[396,535,536],{"class":406}," run",[396,538,539],{"class":406}," pulumi:up",[396,541,542],{"class":528}," # apply changes\n",[314,544,546],{"id":545},"future-shared-state-backend","Future: shared state backend",[293,548,549,550,553],{},"Local state (",[305,551,552],{},"~\u002F.pulumi\u002Fstacks\u002F",") only lives on your machine. To let the team deploy, switch to a GCS bucket:",[387,555,557],{"className":389,"code":556,"language":391,"meta":392,"style":392},"pulumi login gs:\u002F\u002Fyour-state-bucket\n",[305,558,559],{"__ignoreMap":392},[396,560,561,563,565],{"class":398,"line":399},[396,562,429],{"class":413},[396,564,432],{"class":406},[396,566,567],{"class":406}," gs:\u002F\u002Fyour-state-bucket\n",[569,570,571],"style",{},"html pre.shiki code .sj4cs, html code.shiki .sj4cs{--shiki-default:#005CC5;--shiki-dark:#79B8FF}html pre.shiki code .sZZnC, html code.shiki .sZZnC{--shiki-default:#032F62;--shiki-dark:#9ECBFF}html pre.shiki code .sScJk, html code.shiki .sScJk{--shiki-default:#6F42C1;--shiki-dark:#B392F0}html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .dark .shiki span {color: var(--shiki-dark);background: var(--shiki-dark-bg);font-style: var(--shiki-dark-font-style);font-weight: var(--shiki-dark-font-weight);text-decoration: var(--shiki-dark-text-decoration);}html.dark .shiki span {color: var(--shiki-dark);background: var(--shiki-dark-bg);font-style: var(--shiki-dark-font-style);font-weight: var(--shiki-dark-font-weight);text-decoration: var(--shiki-dark-text-decoration);}html pre.shiki code .sJ8bj, html code.shiki .sJ8bj{--shiki-default:#6A737D;--shiki-dark:#6A737D}",{"title":392,"searchDepth":410,"depth":410,"links":573},[574,575,576,577,578,579],{"id":316,"depth":410,"text":317},{"id":352,"depth":410,"text":353},{"id":384,"depth":410,"text":385},{"id":469,"depth":410,"text":470},{"id":512,"depth":410,"text":513},{"id":545,"depth":410,"text":546},"Pulumi project managing the Turborepo remote cache on GCP (studyflash-project-1, europe-west6).","md",null,{},{"title":194,"description":580},"infra\u002Fturborepo-cache\u002FREADME","Sa8Y_Kumia_e1rC4kM8-VCwwIFl4i3ZvX1UUgBwLmjc",1779007964111]