Ajusta CRLF e add files

This commit is contained in:
LeoMortari
2025-11-04 20:03:16 -03:00
parent 9664da1ab6
commit e7e76d15e2
8 changed files with 144 additions and 0 deletions

View File

@@ -3,3 +3,8 @@ PORT=3000
DATABASE_URL="postgresql://username:password@postgres:5432/clipperia?schema=public" DATABASE_URL="postgresql://username:password@postgres:5432/clipperia?schema=public"
KEYCLOAK_URL="https://auth.clipperia.com.br" KEYCLOAK_URL="https://auth.clipperia.com.br"
YOUTUBE_API_URL="https://your-youtube-api-url.com" YOUTUBE_API_URL="https://your-youtube-api-url.com"
MINIO_ENDPOINT="minio.clipperia.com.br"
MINIO_PORT=9000
MINIO_USE_SSL=true
MINIO_ACCESS_KEY="user"
MINIO_SECRET_KEY="password"

View File

@@ -34,6 +34,7 @@
"dayjs": "1.11.13", "dayjs": "1.11.13",
"jwks-rsa": "3.2.0", "jwks-rsa": "3.2.0",
"lodash": "4.17.21", "lodash": "4.17.21",
"minio": "8.0.6",
"passport": "0.7.0", "passport": "0.7.0",
"passport-jwt": "4.0.1" "passport-jwt": "4.0.1"
}, },

View File

@@ -1,12 +1,14 @@
import { Module, MiddlewareConsumer } from '@nestjs/common'; import { Module, MiddlewareConsumer } from '@nestjs/common';
import { ConfigModule } from '@nestjs/config'; import { ConfigModule } from '@nestjs/config';
import { PrismaModule } from './prisma/prisma.module'; import { PrismaModule } from './prisma/prisma.module';
import { MinioModule } from './minio/minio.module';
import { AppController } from './app.controller'; import { AppController } from './app.controller';
import { AppService } from './app.service'; import { AppService } from './app.service';
import { VideosModule } from './modules/videos/videos.module'; import { VideosModule } from './modules/videos/videos.module';
import { AuthModule } from './modules/auth/auth.module'; import { AuthModule } from './modules/auth/auth.module';
import { VideosController } from './modules/videos/videos.controller'; import { VideosController } from './modules/videos/videos.controller';
import { UsuariosModule } from './modules/usuarios/usuarios.module'; import { UsuariosModule } from './modules/usuarios/usuarios.module';
import { FilesModule } from './modules/files/files.module';
import { LoggerMiddleware } from './middleware/logger.middleware'; import { LoggerMiddleware } from './middleware/logger.middleware';
import { RolesGuard } from './modules/auth/roles.guard'; import { RolesGuard } from './modules/auth/roles.guard';
@@ -14,9 +16,11 @@ import { RolesGuard } from './modules/auth/roles.guard';
imports: [ imports: [
ConfigModule.forRoot(), ConfigModule.forRoot(),
PrismaModule, PrismaModule,
MinioModule,
VideosModule, VideosModule,
AuthModule, AuthModule,
UsuariosModule, UsuariosModule,
FilesModule,
], ],
controllers: [AppController], controllers: [AppController],
providers: [AppService, RolesGuard], providers: [AppService, RolesGuard],

11
src/minio/minio.module.ts Normal file
View File

@@ -0,0 +1,11 @@
import { Module, Global } from '@nestjs/common';
import { ConfigModule } from '@nestjs/config';
import { MinioService } from './minio.service';
@Global()
@Module({
imports: [ConfigModule],
providers: [MinioService],
exports: [MinioService],
})
export class MinioModule {}

View File

@@ -0,0 +1,36 @@
import { Injectable, OnModuleInit, OnModuleDestroy } from '@nestjs/common';
import { ConfigService } from '@nestjs/config';
import * as Minio from 'minio';
@Injectable()
export class MinioService implements OnModuleInit, OnModuleDestroy {
private client: Minio.Client;
constructor(private configService: ConfigService) {}
onModuleInit() {
const endpoint = this.configService.get<string>(
'MINIO_ENDPOINT',
'https://minio.clipperia.com.br',
);
const cleanEndpoint = endpoint.replace(/^https?:\/\//, '');
this.client = new Minio.Client({
endPoint: cleanEndpoint,
port: parseInt(this.configService.get<string>('MINIO_PORT', '443')),
useSSL:
this.configService.get<string>('MINIO_USE_SSL', 'true') === 'true',
accessKey: this.configService.get<string>('MINIO_ACCESS_KEY'),
secretKey: this.configService.get<string>('MINIO_SECRET_KEY'),
});
}
async onModuleDestroy(): Promise<void> {
// MinIO client doesn't require explicit disconnect
}
getClient(): Minio.Client {
return this.client;
}
}

View File

@@ -0,0 +1,29 @@
import { Controller, Get, UseGuards } from '@nestjs/common';
import { AuthGuard } from '@nestjs/passport';
import { FilesService, type FileInfo } from './files.service';
import { RolesGuard } from '../auth/roles.guard';
import { Roles } from '../auth/decorator/roles.decorator';
@Controller('files')
@UseGuards(AuthGuard('jwt'), RolesGuard)
export class FilesController {
constructor(private readonly filesService: FilesService) {}
@Get('icons')
@Roles('admin')
async listIcons(): Promise<{
bucket: string;
folder: string;
files: FileInfo[];
total: number;
}> {
const files = await this.filesService.listIconsFromBucket();
return {
bucket: 'clipperia',
folder: 'icons',
files,
total: files.length,
};
}
}

View File

@@ -0,0 +1,9 @@
import { Module } from '@nestjs/common';
import { FilesController } from './files.controller';
import { FilesService } from './files.service';
@Module({
controllers: [FilesController],
providers: [FilesService],
})
export class FilesModule {}

View File

@@ -0,0 +1,49 @@
import { Injectable } from '@nestjs/common';
import { MinioService } from '../../minio/minio.service';
export interface FileInfo {
name: string;
size: number;
lastModified: Date;
etag: string;
}
@Injectable()
export class FilesService {
constructor(private readonly minioService: MinioService) {}
async listIconsFromBucket(): Promise<FileInfo[]> {
const client = this.minioService.getClient();
const bucketName = 'clipperia';
const prefix = 'icons/';
return new Promise((resolve, reject) => {
const files: FileInfo[] = [];
const objectsStream = client.listObjects(bucketName, prefix, true);
objectsStream.on('data', (obj) => {
console.log(obj);
if (obj.name) {
files.push({
name: obj.name,
size: obj.size!,
lastModified: obj.lastModified!,
etag: obj.etag!,
});
}
});
objectsStream.on('error', (err) => {
console.log(err);
reject(err);
});
objectsStream.on('end', () => {
console.log('ok');
resolve(files);
});
});
}
}