Fix crypto.randomUUID crash on non-HTTPS origins
CI / build-and-push (push) Successful in 33s

Replace all crypto.randomUUID() calls with a uuid() utility that
falls back to Math.random-based generation when the Web Crypto API
is unavailable (plain HTTP contexts).

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
2026-04-24 18:50:37 -04:00
parent 68540ebcf9
commit 617b29bd85
5 changed files with 19 additions and 6 deletions
+2 -2
View File
@@ -1,7 +1,7 @@
import { useState } from 'react';
import { Database, ShoppingCart, Zap } from 'lucide-react';
import { useGameStore } from '@/store';
import { formatNumber, formatMoney } from '@ai-tycoon/shared';
import { formatNumber, formatMoney, uuid } from '@ai-tycoon/shared';
import type { OwnedDataset, DataDomain } from '@ai-tycoon/shared';
interface MarketplaceDataset {
@@ -49,7 +49,7 @@ export function DataPage() {
const handlePurchase = (item: MarketplaceDataset) => {
const dataset: OwnedDataset = {
id: crypto.randomUUID(),
id: uuid(),
name: item.name,
domain: item.domain,
sizeTokens: item.sizeTokens,
+3 -2
View File
@@ -19,6 +19,7 @@ import {
GPU_CONFIGS,
FUNDING_ROUNDS,
OPEN_SOURCE_REPUTATION_BOOST,
uuid,
} from '@ai-tycoon/shared';
import { INITIAL_RIVALS } from '@ai-tycoon/game-engine';
@@ -105,7 +106,7 @@ export const useGameStore = create<Store>()(
addNotification: (n) => set((s) => ({
notifications: [
{ ...n, id: crypto.randomUUID(), read: false },
{ ...n, id: uuid(), read: false },
...s.notifications.slice(0, 49),
],
})),
@@ -176,7 +177,7 @@ export const useGameStore = create<Store>()(
if (s.economy.money < buildCost) return s;
const dc: DataCenter = {
id: crypto.randomUUID(),
id: uuid(),
name,
location,
gpus: [],
@@ -1,4 +1,5 @@
import type { GameState, EventState, ActiveEvent, EventDefinition, EventCondition } from '@ai-tycoon/shared';
import { uuid } from '@ai-tycoon/shared';
export interface EventTickResult {
events: EventState;
@@ -73,7 +74,7 @@ export function processEvents(
const activeEvent: ActiveEvent = {
eventId: chosen.id,
instanceId: crypto.randomUUID(),
instanceId: uuid(),
triggeredAtTick: tick,
expiresAtTick: tick + chosen.expiryTicks,
title: chosen.title,
@@ -1,4 +1,5 @@
import type { GameState, ModelsState, TrainedModel, ModelCapabilities } from '@ai-tycoon/shared';
import { uuid } from '@ai-tycoon/shared';
export interface ModelTickResult {
modelsState: ModelsState;
@@ -79,7 +80,7 @@ function createTrainedModel(
const parameterCount = Math.pow(10, generation) * (0.5 + Math.random());
return {
id: crypto.randomUUID(),
id: uuid(),
name,
generation,
parameterCount,
+10
View File
@@ -24,6 +24,16 @@ export function formatFlops(n: number): string {
return `${formatNumber(n)} FLOPS`;
}
export function uuid(): string {
if (typeof crypto !== 'undefined' && crypto.randomUUID) {
return crypto.randomUUID();
}
return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, (c) => {
const r = (Math.random() * 16) | 0;
return (c === 'x' ? r : (r & 0x3) | 0x8).toString(16);
});
}
export function formatDuration(ticks: number): string {
if (ticks < 60) return `${ticks}s`;
if (ticks < 3600) return `${Math.floor(ticks / 60)}m ${ticks % 60}s`;