Thursday, July 21, 2022
HomeWordPress DevelopmentPrisma - Subsequent-generation ORM - DEV Neighborhood

Prisma – Subsequent-generation ORM – DEV Neighborhood


Requisitos:

  • Conocimiento básico en API REST
  • Conocimiento básico de SQL
  • NODE
  • NPM consumer de su preferencia.

En este tutorial veremos como hacer un API REST TODO App con TypeScript, Prisma y esbuild.

⚠️ No necesitas previo conocimiento de ninguna de estas tecnologías.



Que es Prisma?

Es un ORM de próxima generación con soporte para TypeScript con unas herramientas para facilitarnos la vida:

  • Prisma Consumer: Generador de consultas de tipo seguro y autogenerado que se adapta a sus datos.
  • Migraciones: Herramienta para crear y gestionar las migraciones de tu base de datos.
  • Prisma Studio: GUI consumer para tu base de datos.

👀 Para que no haya pierde en donde va cada cosa o si solo vienes a echar un vistazo rápido; te dejo el código en este repo.



Empezemos 🧑‍💻

Si estas usando VSCode puedes usar el plugin oficial de Prisma.



Instalemos nuestras dependencias

# yarn
$ yarn add -D prisma esbuild @varieties/node
$ yarn add @prisma/consumer

# npm
$ npm i -D prisma esbuild @varieties/node
$ npm i @prisma/consumer 
Enter fullscreen mode

Exit fullscreen mode



levantemos nuestro entorno de trabajo

# yarn
$ yarn prisma init

# npm
$ npx prisma init
Enter fullscreen mode

Exit fullscreen mode

Actualicemos nuestras variables de entorno. Para este tutorial practico usaremos SQLite pero puede revisar las que soporta Prisma.

# .env
DATABASE_URL="file:./database.sqlite"
Enter fullscreen mode

Exit fullscreen mode

Actualicemos nuestro schema.prisma

// prisma/schema.prisma
generator consumer {
  supplier = "prisma-client-js"
}

datasource db {
  supplier = "sqlite" // <- usemos SQLite
  url      = env("DATABASE_URL")
}
Enter fullscreen mode

Exit fullscreen mode

Creamos nuestro builder con esbuild para trabajar con TypeScript.

Si no sabes como funciona esto igual luego deberías darte tiempo para leer mi weblog esbuild – Desarrollo sin dolor

// esbuild.dev.js
import { spawn } from 'child_process'
import esbuild from 'esbuild'

let server;

const startServer = (message) => {
    if (server) server.kill('SIGINT')
    server = spawn('node', ['./dist/index.mjs'], { stdio: 'inherit' })
    console.log(`n${message}n`)
}

esbuild.construct({
    entryPoints: ['./src/index.ts'],
    watch: { onRebuild: (err) => !err && startServer('Rebuilded') },
    bundle: true,
    minify: true,
    platform: 'node',
    format: 'esm',
    goal: ['esnext'],
    exterior: ['/node_modules/*'],
    outfile: './dist/index.mjs',
})
    .then(() => startServer('Performed 🚀'))
    .catch(() => course of.exit(1))
Enter fullscreen mode

Exit fullscreen mode

Actualicemos nuestro package deal.json.

// package deal.json
{
  "identify": "prisma",
  "model": "1.0.0",
  "predominant": "index.js",
  "writer": "Ushieru Kokoran (https://ushieru.com/)",
  "license": "MIT",
  "kind": "module", // <- soporte a mjs
  "scripts": {
    "dev": "node esbuild.dev.js" // <- script para desarrollo
  },
  "dependencies": {
    "@prisma/consumer": "^4.1.0",
    "@varieties/koa": "^2.13.5",
    "@varieties/node": "^18.0.6",
    "koa": "^2.13.4"
  },
  "devDependencies": {
    "esbuild": "^0.14.49",
    "prisma": "^4.1.0"
  }
}
Enter fullscreen mode

Exit fullscreen mode

