chore: initialize monorepo with project scaffolding, configs, and CI setup

This commit is contained in:
oib
2025-09-27 06:05:25 +02:00
commit fe29631a86
170 changed files with 13708 additions and 0 deletions

View File

@@ -0,0 +1,207 @@
"use strict";
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
return new (P || (P = Promise))(function (resolve, reject) {
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
step((generator = generator.apply(thisArg, _arguments || [])).next());
});
};
var __generator = (this && this.__generator) || function (thisArg, body) {
var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g = Object.create((typeof Iterator === "function" ? Iterator : Object).prototype);
return g.next = verb(0), g["throw"] = verb(1), g["return"] = verb(2), typeof Symbol === "function" && (g[Symbol.iterator] = function() { return this; }), g;
function verb(n) { return function (v) { return step([n, v]); }; }
function step(op) {
if (f) throw new TypeError("Generator is already executing.");
while (g && (g = 0, op[0] && (_ = 0)), _) try {
if (f = 1, y && (t = op[0] & 2 ? y["return"] : op[0] ? y["throw"] || ((t = y["return"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;
if (y = 0, t) op = [op[0] & 2, t.value];
switch (op[0]) {
case 0: case 1: t = op; break;
case 4: _.label++; return { value: op[1], done: false };
case 5: _.label++; y = op[1]; op = [0]; continue;
case 7: op = _.ops.pop(); _.trys.pop(); continue;
default:
if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; }
if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; }
if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; }
if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; }
if (t[2]) _.ops.pop();
_.trys.pop(); continue;
}
op = body.call(thisArg, _);
} catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; }
if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };
}
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.getDataMode = getDataMode;
exports.setDataMode = setDataMode;
exports.fetchBlocks = fetchBlocks;
exports.fetchTransactions = fetchTransactions;
exports.fetchAddresses = fetchAddresses;
exports.fetchReceipts = fetchReceipts;
var config_1 = require("../config");
var currentMode = config_1.CONFIG.dataMode;
function getDataMode() {
return currentMode;
}
function setDataMode(mode) {
currentMode = mode;
}
function fetchBlocks() {
return __awaiter(this, void 0, void 0, function () {
var data, response, data, error_1;
return __generator(this, function (_a) {
switch (_a.label) {
case 0:
if (!(getDataMode() === "mock")) return [3 /*break*/, 2];
return [4 /*yield*/, fetchMock("blocks")];
case 1:
data = _a.sent();
return [2 /*return*/, data.items];
case 2:
_a.trys.push([2, 5, , 6]);
return [4 /*yield*/, fetch("".concat(config_1.CONFIG.apiBaseUrl, "/v1/blocks"))];
case 3:
response = _a.sent();
if (!response.ok) {
throw new Error("Failed to fetch blocks: ".concat(response.status));
}
return [4 /*yield*/, response.json()];
case 4:
data = (_a.sent());
return [2 /*return*/, data.items];
case 5:
error_1 = _a.sent();
console.warn("[Explorer] Failed to fetch live block data", error_1);
return [2 /*return*/, []];
case 6: return [2 /*return*/];
}
});
});
}
function fetchTransactions() {
return __awaiter(this, void 0, void 0, function () {
var data, response, data, error_2;
return __generator(this, function (_a) {
switch (_a.label) {
case 0:
if (!(getDataMode() === "mock")) return [3 /*break*/, 2];
return [4 /*yield*/, fetchMock("transactions")];
case 1:
data = _a.sent();
return [2 /*return*/, data.items];
case 2:
_a.trys.push([2, 5, , 6]);
return [4 /*yield*/, fetch("".concat(config_1.CONFIG.apiBaseUrl, "/v1/transactions"))];
case 3:
response = _a.sent();
if (!response.ok) {
throw new Error("Failed to fetch transactions: ".concat(response.status));
}
return [4 /*yield*/, response.json()];
case 4:
data = (_a.sent());
return [2 /*return*/, data.items];
case 5:
error_2 = _a.sent();
console.warn("[Explorer] Failed to fetch live transaction data", error_2);
return [2 /*return*/, []];
case 6: return [2 /*return*/];
}
});
});
}
function fetchAddresses() {
return __awaiter(this, void 0, void 0, function () {
var data, response, data, error_3;
return __generator(this, function (_a) {
switch (_a.label) {
case 0:
if (!(getDataMode() === "mock")) return [3 /*break*/, 2];
return [4 /*yield*/, fetchMock("addresses")];
case 1:
data = _a.sent();
return [2 /*return*/, Array.isArray(data) ? data : [data]];
case 2:
_a.trys.push([2, 5, , 6]);
return [4 /*yield*/, fetch("".concat(config_1.CONFIG.apiBaseUrl, "/v1/addresses"))];
case 3:
response = _a.sent();
if (!response.ok) {
throw new Error("Failed to fetch addresses: ".concat(response.status));
}
return [4 /*yield*/, response.json()];
case 4:
data = (_a.sent());
return [2 /*return*/, Array.isArray(data) ? data : data.items];
case 5:
error_3 = _a.sent();
console.warn("[Explorer] Failed to fetch live address data", error_3);
return [2 /*return*/, []];
case 6: return [2 /*return*/];
}
});
});
}
function fetchReceipts() {
return __awaiter(this, void 0, void 0, function () {
var data, response, data, error_4;
return __generator(this, function (_a) {
switch (_a.label) {
case 0:
if (!(getDataMode() === "mock")) return [3 /*break*/, 2];
return [4 /*yield*/, fetchMock("receipts")];
case 1:
data = _a.sent();
return [2 /*return*/, data.items];
case 2:
_a.trys.push([2, 5, , 6]);
return [4 /*yield*/, fetch("".concat(config_1.CONFIG.apiBaseUrl, "/v1/receipts"))];
case 3:
response = _a.sent();
if (!response.ok) {
throw new Error("Failed to fetch receipts: ".concat(response.status));
}
return [4 /*yield*/, response.json()];
case 4:
data = (_a.sent());
return [2 /*return*/, data.items];
case 5:
error_4 = _a.sent();
console.warn("[Explorer] Failed to fetch live receipt data", error_4);
return [2 /*return*/, []];
case 6: return [2 /*return*/];
}
});
});
}
function fetchMock(resource) {
return __awaiter(this, void 0, void 0, function () {
var url, response, error_5;
return __generator(this, function (_a) {
switch (_a.label) {
case 0:
url = "".concat(config_1.CONFIG.mockBasePath, "/").concat(resource, ".json");
_a.label = 1;
case 1:
_a.trys.push([1, 4, , 5]);
return [4 /*yield*/, fetch(url)];
case 2:
response = _a.sent();
if (!response.ok) {
throw new Error("Request failed with status ".concat(response.status));
}
return [4 /*yield*/, response.json()];
case 3: return [2 /*return*/, (_a.sent())];
case 4:
error_5 = _a.sent();
console.warn("[Explorer] Failed to fetch mock data from ".concat(url), error_5);
return [2 /*return*/, []];
case 5: return [2 /*return*/];
}
});
});
}

