This commit is contained in:
2026-03-05 17:37:30 +01:00
commit 37e41a0130
14 changed files with 2366 additions and 0 deletions

6
.gitignore vendored Normal file
View File

@@ -0,0 +1,6 @@
node_modules/
dist/
.env*
*.js
.json

View File

@@ -0,0 +1,61 @@
CREATE EXTENSION pgcrypto; -- para los random bytes
-- 1. Función de generacion de uuidv7 copiada porque no esta en postgre 19
CREATE OR REPLACE FUNCTION
uuid_generate_v7()
RETURNS
uuid
LANGUAGE
plpgsql
PARALLEL SAFE
AS $$
DECLARE
-- The current UNIX timestamp in milliseconds
unix_time_ms CONSTANT bytea NOT NULL DEFAULT substring(int8send((extract(epoch FROM clock_timestamp()) * 1000)::bigint) from 3);
-- The buffer used to create the UUID, starting with the UNIX timestamp and followed by random bytes
buffer bytea NOT NULL DEFAULT unix_time_ms || gen_random_bytes(10);
BEGIN
-- Set most significant 4 bits of 7th byte to 7 (for UUID v7), keeping the last 4 bits unchanged
buffer = set_byte(buffer, 6, (b'0111' || get_byte(buffer, 6)::bit(4))::bit(8)::int);
-- Set most significant 2 bits of 9th byte to 2 (the UUID variant specified in RFC 4122), keeping the last 6 bits unchanged
buffer = set_byte(buffer, 8, (b'10' || get_byte(buffer, 8)::bit(6))::bit(8)::int);
RETURN encode(buffer, 'hex');
END
$$
;
-- 2. Tabla de Tarjetas
-- NUNCA se guarda el numero completo (PAN). Solo los últimos 4 dígitos.
CREATE TABLE payment_cards (
card_id UUID PRIMARY KEY DEFAULT uuid_generate_v7(),
user_id UUID REFERENCES users(user_id),
pan_last_four VARCHAR(4) NOT NULL,
card_holder_name VARCHAR(100) NOT NULL,
status VARCHAR(20) DEFAULT 'PENDING_ACTIVATION', -- PENDING, ACTIVE, BLOCKED, EXPIRED
created_at TIMESTAMPTZ DEFAULT NOW(),
activated_at TIMESTAMPTZ
);
-- 3. Tabla de Códigos de Activación
CREATE TABLE activation_codes (
code_id UUID PRIMARY KEY DEFAULT uuid_generate_v7(),
card_id UUID REFERENCES payment_cards(card_id), -- Una tarjeta, maximo un un código activo borrar o solo con expires_at?
code_hash TEXT NOT NULL, -- Guardar el código hasheado, el original se imprime y se manda
is_used BOOLEAN DEFAULT FALSE,
expires_at TIMESTAMPTZ NOT NULL,
created_at TIMESTAMPTZ DEFAULT NOW()
);
-- 4. Registro de Auditoría y Dispositivos (Log de Uso)
CREATE TABLE activation_logs (
log_id UUID PRIMARY KEY DEFAULT uuid_generate_v7(),
card_id UUID REFERENCES payment_cards(card_id),
code_id UUID REFERENCES activation_codes(code_id),
action_type VARCHAR(50) NOT NULL, -- TODO: CREAR ENUM'GENERATED', 'ATTEMPT_FAILED', 'ACTIVATED'
ip_address INET,
device_info JSONB, -- Almacena user-agent, modelo, SO, etc.
geo_location VARCHAR(100), -- Opcional: Ciudad/País derivado de IP
created_at TIMESTAMPTZ DEFAULT NOW()
);

View File

@@ -0,0 +1,22 @@
# Stage base para coordinar las fases de build y ejecucion
FROM node:22-alpine AS base
# Hace falta para la herramienta de migraciones, cuando se publique se
# sustituira por el paquete de npm
RUN apk --no-cache add git=latest
WORKDIR /usr/local/app
COPY ./package.json ./package.lock ./
# copia el codigo en general
COPY tsconfig*.json ./
COPY .env* ./
COPY ./.yarnrc.yml ./
COPY ./deployment/local/docker/start.sh ./
# Copiar el archivo de migrations? porque ahora no creo que se esté lanzando nada
COPY ./deployment/database/migrations ./deployment/database/migrations
RUN npm install --production && \
npm build && \
chmod +x start.sh
EXPOSE ${PORT}
ENTRYPOINT [ "./start.sh" ]

View File

@@ -0,0 +1,57 @@
name: sf-nfc-server
networks:
default:
driver: bridge
name: network-test # Tiene que coincidir con el compose objetivo
services:
sf-nfc-api:
container_name: sf-nfc-api
image: sf-nfc-api
build:
context: ./
dockerfile: deployment/local/docker/Dockerfile.local
args:
PORT: "${PORT:-3000}"
develop:
watch:
- path: ./src
action: sync
target: /usr/local/app/packages
- path: ./package.json
action: rebuild
ports:
- ${PORT}:${PORT}
env_file:
- .env
restart: unless-stopped
healthcheck:
test:
[
"CMD-SHELL",
'node -e "fetch(''http://localhost:'' + (process.env.PORT || 3000) + ''/health'').then(r => { if (!r.ok) process.exit(1) }).catch(() => process.exit(1))"',
]
interval: 10s
timeout: 5s
retries: 5
start_period: 15s
depends_on:
postgresql-nfc:
condition: service_healthy
postgresql-nfc:
container_name: postgresql-nfc
image: postgres:16.1
env_file:
- .env
ports:
- "${POSTGRES_PORT}:${POSTGRES_PORT}"
volumes:
- ./sql-data/:/var/lib/postgres/data
- ./deployment/database/init.sql:/docker-entrypoint-initdb.d/init.sql
healthcheck:
test: ["CMD-SHELL", "pg_isready -U $${POSTGRES_USER} -d $${POSTGRES_DB}"]
interval: 5s
retries: 5
start_period: 5s
timeout: 5s

