chore: remove sqlite
parent
9572d1a1c5
commit
692225cfff
|
@ -3,7 +3,7 @@
|
|||
"collection": "@nestjs/schematics",
|
||||
"sourceRoot": "src",
|
||||
"compilerOptions": {
|
||||
"assets": ["**/*.yml"],
|
||||
"assets": ["**/*.yml", "**/*.json"],
|
||||
"watchAssets": true,
|
||||
"deleteOutDir": true
|
||||
}
|
||||
|
|
|
@ -19,7 +19,6 @@
|
|||
"@nestjs/jwt": "^10.2.0",
|
||||
"@nestjs/passport": "^10.0.3",
|
||||
"@nestjs/platform-express": "^10.3.10",
|
||||
"@nestjs/typeorm": "^10.0.2",
|
||||
"@types/js-yaml": "^4.0.9",
|
||||
"bcryptjs": "^2.4.3",
|
||||
"class-transformer": "^0.5.1",
|
||||
|
@ -31,9 +30,7 @@
|
|||
"passport-jwt": "^4.0.1",
|
||||
"passport-local": "^1.0.0",
|
||||
"reflect-metadata": "^0.2.2",
|
||||
"rxjs": "^7.8.1",
|
||||
"sqlite3": "^5.1.7",
|
||||
"typeorm": "^0.3.20"
|
||||
"rxjs": "^7.8.1"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@nestjs/cli": "^10.4.0",
|
||||
|
|
|
@ -1,23 +1,16 @@
|
|||
import configuration from '@/config/index';
|
||||
import { Module } from '@nestjs/common';
|
||||
import { ConfigModule } from '@nestjs/config';
|
||||
import { TypeOrmModule } from '@nestjs/typeorm';
|
||||
import Joi from 'joi';
|
||||
|
||||
import { AuthModule } from './modules/auth/auth.module';
|
||||
import { DatabaseModule } from './modules/database/database.module';
|
||||
import { HealthModule } from './modules/health/health.module';
|
||||
import { MenuModule } from './modules/menu/menu.module';
|
||||
import { MockModule } from './modules/mock/mock.module';
|
||||
import { UsersModule } from './modules/users/users.module';
|
||||
|
||||
@Module({
|
||||
imports: [
|
||||
TypeOrmModule.forRoot({
|
||||
autoLoadEntities: true,
|
||||
database: 'data/db.sqlite',
|
||||
synchronize: true,
|
||||
type: 'sqlite',
|
||||
}),
|
||||
ConfigModule.forRoot({
|
||||
cache: true,
|
||||
isGlobal: true,
|
||||
|
@ -34,8 +27,8 @@ import { UsersModule } from './modules/users/users.module';
|
|||
HealthModule,
|
||||
AuthModule,
|
||||
UsersModule,
|
||||
DatabaseModule,
|
||||
MenuModule,
|
||||
MockModule,
|
||||
],
|
||||
})
|
||||
export class AppModule {}
|
||||
|
|
|
@ -1,33 +1,20 @@
|
|||
import { Column, Entity, PrimaryGeneratedColumn } from 'typeorm';
|
||||
|
||||
@Entity()
|
||||
class UserEntity {
|
||||
@PrimaryGeneratedColumn()
|
||||
id: number;
|
||||
/**
|
||||
* 密码
|
||||
*/
|
||||
@Column()
|
||||
password: string;
|
||||
/**
|
||||
* 真实姓名
|
||||
*/
|
||||
@Column()
|
||||
realName: string;
|
||||
/**
|
||||
* 角色
|
||||
*/
|
||||
@Column('text', {
|
||||
transformer: {
|
||||
from: (value: string) => JSON.parse(value),
|
||||
to: (value: string[]) => JSON.stringify(value),
|
||||
},
|
||||
})
|
||||
roles: string[];
|
||||
/**
|
||||
* 用户名
|
||||
*/
|
||||
@Column({ unique: true })
|
||||
username: string;
|
||||
}
|
||||
|
||||
|
|
|
@ -1,12 +0,0 @@
|
|||
import { UserEntity } from '@/models/entity/user.entity';
|
||||
import { Module } from '@nestjs/common';
|
||||
import { TypeOrmModule } from '@nestjs/typeorm';
|
||||
|
||||
import { UsersModule } from '../users/users.module';
|
||||
import { DatabaseService } from './database.service';
|
||||
|
||||
@Module({
|
||||
imports: [UsersModule, TypeOrmModule.forFeature([UserEntity])],
|
||||
providers: [DatabaseService],
|
||||
})
|
||||
export class DatabaseModule {}
|
|
@ -1,19 +0,0 @@
|
|||
import { Test, TestingModule } from '@nestjs/testing';
|
||||
|
||||
import { DatabaseService } from './database.service';
|
||||
|
||||
describe('databaseService', () => {
|
||||
let service: DatabaseService;
|
||||
|
||||
beforeEach(async () => {
|
||||
const module: TestingModule = await Test.createTestingModule({
|
||||
providers: [DatabaseService],
|
||||
}).compile();
|
||||
|
||||
service = module.get<DatabaseService>(DatabaseService);
|
||||
});
|
||||
|
||||
it('should be defined', () => {
|
||||
expect(service).toBeDefined();
|
||||
});
|
||||
});
|
|
@ -1,40 +0,0 @@
|
|||
import type { Repository } from 'typeorm';
|
||||
|
||||
import { UserEntity } from '@/models/entity/user.entity';
|
||||
import { Injectable, type OnModuleInit } from '@nestjs/common';
|
||||
import { InjectRepository } from '@nestjs/typeorm';
|
||||
|
||||
import { UsersService } from '../users/users.service';
|
||||
|
||||
@Injectable()
|
||||
export class DatabaseService implements OnModuleInit {
|
||||
constructor(
|
||||
@InjectRepository(UserEntity)
|
||||
private usersRepository: Repository<UserEntity>,
|
||||
private userService: UsersService,
|
||||
) {}
|
||||
async onModuleInit() {
|
||||
// data/db.sqlite会被git忽略,方式数据库文件被提交到git
|
||||
// 清空表,并初始化两条数据
|
||||
await this.usersRepository.clear();
|
||||
|
||||
await this.userService.create({
|
||||
id: 0,
|
||||
password: '123456',
|
||||
realName: 'Administrator',
|
||||
roles: ['admin'],
|
||||
username: 'vben',
|
||||
});
|
||||
|
||||
await this.userService.create({
|
||||
id: 1,
|
||||
password: '123456',
|
||||
realName: 'Jack',
|
||||
roles: ['user'],
|
||||
username: 'jack',
|
||||
});
|
||||
|
||||
const count = await this.usersRepository.count();
|
||||
console.log('Database has been initialized with seed data, count:', count);
|
||||
}
|
||||
}
|
|
@ -0,0 +1 @@
|
|||
{}
|
|
@ -0,0 +1,13 @@
|
|||
interface User {
|
||||
id: number;
|
||||
password: string;
|
||||
realName: string;
|
||||
roles: string[];
|
||||
username: string;
|
||||
}
|
||||
|
||||
interface MockDatabaseData {
|
||||
users: User[];
|
||||
}
|
||||
|
||||
export type { MockDatabaseData, User };
|
|
@ -0,0 +1,9 @@
|
|||
import { Module } from '@nestjs/common';
|
||||
|
||||
import { MockService } from './mock.service';
|
||||
|
||||
@Module({
|
||||
exports: [MockService],
|
||||
providers: [MockService],
|
||||
})
|
||||
export class MockModule {}
|
|
@ -0,0 +1,71 @@
|
|||
import type { MockDatabaseData } from './mock.interface';
|
||||
|
||||
import fs from 'node:fs';
|
||||
import path from 'node:path';
|
||||
|
||||
import { Injectable, type OnModuleInit } from '@nestjs/common';
|
||||
import bcrypt from 'bcryptjs';
|
||||
|
||||
@Injectable()
|
||||
export class MockService implements OnModuleInit {
|
||||
private data: MockDatabaseData;
|
||||
private readonly filePath: string;
|
||||
|
||||
constructor() {
|
||||
this.filePath = path.join(__dirname, '.', 'mock-db.json');
|
||||
this.loadData();
|
||||
}
|
||||
|
||||
private loadData() {
|
||||
const fileData = fs.readFileSync(this.filePath, 'utf8');
|
||||
this.data = JSON.parse(fileData);
|
||||
}
|
||||
|
||||
private saveData() {
|
||||
fs.writeFileSync(this.filePath, JSON.stringify(this.data, null, 2));
|
||||
}
|
||||
|
||||
addItem(collection: string, item: any) {
|
||||
this.data[collection].push(item);
|
||||
this.saveData();
|
||||
return item;
|
||||
}
|
||||
|
||||
clearCollection(collection: string) {
|
||||
this.data[collection] = [];
|
||||
this.saveData();
|
||||
return this.data[collection];
|
||||
}
|
||||
|
||||
findAll(collection: string) {
|
||||
return this.data[collection];
|
||||
}
|
||||
|
||||
findOneById(collection: string, id: number) {
|
||||
return this.data[collection].find((item) => item.id === id);
|
||||
}
|
||||
|
||||
async onModuleInit() {
|
||||
// 清空表,并初始化两条数据
|
||||
await this.clearCollection('users');
|
||||
|
||||
// 密码哈希
|
||||
const hashPassword = await bcrypt.hash('123456', 10);
|
||||
await this.addItem('users', {
|
||||
id: 0,
|
||||
password: hashPassword,
|
||||
realName: 'Vben',
|
||||
roles: ['admin'],
|
||||
username: 'vben',
|
||||
});
|
||||
await this.addItem('users', {
|
||||
id: 1,
|
||||
password: hashPassword,
|
||||
realName: 'Jack',
|
||||
roles: ['user'],
|
||||
username: 'jack',
|
||||
});
|
||||
const count = await this.findAll('users').length;
|
||||
console.log('Database has been initialized with seed data, count:', count);
|
||||
}
|
||||
}
|
|
@ -1,12 +1,11 @@
|
|||
import { UserEntity } from '@/models/entity/user.entity';
|
||||
import { Module } from '@nestjs/common';
|
||||
import { TypeOrmModule } from '@nestjs/typeorm';
|
||||
|
||||
import { MockModule } from '../mock/mock.module';
|
||||
import { UsersService } from './users.service';
|
||||
|
||||
@Module({
|
||||
exports: [UsersService],
|
||||
imports: [TypeOrmModule.forFeature([UserEntity])],
|
||||
imports: [MockModule],
|
||||
providers: [UsersService],
|
||||
})
|
||||
export class UsersModule {}
|
||||
|
|
|
@ -1,28 +1,18 @@
|
|||
import type { CreateUserDto } from '@/models/dto/user.dto';
|
||||
import type { Repository } from 'typeorm';
|
||||
|
||||
import { UserEntity } from '@/models/entity/user.entity';
|
||||
import { Injectable } from '@nestjs/common';
|
||||
import { InjectRepository } from '@nestjs/typeorm';
|
||||
import bcrypt from 'bcryptjs';
|
||||
|
||||
import { MockService } from '../mock/mock.service';
|
||||
|
||||
@Injectable()
|
||||
export class UsersService {
|
||||
constructor(
|
||||
@InjectRepository(UserEntity)
|
||||
private usersRepository: Repository<UserEntity>,
|
||||
) {}
|
||||
|
||||
async create(user: CreateUserDto): Promise<UserEntity> {
|
||||
user.password = await bcrypt.hash(user.password, 10); // 密码哈希
|
||||
return this.usersRepository.save(user);
|
||||
}
|
||||
constructor(private mockService: MockService) {}
|
||||
|
||||
/**
|
||||
* Find user by username
|
||||
* @param username
|
||||
*/
|
||||
async findOne(username: string): Promise<UserEntity | undefined> {
|
||||
return await this.usersRepository.findOne({ where: { username } });
|
||||
const allUsers = await this.mockService.findAll('users');
|
||||
return allUsers.find((user) => user.username === username);
|
||||
}
|
||||
}
|
||||
|
|
|
@ -38,50 +38,3 @@ const tokenTheme = computed(() => {
|
|||
</ConfigProvider>
|
||||
</GlobalProvider>
|
||||
</template>
|
||||
|
||||
<!-- <style>
|
||||
:root {
|
||||
--background: 0 0% 100%;
|
||||
--foreground: 240 10% 3.9%;
|
||||
--card: 0 0% 100%;
|
||||
--card-foreground: 240 10% 3.9%;
|
||||
--popover: 0 0% 100%;
|
||||
--popover-foreground: 240 10% 3.9%;
|
||||
--primary: 240 5.9% 10%;
|
||||
--primary-foreground: 0 0% 98%;
|
||||
--secondary: 240 4.8% 95.9%;
|
||||
--secondary-foreground: 240 5.9% 10%;
|
||||
--muted: 240 4.8% 95.9%;
|
||||
--muted-foreground: 240 3.8% 46.1%;
|
||||
--accent: 240 4.8% 95.9%;
|
||||
--accent-foreground: 240 5.9% 10%;
|
||||
--destructive: 0 84.2% 60.2%;
|
||||
--destructive-foreground: 0 0% 98%;
|
||||
--border: 240 5.9% 90%;
|
||||
--input: 240 5.9% 90%;
|
||||
--ring: 240 5.9% 10%;
|
||||
--radius: 0.25rem;
|
||||
}
|
||||
|
||||
.dark {
|
||||
--background: 240 10% 3.9%;
|
||||
--foreground: 0 0% 98%;
|
||||
--card: 240 10% 3.9%;
|
||||
--card-foreground: 0 0% 98%;
|
||||
--popover: 240 10% 3.9%;
|
||||
--popover-foreground: 0 0% 98%;
|
||||
--primary: 0 0% 98%;
|
||||
--primary-foreground: 240 5.9% 10%;
|
||||
--secondary: 240 3.7% 15.9%;
|
||||
--secondary-foreground: 0 0% 98%;
|
||||
--muted: 240 3.7% 15.9%;
|
||||
--muted-foreground: 240 5% 64.9%;
|
||||
--accent: 240 3.7% 15.9%;
|
||||
--accent-foreground: 0 0% 98%;
|
||||
--destructive: 0 62.8% 30.6%;
|
||||
--destructive-foreground: 0 0% 98%;
|
||||
--border: 240 3.7% 15.9%;
|
||||
--input: 240 3.7% 15.9%;
|
||||
--ring: 240 4.9% 83.9%;
|
||||
}
|
||||
</style> -->
|
||||
|
|
|
@ -5,7 +5,6 @@
|
|||
"words": [
|
||||
"clsx",
|
||||
"esno",
|
||||
"typeorm",
|
||||
"unref",
|
||||
"taze",
|
||||
"acmr",
|
||||
|
|
|
@ -12,6 +12,7 @@ const customConfig: Linter.FlatConfig[] = [
|
|||
rules: {
|
||||
'@typescript-eslint/no-extraneous-class': 'off',
|
||||
'no-console': 'off',
|
||||
'unicorn/prefer-module': 'off',
|
||||
},
|
||||
},
|
||||
];
|
||||
|
|
|
@ -115,6 +115,7 @@ function useEcharts(chartRef: Ref<EchartsUIType>) {
|
|||
});
|
||||
return {
|
||||
renderEcharts,
|
||||
resize,
|
||||
};
|
||||
}
|
||||
|
||||
|
|
531
pnpm-lock.yaml
531
pnpm-lock.yaml
File diff suppressed because it is too large
Load Diff
Loading…
Reference in New Issue