View File

@@ -0,0 +1,112 @@
import { CONFIG, type DataMode } from "../config";
import type {
BlockListResponse,
TransactionListResponse,
AddressDetailResponse,
ReceiptListResponse,
BlockSummary,
TransactionSummary,
AddressSummary,
ReceiptSummary,
} from "./models.ts";
let currentMode: DataMode = CONFIG.dataMode;
export function getDataMode(): DataMode {
return currentMode;
}
export function setDataMode(mode: DataMode): void {
currentMode = mode;
}
export async function fetchBlocks(): Promise<BlockSummary[]> {
if (getDataMode() === "mock") {
const data = await fetchMock<BlockListResponse>("blocks");
return data.items;
}
try {
const response = await fetch(`${CONFIG.apiBaseUrl}/v1/blocks`);
if (!response.ok) {
throw new Error(`Failed to fetch blocks: ${response.status}`);
}
const data = (await response.json()) as BlockListResponse;
return data.items;
} catch (error) {
console.warn("[Explorer] Failed to fetch live block data", error);
return [];
}
}
export async function fetchTransactions(): Promise<TransactionSummary[]> {
if (getDataMode() === "mock") {
const data = await fetchMock<TransactionListResponse>("transactions");
return data.items;
}
try {
const response = await fetch(`${CONFIG.apiBaseUrl}/v1/transactions`);
if (!response.ok) {
throw new Error(`Failed to fetch transactions: ${response.status}`);
}
const data = (await response.json()) as TransactionListResponse;
return data.items;
} catch (error) {
console.warn("[Explorer] Failed to fetch live transaction data", error);
return [];
}
}
export async function fetchAddresses(): Promise<AddressSummary[]> {
if (getDataMode() === "mock") {
const data = await fetchMock<AddressDetailResponse | AddressDetailResponse[]>("addresses");
return Array.isArray(data) ? data : [data];
}
try {
const response = await fetch(`${CONFIG.apiBaseUrl}/v1/addresses`);
if (!response.ok) {
throw new Error(`Failed to fetch addresses: ${response.status}`);
}
const data = (await response.json()) as { items: AddressDetailResponse[] } | AddressDetailResponse[];
return Array.isArray(data) ? data : data.items;
} catch (error) {
console.warn("[Explorer] Failed to fetch live address data", error);
return [];
}
}
export async function fetchReceipts(): Promise<ReceiptSummary[]> {
if (getDataMode() === "mock") {
const data = await fetchMock<ReceiptListResponse>("receipts");
return data.items;
}
try {
const response = await fetch(`${CONFIG.apiBaseUrl}/v1/receipts`);
if (!response.ok) {
throw new Error(`Failed to fetch receipts: ${response.status}`);
}
const data = (await response.json()) as ReceiptListResponse;
return data.items;
} catch (error) {
console.warn("[Explorer] Failed to fetch live receipt data", error);
return [];
}
}
async function fetchMock<T>(resource: string): Promise<T> {
const url = `${CONFIG.mockBasePath}/${resource}.json`;
try {
const response = await fetch(url);
if (!response.ok) {
throw new Error(`Request failed with status ${response.status}`);
}
return (await response.json()) as T;
} catch (error) {
console.warn(`[Explorer] Failed to fetch mock data from ${url}`, error);
return [] as unknown as T;
}
}

View File

@@ -0,0 +1,2 @@
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });

View File

@@ -0,0 +1,57 @@
export interface BlockSummary {
height: number;
hash: string;
timestamp: string;
txCount: number;
proposer: string;
}
export interface BlockListResponse {
items: BlockSummary[];
next_offset?: number | string | null;
}
export interface TransactionSummary {
hash: string;
block: number | string;
from: string;
to: string | null;
value: string;
status: string;
}
export interface TransactionListResponse {
items: TransactionSummary[];
next_offset?: number | string | null;
}
export interface AddressSummary {
address: string;
balance: string;
txCount: number;
lastActive: string;
recentTransactions?: string[];
}
export interface AddressDetailResponse extends AddressSummary {}
export interface AddressListResponse {
items: AddressSummary[];
next_offset?: number | string | null;
}
export interface ReceiptSummary {
receiptId: string;
miner: string;
coordinator: string;
issuedAt: string;
status: string;
payload?: {
minerSignature?: string;
coordinatorSignature?: string;
};
}
export interface ReceiptListResponse {
jobId: string;
items: ReceiptSummary[];
}