View File

@@ -0,0 +1,3 @@
#!/bin/sh
echo "Lanzando migraciones e iniciando servidor"
npm run migrate && npm run start

9
docs/nfc-server/.gitignore vendored Normal file
View File

@@ -0,0 +1,9 @@
# Secrets
.env*
# Dependencies
node_modules
# OS files
.DS_Store
Thumbs.db

View File

@@ -0,0 +1,10 @@
opencollection: 1.0.0
info:
name: nfc-server
bundled: false
extensions:
bruno:
ignore:
- node_modules
- .git

2052
package-lock.json generated Normal file

File diff suppressed because it is too large Load Diff

29
package.json Normal file
View File

@@ -0,0 +1,29 @@
{
"name": "sf-nfc-server",
"version": "1.0.0",
"description": "",
"main": "index.js",
"type": "module",
"scripts": {
"test": "echo \"Error: no test specified\" && exit 1",
"dev": "tsx src/main.ts",
"build": "tsc",
"start": "node dist/main.js",
"migrate": "db-migrate -e .env -m ./deployment/database/migrations/ -t 99.0.0"
},
"author": "",
"license": "ISC",
"devDependencies": {
"@types/express": "^5.0.6",
"@types/node": "^25.3.3",
"@types/pg": "^8.18.0",
"ts-node": "^10.9.2",
"typescript": "^5.9.3"
},
"dependencies": {
"db-migrate": "^0.11.14",
"dotenv": "^17.3.1",
"express": "^5.2.1",
"pg": "^8.20.0"
}
}

View File

@@ -0,0 +1,17 @@
import { type Request, type Response, type NextFunction } from 'express';
export interface AppError extends Error {
status?: number;
}
export const errorHandler = (
err: AppError,
req: Request,
res: Response,
next: NextFunction
) => {
console.error(err);
res.status(err.status || 500).json({
message: err.message || 'Internal Server Error',
});
};

14
src/config/env.config.ts Normal file
View File

@@ -0,0 +1,14 @@
process.loadEnvFile(".env")
export const env = {
PORT: Number(process.env.PORT),
HOST: process.env.HOST,
POSTGRES_HOST: process.env.POSTGRES_HOST,
POSTGRES_PORT: process.env.POSTGRES_PORT,
POSTGRES_USER: process.env.POSTGRES_USER,
POSTGRES_PASSWORD: process.env.POSTGRES_PASSWORD
}
console.log(env)

12
src/domain/nfcRegistry.ts Normal file
View File

@@ -0,0 +1,12 @@
export type nfcRegsitry = {
account_id: string,
account_number: string
}
export type activationCodes = {
id: number,
code: string,
account_id: string,
creation_date: string,
expiration_date: string
}

29
src/main.ts Normal file
View File

@@ -0,0 +1,29 @@
import express, { Router, type Request, type Response } from 'express';
import { errorHandler } from './aplication/middleware.js';
import { env } from './config/env.config.js';
import assert from 'assert';
const PORT = env.PORT
const HOSTNAME = env.HOST
assert(HOSTNAME != undefined)
const router = Router();
router.get("/health", (req: Request, res: Response) => {
res.json({ ok: true })
})
const app = express();
app.use(express.json());
// Routes
app.use('/', router);
// Global error handler (should be after routes)
app.use(errorHandler);
app.listen(PORT, HOSTNAME, (n) => {
console.log(`[o] Servidor escuchando en ${HOSTNAME}:${PORT}`)
})

45
tsconfig.json Normal file
View File

@@ -0,0 +1,45 @@
{
// Visit https://aka.ms/tsconfig to read more about this file
"compilerOptions": {
// File Layout
"baseUrl": "./",
"rootDir": "./src",
"outDir": "./dist",
// Environment Settings
// See also https://aka.ms/tsconfig/module
"module": "nodenext",
"target": "esnext",
"moduleResolution": "nodenext",
// For nodejs:
"lib": [
"esnext"
],
"types": [
"node"
],
// and npm install -D @types/node
// Other Outputs
"sourceMap": true,
"declaration": true,
"declarationMap": true,
// Stricter Typechecking Options
"noUncheckedIndexedAccess": true,
"exactOptionalPropertyTypes": true,
// Style Options
// "noImplicitReturns": true,
// "noImplicitOverride": true,
// "noUnusedLocals": true,
// "noUnusedParameters": true,
// "noFallthroughCasesInSwitch": true,
// "noPropertyAccessFromIndexSignature": true,
// Recommended Options
"strict": true,
"jsx": "react-jsx",
"verbatimModuleSyntax": true,
"isolatedModules": true,
"noUncheckedSideEffectImports": true,
"moduleDetection": "force",
"skipLibCheck": true,
"esModuleInterop": true
}
}