Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[Server] 0.4.6 배포 #199

Merged
merged 32 commits into from
Dec 7, 2023
Merged
Changes from 1 commit
Commits
Show all changes
32 commits
Select commit Hold shift + click to select a range
86520a7
[Server] 개발환경 세팅 (#4)
pminsung12 Nov 9, 2023
8fe9ba6
Server/feature/#13 (#25)
pminsung12 Nov 14, 2023
2d84f75
테스트용 유저 API 구현 (#30)
yangdongsuk Nov 15, 2023
2b6fb53
[Server] 유닛 테스트 환경 세팅 (#32)
yangdongsuk Nov 15, 2023
112d9ba
[Server] Users resource 이름 변경 (#34)
yangdongsuk Nov 15, 2023
375e785
feature: usersService 테스트 코드 작성 (#39)
yangdongsuk Nov 16, 2023
5b230ec
[Server] Folder entity 생성 및 crud 구현 (#42)
pminsung12 Nov 16, 2023
da9f6d3
feature: docker파일 수정 (#57)
yangdongsuk Nov 20, 2023
e4936cb
feat: private checklist entity 생성 및 crud 구현 (#61)
pminsung12 Nov 21, 2023
5cfdb02
feat: checklist 폴더 분리 & dto 빈문자열 검증 추가 (#66)
yangdongsuk Nov 22, 2023
de6b5f3
[Server] Winston으로 로그 관리 (#70)
pminsung12 Nov 23, 2023
f795888
feat: jwt access, refresh token 기반 인가 구현
yangdongsuk Nov 23, 2023
304ea79
[Server] shared-checklist 소켓 구현 (#78)
yangdongsuk Nov 23, 2023
ff9598d
[Server] apple oauth api 구현 (#86)
pminsung12 Nov 26, 2023
4616d27
[Server] access 토큰 재발급시 유저 정보 없는 버그 수정 (#83)
yangdongsuk Nov 26, 2023
7091fad
[Server] privateChecklist의 내용 저장 api 구현 (#88)
yangdongsuk Nov 27, 2023
7f88671
[Server] apple oauth 로그인 로직 수정 (#118)
pminsung12 Nov 27, 2023
fd3703c
[Server] Clova Studio api 구현 (#126)
pminsung12 Nov 29, 2023
ac8ab73
feat: AccessTokenGuard 구현 및 적용 (#129)
yangdongsuk Nov 29, 2023
70043ff
Server/feature/#128 (#139)
pminsung12 Nov 29, 2023
2b1eaf8
feat: 공유 체크리스트 API 및 소켓 작업 구현 (#140)
yangdongsuk Nov 30, 2023
283e6fa
🔐feat: 개발용 임시로 액세스,리프레시 토큰들 만료기한 일주일로 설정 (#142)
pminsung12 Nov 30, 2023
5e5fd4e
feat: 공유 체크리스트 아이템 권한 문제 및 uuid 문제 해결 (#146)
yangdongsuk Nov 30, 2023
adc3269
feat: 소켓 다중 서버 지원 (#159)
yangdongsuk Dec 4, 2023
347cb90
feat: 소켓 editing 이벤트 추가 (#164)
yangdongsuk Dec 4, 2023
620c06d
fix: 레디스 연결 수정 (#168)
yangdongsuk Dec 4, 2023
9240dde
[Server] object 형태가 들어오면 redis에 저장 안되는 문제 수정 (#171)
pminsung12 Dec 4, 2023
7ae37bb
feat: 웹소켓 히스토리 버그 수정 및 콘솔 로그 추가 (#175)
yangdongsuk Dec 4, 2023
d047b41
[Server] 로거 기능 확대 (#176)
pminsung12 Dec 4, 2023
55ade94
🐛fix: redis module password 부분 주석처리 (#180)
pminsung12 Dec 4, 2023
0014655
[Server] 피드화면 api 구현 (#196)
pminsung12 Dec 6, 2023
5c2f20a
feat: 관리 페이지와 api 구현 (#197)
yangdongsuk Dec 7, 2023
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
feat: 관리 페이지와 api 구현 (#197)
* feat: admin resource 구현

* feat: 관리자 페이지 추가

* feat: cors 설정 추가

* feat: 임시로 관리자 페이지 권한 제거

* feat: admin sse api 구현

* refactor: redis sub 서비스 구현

* feat: 어드민 페이지 api 기능 추가

* feat: redis pub 서비스 추가

* refactor: admin controller에 redis service로 교체

* feat: log interceptor에 redis pub 추가

* refactor: channels const로 분리

* style: 불필요한 주석 제거

* feat: ws 로그 redis pub 추가

* feat: admin page 박스 누르면 펼치기/접기 기능 추가

* feat: admin 페이지 색깔 변경

* feat: 관리자 페이지 로그인 기능 구현
yangdongsuk authored Dec 7, 2023
commit 5c2f20ae15385be121fb7535b9a7291094853757
1 change: 1 addition & 0 deletions server/package.json
Original file line number Diff line number Diff line change
@@ -34,6 +34,7 @@
"axios": "^1.6.2",
"class-transformer": "^0.5.1",
"class-validator": "^0.14.0",
"cookie-parser": "^1.4.6",
"nest-winston": "^1.9.4",
"pg": "^8.11.3",
"redis": "^4.6.11",
15 changes: 13 additions & 2 deletions server/redis/redis.module.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
import { DynamicModule, Global, Module } from '@nestjs/common';
import { RedisClientType, createClient } from 'redis';
import { RedisService } from './redis.service';

/**
* RedisModule은 NestJS 애플리케이션에서 Redis 클라이언트를 설정하고 관리하는데 사용되는 모듈이다.
@@ -64,8 +65,18 @@ export class RedisModule {
*/
return {
module: RedisModule,
providers: [redisProvider, redisPubProvider, redisSubProvider],
exports: [redisProvider, redisPubProvider, redisSubProvider],
providers: [
redisProvider,
redisPubProvider,
redisSubProvider,
RedisService,
],
exports: [
redisProvider,
redisPubProvider,
redisSubProvider,
RedisService,
],
};
}
}
21 changes: 21 additions & 0 deletions server/redis/redis.service.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
import { Inject, Injectable } from '@nestjs/common';
import { RedisClientType } from 'redis';

@Injectable()
export class RedisService {
constructor(
@Inject('REDIS_SUB_CLIENT')
private readonly redisSubscriber: RedisClientType,
@Inject('REDIS_PUB_CLIENT')
private readonly redisPublisher: RedisClientType,
) {}

subscribeToChannel(channel: string, callback: Function) {
this.redisSubscriber.subscribe(channel, (message) => {
callback(message);
});
}
publishToChannel(channel: string, message: string) {
this.redisPublisher.publish(channel, message);
}
}
20 changes: 20 additions & 0 deletions server/src/admin/admin.controller.spec.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
import { Test, TestingModule } from '@nestjs/testing';
import { AdminController } from './admin.controller';
import { AdminService } from './admin.service';

describe('AdminController', () => {
let controller: AdminController;

beforeEach(async () => {
const module: TestingModule = await Test.createTestingModule({
controllers: [AdminController],
providers: [AdminService],
}).compile();

controller = module.get<AdminController>(AdminController);
});

it('should be defined', () => {
expect(controller).toBeDefined();
});
});
59 changes: 59 additions & 0 deletions server/src/admin/admin.controller.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,59 @@
import {
Controller,
Get,
Req,
Res,
UnauthorizedException,
} from '@nestjs/common';
import { Response } from 'express';
import { RedisService } from 'redis/redis.service';
import { UserId } from 'src/users/decorator/userId.decorator';
import { channels } from './const/channels.const';

@Controller('admin')
export class AdminController {
constructor(private readonly redisService: RedisService) {}

@Get('events')
sse(@Req() req, @Res() res: Response, @UserId() userId: number) {
res.setHeader('Content-Type', 'text/event-stream');
res.setHeader('Cache-Control', 'no-cache');
res.setHeader('Connection', 'keep-alive');
res.flushHeaders();

if (userId !== 1) {
throw new UnauthorizedException('관리자가 아닙니다.');
}

const changeFormat = (channel, message) => {
const result = { channel, message };
return JSON.stringify(result);
};

res.write(`data: ${changeFormat('notice', 'Server connected')}\n\n`);
channels.forEach((channel) => {
this.redisService.subscribeToChannel(channel, (message) => {
res.write(`data: ${changeFormat(channel, message)}\n\n`);
});
});

req.on('close', () => {
res.end();
});
}
@Get('generate')
generate(@UserId() userId: number) {
if (userId !== 1) {
throw new UnauthorizedException('관리자가 아닙니다.');
}
this.redisService.publishToChannel('channel', 'processAiResult');
}

@Get('category')
category(@UserId() userId: number) {
if (userId !== 1) {
throw new UnauthorizedException('관리자가 아닙니다.');
}
this.redisService.publishToChannel('channel', 'processCategory');
}
}
9 changes: 9 additions & 0 deletions server/src/admin/admin.module.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
import { Module } from '@nestjs/common';
import { AdminController } from './admin.controller';
import { AdminService } from './admin.service';

@Module({
controllers: [AdminController],
providers: [AdminService],
})
export class AdminModule {}
18 changes: 18 additions & 0 deletions server/src/admin/admin.service.spec.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
import { Test, TestingModule } from '@nestjs/testing';
import { AdminService } from './admin.service';

describe('AdminService', () => {
let service: AdminService;

beforeEach(async () => {
const module: TestingModule = await Test.createTestingModule({
providers: [AdminService],
}).compile();

service = module.get<AdminService>(AdminService);
});

it('should be defined', () => {
expect(service).toBeDefined();
});
});
4 changes: 4 additions & 0 deletions server/src/admin/admin.service.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
import { Injectable } from '@nestjs/common';

@Injectable()
export class AdminService {}
7 changes: 7 additions & 0 deletions server/src/admin/const/channels.const.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
export const channels = [
'channel',
'sharedChecklist',
'ai_result',
'httpLog',
'wsLog',
];
Loading