#289: Add init command for easy installation
This commit is contained in:
parent
dd9e039649
commit
75333d2659
3
public/.gitignore
vendored
3
public/.gitignore
vendored
@ -1,2 +1,3 @@
|
||||
**
|
||||
!.gitignore
|
||||
**
|
||||
!assets.zip
|
||||
|
BIN
public/assets.zip
Normal file
BIN
public/assets.zip
Normal file
Binary file not shown.
277
src/commands/init.ts
Normal file
277
src/commands/init.ts
Normal file
@ -0,0 +1,277 @@
|
||||
import { Server } from 'socket.io'
|
||||
import prisma from '../utilities/prisma'
|
||||
import fs from 'fs'
|
||||
import { getPublicPath, getRootPath } from '../utilities/storage'
|
||||
import sharp from 'sharp'
|
||||
import { CharacterGender, CharacterRace } from '@prisma/client'
|
||||
import bcrypt from 'bcryptjs'
|
||||
|
||||
/**
|
||||
* This script adds demo data to the database
|
||||
*/
|
||||
export default class InitCommand {
|
||||
constructor(private readonly io: Server) {}
|
||||
|
||||
public async execute(): Promise<void> {
|
||||
// Assets
|
||||
await this.importTiles()
|
||||
await this.importObjects()
|
||||
await this.createCharacterSprite()
|
||||
await this.createCharacterHair()
|
||||
|
||||
// Zone
|
||||
await this.createZone()
|
||||
|
||||
// User
|
||||
await this.createUser()
|
||||
|
||||
// Stop process
|
||||
process.exit(0)
|
||||
}
|
||||
|
||||
// Read tiles from getPublicPath('tiles') and create them in the database
|
||||
private async importTiles(): Promise<void> {
|
||||
for (const tile of fs.readdirSync(getPublicPath('tiles'))) {
|
||||
await prisma.tile.create({
|
||||
data: {
|
||||
id: tile.split('.')[0],
|
||||
name: 'New tile'
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
// Read objects from getPublicPath('objects') and create them in the database
|
||||
private async importObjects(): Promise<void> {
|
||||
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)
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
private async createCharacterSprite(): Promise<void> {
|
||||
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
|
||||
},
|
||||
]
|
||||
}
|
||||
}
|
||||
})
|
||||
|
||||
await prisma.characterType.create({
|
||||
data: {
|
||||
id: 1,
|
||||
name: 'New character type',
|
||||
gender: CharacterGender.MALE,
|
||||
race: CharacterRace.HUMAN,
|
||||
spriteId: '023d1e9d-f57f-4faa-8412-86c07107cf85'
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
private async createCharacterHair(): Promise<void> {
|
||||
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
|
||||
}
|
||||
]
|
||||
}
|
||||
}
|
||||
})
|
||||
|
||||
await prisma.characterHair.create({
|
||||
data: {
|
||||
id: 1,
|
||||
name: 'Hair 1',
|
||||
gender: CharacterGender.MALE,
|
||||
isSelectable: true,
|
||||
spriteId: '922ee95f-1500-49c0-8ead-f8cc46dad136'
|
||||
}
|
||||
})
|
||||
|
||||
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
|
||||
}
|
||||
]
|
||||
}
|
||||
}
|
||||
})
|
||||
|
||||
await prisma.characterHair.create({
|
||||
data: {
|
||||
id: 2,
|
||||
name: 'Hair 2',
|
||||
gender: CharacterGender.MALE,
|
||||
isSelectable: true,
|
||||
spriteId: 'a53811e2-3e85-4db3-875e-f2f989ca800d'
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
private async createZone(): Promise<void> {
|
||||
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
|
||||
}
|
||||
]
|
||||
},
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
private async createUser(): Promise<void> {
|
||||
await prisma.user.create({
|
||||
data: {
|
||||
id: 1,
|
||||
username: 'root',
|
||||
email: 'local@host',
|
||||
password: await bcrypt.hash('password', 10),
|
||||
online: false
|
||||
}
|
||||
})
|
||||
|
||||
await prisma.character.create({
|
||||
data: {
|
||||
id: 1,
|
||||
userId: 1,
|
||||
name: 'root',
|
||||
role: 'gm',
|
||||
characterTypeId: 1,
|
||||
characterHairId: 1
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
@ -9,11 +9,6 @@ class ZoneManager {
|
||||
private readonly zones = new Map<number, LoadedZone>()
|
||||
|
||||
public async boot(): Promise<void> {
|
||||
// Create first zone if it doesn't exist
|
||||
if (!(await ZoneRepository.getById(1))) {
|
||||
await new ZoneService().createDemoZone()
|
||||
}
|
||||
|
||||
const zones = await ZoneRepository.getAll()
|
||||
await Promise.all(zones.map((zone) => this.loadZone(zone)))
|
||||
|
||||
|
@ -2,38 +2,7 @@ import prisma from '../utilities/prisma'
|
||||
import { gameLogger } from '../utilities/logger'
|
||||
|
||||
class ZoneService {
|
||||
async createDemoZone(): Promise<boolean> {
|
||||
try {
|
||||
const tiles = [
|
||||
['blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile'],
|
||||
['blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile'],
|
||||
['blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile'],
|
||||
['blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile'],
|
||||
['blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile'],
|
||||
['blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile'],
|
||||
['blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile'],
|
||||
['blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile'],
|
||||
['blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile'],
|
||||
['blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile', 'blank_tile']
|
||||
]
|
||||
|
||||
await prisma.zone.create({
|
||||
data: {
|
||||
name: 'Demo zone',
|
||||
width: 10,
|
||||
height: 10,
|
||||
tiles
|
||||
}
|
||||
})
|
||||
|
||||
gameLogger.info('Demo zone created.')
|
||||
|
||||
return true
|
||||
} catch (error: any) {
|
||||
gameLogger.error(`Failed to create demo zone: ${error instanceof Error ? error.message : String(error)}`)
|
||||
return false
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
export default ZoneService
|
||||
|
Loading…
x
Reference in New Issue
Block a user