diff --git a/README.md b/README.md new file mode 100644 index 0000000..4657d72 --- /dev/null +++ b/README.md @@ -0,0 +1,40 @@ +# Noxious game server + +This is the server for the Noxious game. + +## Installation + +1. Clone the repository +2. Install dependencies with `npm install` +3. Copy the `.env.example` file to `.env` and fill in the required variables +4. Run the server with `npm run dev` + +## Commands + +### `npm run dev` + +Starts the server in development mode. + +### `npm run build` + +Builds the server for production. + +### `npm run format` + +Formats the code using Prettier. + +## MikroORM + +MikroORM is used as the ORM for the server. + +### Create init. migrations + +Run `npx mikro-orm migration:create --initial` to create a new initial migration. + +### Create migrations + +Run `npx mikro-orm migration:create` to create a new migration. + +### Apply migrations + +Run `npx mikro-orm migration:up` to apply all pending migrations. \ No newline at end of file diff --git a/migrations/Migration20241225124810.ts b/migrations/Migration20241225180201.ts similarity index 99% rename from migrations/Migration20241225124810.ts rename to migrations/Migration20241225180201.ts index 0dc7132..875aac3 100644 --- a/migrations/Migration20241225124810.ts +++ b/migrations/Migration20241225180201.ts @@ -1,6 +1,6 @@ import { Migration } from '@mikro-orm/migrations'; -export class Migration20241225124810 extends Migration { +export class Migration20241225180201 extends Migration { override async up(): Promise { this.addSql(`create table \`map_object\` (\`id\` varchar(255) not null, \`name\` varchar(255) not null, \`tags\` json null, \`origin_x\` int not null default 0, \`origin_y\` int not null default 0, \`is_animated\` tinyint(1) not null default false, \`frame_rate\` int not null default 0, \`frame_width\` int not null default 0, \`frame_height\` int not null default 0, \`created_at\` datetime not null, \`updated_at\` datetime not null, primary key (\`id\`)) default character set utf8mb4 engine = InnoDB;`); diff --git a/src/commands/init.ts b/src/commands/init.ts index 85b0e20..6b4ac9c 100644 --- a/src/commands/init.ts +++ b/src/commands/init.ts @@ -1,14 +1,20 @@ import { Server } from 'socket.io' -import prisma from '#application/prisma' import fs from 'fs' -import { getPublicPath, getRootPath } from '#application/storage' +import { getPublicPath } from '#application/storage' import sharp from 'sharp' -import { CharacterEquipmentSlotType, CharacterGender, CharacterRace } from '@prisma/client' import bcrypt from 'bcryptjs' +import { Tile } from '#entities/tile' +import { MapObject } from '#entities/mapObject' +import { SpriteAction } from '#entities/spriteAction' +import { Sprite } from '#entities/sprite' +import { Zone } from '#entities/zone' +import { ZoneEffect } from '#entities/zoneEffect' +import { User } from '#entities/user' +import { Character } from '#entities/character' +import CharacterTypeRepository from '#repositories/characterTypeRepository' +import CharacterHairRepository from '#repositories/characterHairRepository' +import ZoneRepository from '#repositories/zoneRepository' -/** - * This script adds demo data to the database - */ export default class InitCommand { constructor(private readonly io: Server) {} @@ -18,7 +24,7 @@ export default class InitCommand { await this.importObjects() await this.createCharacterSprite() await this.createCharacterHair() - await this.createCharacterEquipment() + // await this.createCharacterEquipment() // Zone await this.createZone() @@ -30,296 +36,193 @@ export default class InitCommand { process.exit(0) } - // Read tiles from getPublicPath('tiles') and create them in the database private async importTiles(): Promise { for (const tile of fs.readdirSync(getPublicPath('tiles'))) { - await prisma.tile.create({ - data: { - id: tile.split('.')[0], - name: 'New tile' - } - }) + const newTile = new Tile() + newTile.id = tile.split('.')[0] as any + newTile.name = 'New tile' + await newTile.save() } } - // Read objects from getPublicPath('objects') and create them in the database private async importObjects(): Promise { for (const object of fs.readdirSync(getPublicPath('objects'))) { - await prisma.object.create({ - data: { - id: object.split('.')[0], - name: 'New object', - // Use sharp to get the dimensions of the image - frameHeight: await sharp(getPublicPath('objects', object)) - .metadata() - .then((metadata) => metadata.height), - frameWidth: await sharp(getPublicPath('objects', object)) - .metadata() - .then((metadata) => metadata.width) - } - }) + const newMapObject = new MapObject() + newMapObject.id = object.split('.')[0] as any + newMapObject.name = 'New object' + + newMapObject.frameWidth = await sharp(getPublicPath('objects', object)) + .metadata() + .then((metadata) => metadata.height) ?? 0 + + newMapObject.frameHeight = await sharp(getPublicPath('objects', object)) + .metadata() + .then((metadata) => metadata.width) ?? 0 + + await newMapObject.save() } } private async createCharacterSprite(): Promise { - await prisma.sprite.create({ - data: { - id: '023d1e9d-f57f-4faa-8412-86c07107cf85', - name: 'Character', - spriteActions: { - create: [ - { - action: 'idle_right_down', - sprites: [ - '' - ], - originX: 0, - originY: 0, - isAnimated: false, - isLooping: false, - frameWidth: 64, - frameHeight: 94, - frameRate: 0 - }, - { - action: 'idle_left_up', - sprites: [ - '' - ], - originX: 0, - originY: 0, - isAnimated: false, - isLooping: false, - frameWidth: 64, - frameHeight: 94, - frameRate: 0 - }, - { - action: 'walk_right_down', - sprites: [ - '', - '', - '', - '' - ], - originX: 0, - originY: 0, - isAnimated: true, - isLooping: false, - frameWidth: 64, - frameHeight: 94, - frameRate: 7 - }, - { - action: 'walk_left_up', - sprites: [ - '', - '', - '', - '' - ], - originX: 0, - originY: 0, - isAnimated: true, - isLooping: false, - frameWidth: 64, - frameHeight: 94, - frameRate: 7 - } - ] - } - } - }) + const characterSprite = new Sprite() + characterSprite.id = '023d1e9d-f57f-4faa-8412-86c07107cf85' + characterSprite.name = 'Character' - await prisma.characterType.create({ - data: { - id: 1, - name: 'New character type', - gender: CharacterGender.MALE, - race: CharacterRace.HUMAN, - spriteId: '023d1e9d-f57f-4faa-8412-86c07107cf85' - } - }) + const idleRightDownAction = new SpriteAction() + idleRightDownAction.action = 'idle_right_down' + idleRightDownAction.sprites = ['data:image/png;base64,...'] // Base64 sprite data + idleRightDownAction.originX = 0 + idleRightDownAction.originY = 0 + idleRightDownAction.isAnimated = false + idleRightDownAction.isLooping = false + idleRightDownAction.frameWidth = 64 + idleRightDownAction.frameHeight = 94 + idleRightDownAction.frameRate = 0 + idleRightDownAction.sprite = characterSprite + await idleRightDownAction.save() + + const idleLeftUpAction = new SpriteAction() + idleLeftUpAction.action = 'idle_left_up' + idleLeftUpAction.sprites = ['data:image/png;base64,...'] // Base64 sprite data + idleLeftUpAction.originX = 0 + idleLeftUpAction.originY = 0 + idleLeftUpAction.isAnimated = false + idleLeftUpAction.isLooping = false + idleLeftUpAction.frameWidth = 64 + idleLeftUpAction.frameHeight = 94 + idleLeftUpAction.frameRate = 0 + idleLeftUpAction.sprite = characterSprite + await idleLeftUpAction.save() + + const walkRightDownAction = new SpriteAction() + walkRightDownAction.action = 'walk_right_down' + walkRightDownAction.sprites = ['data:image/png;base64,...'] // Base64 sprite data array + walkRightDownAction.originX = 0 + walkRightDownAction.originY = 0 + walkRightDownAction.isAnimated = true + walkRightDownAction.isLooping = false + walkRightDownAction.frameWidth = 64 + walkRightDownAction.frameHeight = 94 + walkRightDownAction.frameRate = 7 + walkRightDownAction.sprite = characterSprite + await walkRightDownAction.save() + + const walkLeftUpAction = new SpriteAction() + walkLeftUpAction.action = 'walk_left_up' + walkLeftUpAction.sprites = ['data:image/png;base64,...'] // Base64 sprite data array + walkLeftUpAction.originX = 0 + walkLeftUpAction.originY = 0 + walkLeftUpAction.isAnimated = true + walkLeftUpAction.isLooping = false + walkLeftUpAction.frameWidth = 64 + walkLeftUpAction.frameHeight = 94 + walkLeftUpAction.frameRate = 7 + walkLeftUpAction.sprite = characterSprite + await walkLeftUpAction.save() + + await characterSprite.save() } private async createCharacterHair(): Promise { - await prisma.sprite.create({ - data: { - id: '922ee95f-1500-49c0-8ead-f8cc46dad136', - name: 'Hair 1', - spriteActions: { - create: [ - { - action: 'front', - sprites: [ - '' - ], - originX: 0.5, - originY: 5.34, - isAnimated: false, - isLooping: false, - frameWidth: 64, - frameHeight: 18, - frameRate: 0 - }, - { - action: 'back', - sprites: [ - '' - ], - originX: 0.5, - originY: 4.34, - isAnimated: false, - isLooping: false, - frameWidth: 64, - frameHeight: 22, - frameRate: 0 - } - ] - } - } - }) + const hairSprite = new Sprite() + hairSprite.id = '922ee95f-1500-49c0-8ead-f8cc46dad136' + hairSprite.name = 'Hair 1' - await prisma.characterHair.create({ - data: { - id: 1, - name: 'Hair 1', - gender: CharacterGender.MALE, - isSelectable: true, - spriteId: '922ee95f-1500-49c0-8ead-f8cc46dad136' - } - }) + const frontAction = new SpriteAction() + frontAction.action = 'front' + frontAction.sprites = ['data:image/png;base64,...'] // Base64 sprite data + frontAction.originX = 0.5 + frontAction.originY = 5.34 + frontAction.isAnimated = false + frontAction.isLooping = false + frontAction.frameWidth = 64 + frontAction.frameHeight = 18 + frontAction.frameRate = 0 + frontAction.sprite = hairSprite + await frontAction.save() - await prisma.sprite.create({ - data: { - id: 'a53811e2-3e85-4db3-875e-f2f989ca800d', - name: 'Hair 2', - spriteActions: { - create: [ - { - action: 'front', - sprites: [ - '' - ], - originX: 0.5, - originY: 5.34, - isAnimated: false, - isLooping: false, - frameWidth: 64, - frameHeight: 18, - frameRate: 0 - }, - { - action: 'back', - sprites: [ - '' - ], - originX: 0.5, - originY: 4.34, - isAnimated: false, - isLooping: false, - frameWidth: 64, - frameHeight: 22, - frameRate: 0 - } - ] - } - } - }) + const backAction = new SpriteAction() + backAction.action = 'back' + backAction.sprites = ['data:image/png;base64,...'] // Base64 sprite data + backAction.originX = 0.5 + backAction.originY = 4.34 + backAction.isAnimated = false + backAction.isLooping = false + backAction.frameWidth = 64 + backAction.frameHeight = 22 + backAction.frameRate = 0 + backAction.sprite = hairSprite + await backAction.save() - await prisma.characterHair.create({ - data: { - id: 2, - name: 'Hair 2', - gender: CharacterGender.MALE, - isSelectable: true, - spriteId: 'a53811e2-3e85-4db3-875e-f2f989ca800d' - } - }) + await hairSprite.save() } private async createCharacterEquipment(): Promise { - await prisma.sprite.create({ - data: { - id: '5b3932dd-0791-4bb7-bb1e-da9833c3cc50', - name: 'Male shirt', - spriteActions: { - create: [ - { - action: 'idle_right_down', - sprites: [ - '' - ] - }, - { - action: 'idle_left_up', - sprites: [ - '' - ] - }, - { - action: 'walking_right_down', - sprites: [ - '', - '', - '', - '' - ] - }, - { - action: 'walking_left_up', - sprites: [ - '', - '', - '', - '' - ] - } - ] - } - } - }) + const equipmentSprite = new Sprite() + equipmentSprite.id = '5b3932dd-0791-4bb7-bb1e-da9833c3cc50' + equipmentSprite.name = 'Male shirt' + + // Create actions similar to createCharacterSprite() + // with appropriate sprite data and parameters + const actions = [ + { + action: 'idle_right_down', + sprites: ['data:image/png;base64,...'], + originX: 0, + originY: 0, + isAnimated: false, + isLooping: false, + frameWidth: 64, + frameHeight: 94, + frameRate: 0 + }, + // Add other actions... + ] + + for (const actionData of actions) { + const action = new SpriteAction() + Object.assign(action, actionData) + action.sprite = equipmentSprite + await action.save() + } + + await equipmentSprite.save() } private async createZone(): Promise { - const zone = await prisma.zone.create({ - data: { - name: 'New zone', - width: 100, - height: 100, - tiles: Array.from({ length: 100 }, () => Array.from({ length: 100 }, () => 'a2fd8d6f-5042-437a-9c1e-c66b91ecc35b')), - zoneEffects: { - create: [ - { - effect: 'light', - strength: 100 - } - ] - } - } - }) + const zone = new Zone() + zone.name = 'New zone' + zone.width = 100 + zone.height = 100 + zone.tiles = Array.from({ length: 100 }, () => + Array.from({ length: 100 }, () => 'a2fd8d6f-5042-437a-9c1e-c66b91ecc35b') + ) + await zone.save() + + const effect = new ZoneEffect() + effect.effect = 'light' + effect.strength = 100 + effect.zone = zone + await effect.save() } private async createUser(): Promise { - await prisma.user.create({ - data: { - id: 1, - username: 'root', - email: 'local@host', - password: await bcrypt.hash('password', 10), - online: false - } - }) + const user = new User() + user.id = 1 + user.username = 'root' + user.email = 'local@host' + user.password = await bcrypt.hash('password', 10) + user.online = false + await user.save() - await prisma.character.create({ - data: { - id: 1, - userId: 1, - name: 'root', - role: 'gm', - characterTypeId: 1, - characterHairId: 1 - } - }) + const character = new Character() + character.id = 1 + character.user = user + character.name = 'root' + character.role = 'gm' + character.zone = await ZoneRepository.getFirst() ?? undefined + character.characterType = await CharacterTypeRepository.getFirst() ?? undefined + character.characterHair = await CharacterHairRepository.getFirst() ?? undefined + await character.save() } -} +} \ No newline at end of file diff --git a/src/entities/zoneEffect.ts b/src/entities/zoneEffect.ts index b9e4f27..d34e810 100644 --- a/src/entities/zoneEffect.ts +++ b/src/entities/zoneEffect.ts @@ -1,3 +1,4 @@ +import { randomUUID } from 'node:crypto' import { Entity, ManyToOne, PrimaryKey, Property } from '@mikro-orm/core' import { BaseEntity } from '#application/bases/baseEntity' import { Zone } from './zone' @@ -5,7 +6,7 @@ import { Zone } from './zone' @Entity() export class ZoneEffect extends BaseEntity { @PrimaryKey() - id!: string + id = randomUUID() @ManyToOne(() => Zone) zone!: Zone diff --git a/src/repositories/characterHairRepository.ts b/src/repositories/characterHairRepository.ts index 130363a..9f422bc 100644 --- a/src/repositories/characterHairRepository.ts +++ b/src/repositories/characterHairRepository.ts @@ -3,6 +3,16 @@ import { BaseRepository } from '#application/bases/baseRepository' import { CharacterHair } from '#entities/characterHair' class CharacterHairRepository extends BaseRepository { + async getFirst() { + try { + const repository = this.em.getRepository(CharacterHair) + return await repository.findOne({ id: { $exists: true } }) + } catch (error: any) { + appLogger.error(`Failed to get first character hair: ${error instanceof Error ? error.message : String(error)}`) + return null + } + } + async getAll() { try { const repository = this.em.getRepository(CharacterHair) diff --git a/src/repositories/characterTypeRepository.ts b/src/repositories/characterTypeRepository.ts index b51e5fa..fe0cd28 100644 --- a/src/repositories/characterTypeRepository.ts +++ b/src/repositories/characterTypeRepository.ts @@ -3,6 +3,16 @@ import { BaseRepository } from '#application/bases/baseRepository' import { CharacterType } from '#entities/characterType' class CharacterTypeRepository extends BaseRepository { + async getFirst() { + try { + const repository = this.em.getRepository(CharacterType) + return await repository.findOne({ id: { $exists: true } }) + } catch (error: any) { + appLogger.error(`Failed to get first character type: ${error instanceof Error ? error.message : String(error)}`) + return null + } + } + async getAll() { try { const repository = this.em.getRepository(CharacterType) diff --git a/src/repositories/zoneRepository.ts b/src/repositories/zoneRepository.ts index 1fc7a4f..575278b 100644 --- a/src/repositories/zoneRepository.ts +++ b/src/repositories/zoneRepository.ts @@ -5,6 +5,16 @@ import { ZoneObject } from '#entities/zoneObject' import { Zone } from '#entities/zone' class ZoneRepository extends BaseRepository { + async getFirst(): Promise { + try { + const repository = this.em.getRepository(Zone) + return await repository.findOne({ id: { $exists: true } }) + } catch (error: any) { + appLogger.error(`Failed to get first zone: ${error instanceof Error ? error.message : String(error)}`) + return null + } + } + async getAll(): Promise { try { const repository = this.em.getRepository(Zone) diff --git a/src/socketEvents/character/create.ts b/src/socketEvents/character/create.ts index c4b20f7..2f7799b 100644 --- a/src/socketEvents/character/create.ts +++ b/src/socketEvents/character/create.ts @@ -38,7 +38,9 @@ export default class CharacterCreateEvent { } const characterService = new CharacterService() - const character: Character = await characterService.create(data.name, user_id) + const character = await characterService.create(data.name, user_id) + + if (!character) return this.socket.emit('notification', { message: 'Failed to create character. Please try again (later).' }) characters = [...characters, character]