Creemos el modelo de nuestras tareas. Si estas usando otra DB que no sea SQLite puede usar enums en el standing. (defining-enums)

generator consumer {
  supplier = "prisma-client-js"
}

datasource db {
  supplier = "sqlite"
  url      = env("DATABASE_URL")
}

mannequin Activity {
  id          String   @id @default(cuid())
  identify        String
  description String
  standing      String   @default("OPEN") // OPEN, IN_PROGRESS, DONE
  createdAt   DateTime @default(now())
  updatedAt   DateTime @updatedAt
}
Enter fullscreen mode

Exit fullscreen mode

Ahora debemos cargar nuestros modelos a la base de datos.

# yarn
$ yarn prisma db push

# npm
$ npx prisma db push
Enter fullscreen mode

Exit fullscreen mode

terminal prisma 1

Generate Prisma Consumer…? Asi es, prisma genera los varieties de los objetos a partir de las tablas de la base de datos al vuelo. Es posible que VSCode no se de cuenta de este cambio así que hay que reiniciar el server de TypeScript.
Para eso abrimos la paleta de comandos y ejecutamos: TypeScript: Reset TS server.

Restet TS server

Creemos un cliente de prisma para tener acceso a nuestro ORM y su respectivo CRUD (Create, Learn, Replace, Delete).

// src/crudTask.ts
import { PrismaClient } from '@prisma/consumer';

const prisma = new PrismaClient();

// Tenemos acceso a nuestro modelo activity que es un mapeo exacto
// a nuestra tabla en la base de datos. Y con TypeScript tenemos
// el autocompletado para no cometer errores.

export const createTask = async (identify: string, description: string) => {
    return await prisma.activity.create({
        knowledge: { identify, description }
    })
}

export const readTasks = async () => {
    return await prisma.activity.findMany()
}

export const readOneTask = async (id: string) => {
    return await prisma.activity.findUnique({ the place: { id } })
}

export const updateTask = async (id: string, identify: string, description: string, standing: 'OPEN' | 'IN_PROGRESS' | 'DONE') => {
    return await prisma.activity.replace({
        knowledge: { identify, description, standing },
        the place: { id }
    })
}

export const deleteTask = async (id: string) => {
    return await prisma.activity.delete({ the place: { id } })
}
Enter fullscreen mode

Exit fullscreen mode

Y ya lo tenemos 🎉🎉 Puedes envolverlo en cualquier backend que quieras en src/index.ts.

Si usas Subsequent.js debes hacer unos cambios en la declaracion del cliente. Te dejo la Doc

Te muestro este pequeño ejemplo con Koa:

// src/index.ts
import Koa from 'koa';
import Router from 'koa-router';
import koaBody from 'koa-body';
import { createTask, readTasks, readOneTask, updateTask, deleteTask } from './crudTask'

const app = new Koa();

app.use(koaBody({ jsonLimit: '5kb' }));

const router = new Router();

router.get('/', (ctx) => ctx.physique = 'Good day World');

router.submit('/duties', async (ctx) =>
    ctx.physique = await createTask(ctx.request.physique.identify, ctx.request.physique.description));

router.get('/duties', async (ctx) =>
    ctx.physique = await readTasks());

router.get('/duties/:id', async (ctx) =>
    ctx.physique = await readOneTask(ctx.params.id));

router.put('/duties/:id', async (ctx) =>
    ctx.physique = await updateTask(
        ctx.params.id,
        ctx.request.physique.identify,
        ctx.request.physique.description,
        ctx.request.physique.standing
    ));

router.delete('/duties/:id', async (ctx) =>
    ctx.physique = await deleteTask(ctx.params.id));

app.use(router.routes());

app.hear(3000);
Enter fullscreen mode

Exit fullscreen mode

Y ya lo tenemos, vemos un ejemplo con Relaciones? cuéntame en los comentarios.

Y Joyful Hacking 🎉👨‍💻

RELATED ARTICLES

LEAVE A REPLY

Please enter your comment!
Please enter your name here

- Advertisment -
Google search engine

Most Popular

Recent Comments