1 Commits

Author SHA1 Message Date
Ivan
7ea8505873 feat(repos): 添加消息队列分发。BAK 2021-02-25 16:11:06 +08:00
15 changed files with 72 additions and 76 deletions

View File

@@ -1,7 +1,6 @@
env: dev env: dev
http: http:
port: 7122 port: 7122
db: db:
postgres: postgres:
host: 192.168.31.194 host: 192.168.31.194
@@ -9,5 +8,9 @@ db:
database: fennec database: fennec
username: fennec username: fennec
password: password:
redis:
mq:
host: localhost
port: 6379
workspaces: workspaces:
root: '/Users/ivanli/Projects/fennec/workspaces' root: '/Users/ivanli/Projects/fennec/workspaces'

26
package-lock.json generated
View File

@@ -9087,6 +9087,14 @@
} }
} }
}, },
"mkdirp": {
"version": "0.5.5",
"resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.5.tgz",
"integrity": "sha512-NKmAlESf6jMGym1++R0Ra7wvhV+wFW63FaSOFPwRahvea0gMUcGUhVeAg/0BC0wiv9ih5NYPB1Wn1UEI1/L+xQ==",
"requires": {
"minimist": "^1.2.5"
}
},
"moment": { "moment": {
"version": "2.29.1", "version": "2.29.1",
"resolved": "https://registry.npmjs.org/moment/-/moment-2.29.1.tgz", "resolved": "https://registry.npmjs.org/moment/-/moment-2.29.1.tgz",
@@ -9118,16 +9126,6 @@
"on-finished": "^2.3.0", "on-finished": "^2.3.0",
"type-is": "^1.6.4", "type-is": "^1.6.4",
"xtend": "^4.0.0" "xtend": "^4.0.0"
},
"dependencies": {
"mkdirp": {
"version": "0.5.5",
"resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.5.tgz",
"integrity": "sha512-NKmAlESf6jMGym1++R0Ra7wvhV+wFW63FaSOFPwRahvea0gMUcGUhVeAg/0BC0wiv9ih5NYPB1Wn1UEI1/L+xQ==",
"requires": {
"minimist": "^1.2.5"
}
}
} }
}, },
"multimatch": { "multimatch": {
@@ -9491,14 +9489,6 @@
"minipass": "^2.9.0" "minipass": "^2.9.0"
} }
}, },
"mkdirp": {
"version": "0.5.5",
"resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.5.tgz",
"integrity": "sha512-NKmAlESf6jMGym1++R0Ra7wvhV+wFW63FaSOFPwRahvea0gMUcGUhVeAg/0BC0wiv9ih5NYPB1Wn1UEI1/L+xQ==",
"requires": {
"minimist": "^1.2.5"
}
},
"rimraf": { "rimraf": {
"version": "2.7.1", "version": "2.7.1",
"resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz",

View File

@@ -8,6 +8,7 @@ import { AppService } from './app.service';
import { ProjectsModule } from './projects/projects.module'; import { ProjectsModule } from './projects/projects.module';
import { ReposModule } from './repos/repos.module'; import { ReposModule } from './repos/repos.module';
import configuration from './commons/config/configuration'; import configuration from './commons/config/configuration';
import { BullModule } from '@nestjs/bull';
@Module({ @Module({
imports: [ imports: [
@@ -37,6 +38,16 @@ import configuration from './commons/config/configuration';
}), }),
inject: [ConfigService], inject: [ConfigService],
}), }),
BullModule.forRootAsync({
imports: [ConfigModule],
useFactory: async (configService: ConfigService) => ({
redis: {
host: configService.get<string>('redis.mq.host', 'localhost'),
port: configService.get<number>('redis.mq.port', 6379),
},
}),
inject: [ConfigService],
}),
ProjectsModule, ProjectsModule,
ReposModule, ReposModule,
], ],

View File

@@ -3,7 +3,6 @@ import {
IsOptional, IsOptional,
IsString, IsString,
IsUrl, IsUrl,
Matches,
MaxLength, MaxLength,
MinLength, MinLength,
} from 'class-validator'; } from 'class-validator';
@@ -20,9 +19,7 @@ export class CreateProjectInput {
@MinLength(2) @MinLength(2)
comment: string; comment: string;
@Matches( @IsUrl({ protocols: ['ssh'] })
/^(?:ssh:\/\/)?(?:[\w\d-_]+@)(?:[\w\d-_]+\.)*\w{2,10}(?::\d{1,5})?(?:\/[\w\d-_.]+)*/,
)
@MaxLength(256) @MaxLength(256)
sshUrl: string; sshUrl: string;

View File

@@ -1,16 +0,0 @@
import { InputType } from '@nestjs/graphql';
import { IsOptional, IsString, IsUUID } from 'class-validator';
@InputType()
export class CheckoutInput {
@IsUUID()
projectId: string;
@IsString()
@IsOptional()
branch?: string;
@IsString()
@IsOptional()
commitNumber?: string;
}

View File

@@ -1,4 +1,4 @@
import { InputType } from '@nestjs/graphql'; import { InputType, ObjectType } from '@nestjs/graphql';
import { IsOptional, IsString, IsUUID } from 'class-validator'; import { IsOptional, IsString, IsUUID } from 'class-validator';
@InputType() @InputType()

View File

@@ -0,0 +1 @@
export const WORKSPACE_ACTION = 'workspace-action';

View File

@@ -4,10 +4,17 @@ import { Project } from '../projects/project.entity';
import { ReposResolver } from './repos.resolver'; import { ReposResolver } from './repos.resolver';
import { ReposService } from './repos.service'; import { ReposService } from './repos.service';
import { ConfigModule } from '@nestjs/config'; import { ConfigModule } from '@nestjs/config';
import { ProjectsModule } from '../projects/projects.module'; import { BullModule } from '@nestjs/bull';
import { WORKSPACE_ACTION } from './repos.constants';
@Module({ @Module({
imports: [TypeOrmModule.forFeature([Project]), ConfigModule, ProjectsModule], imports: [
TypeOrmModule.forFeature([Project]),
ConfigModule,
BullModule.registerQueue({
name: WORKSPACE_ACTION,
}),
],
providers: [ReposResolver, ReposService], providers: [ReposResolver, ReposService],
}) })
export class ReposModule {} export class ReposModule {}

View File

@@ -1,7 +1,6 @@
import { Test, TestingModule } from '@nestjs/testing'; import { Test, TestingModule } from '@nestjs/testing';
import { ReposResolver } from './repos.resolver'; import { ReposResolver } from './repos.resolver';
import { ReposService } from './repos.service'; import { ReposService } from './repos.service';
import { ProjectsService } from '../projects/projects.service';
describe('ReposResolver', () => { describe('ReposResolver', () => {
let resolver: ReposResolver; let resolver: ReposResolver;
@@ -14,10 +13,6 @@ describe('ReposResolver', () => {
provide: ReposService, provide: ReposService,
useValue: {}, useValue: {},
}, },
{
provide: ProjectsService,
useValue: {},
},
], ],
}).compile(); }).compile();

View File

@@ -1,24 +1,19 @@
import { Args, Mutation, Query, Resolver } from '@nestjs/graphql'; import { Args, Query, Resolver } from '@nestjs/graphql';
import { ListLogsArgs } from './dtos/list-logs.args'; import { ListLogsArgs } from './dtos/list-logs.args';
import { ReposService } from './repos.service'; import { ReposService } from './repos.service';
import { LogList } from './dtos/log-list.model'; import { LogList } from './dtos/log-list.model';
import { ListBranchesArgs } from './dtos/list-branches.args'; import { ListBranchesArgs } from './dtos/list-branches.args';
import { BranchList } from './dtos/branch-list.model'; import { BranchList } from './dtos/branch-list.model';
import { CheckoutInput } from './dtos/checkout.input';
import { ProjectsService } from '../projects/projects.service';
@Resolver() @Resolver()
export class ReposResolver { export class ReposResolver {
constructor( constructor(private readonly service: ReposService) {}
private readonly service: ReposService,
private readonly projectService: ProjectsService,
) {}
@Query(() => LogList) @Query(() => LogList)
async listLogs(@Args('listLogsArgs') dto: ListLogsArgs) { async listLogs(@Args('listLogsArgs') dto: ListLogsArgs) {
return await this.service.listLogs(dto); return await this.service.listLogs(dto);
} }
@Query(() => BranchList) @Query(() => BranchList)
async listBranches( async ListBranchesArgs(
@Args('listBranchesArgs') dto: ListBranchesArgs, @Args('listBranchesArgs') dto: ListBranchesArgs,
): Promise<BranchList> { ): Promise<BranchList> {
return await this.service.listBranches(dto).then((data) => { return await this.service.listBranches(dto).then((data) => {
@@ -28,10 +23,4 @@ export class ReposResolver {
}; };
}); });
} }
@Mutation(() => Boolean)
async checkout(@Args('checkoutInput') dto: CheckoutInput): Promise<true> {
const project = await this.projectService.findOne(dto.projectId);
await this.service.checkoutCommit(project, dto.commitNumber);
return true;
}
} }

View File

@@ -75,7 +75,7 @@ describe('ReposService', () => {
it('should be checkout', async () => { it('should be checkout', async () => {
await service.checkoutBranch(getTest1Project(), 'master'); await service.checkoutBranch(getTest1Project(), 'master');
const filePath = join( const filePath = join(
service.getWorkspaceRoot(getTest1Project(), 'master'), service.getWorkspaceRoot(getTest1Project()),
'README.md', 'README.md',
); );
const text = await readFile(filePath, { encoding: 'utf-8' }); const text = await readFile(filePath, { encoding: 'utf-8' });
@@ -86,7 +86,7 @@ describe('ReposService', () => {
await service.checkoutBranch(getTest1Project(), 'branch-a'); await service.checkoutBranch(getTest1Project(), 'branch-a');
await service.checkoutBranch(getTest1Project(), 'branch-b'); await service.checkoutBranch(getTest1Project(), 'branch-b');
const filePath = join( const filePath = join(
service.getWorkspaceRoot(getTest1Project(), 'branch-b'), service.getWorkspaceRoot(getTest1Project()),
'branch-b.md', 'branch-b.md',
); );
const text = await readFile(filePath, { encoding: 'utf-8' }); const text = await readFile(filePath, { encoding: 'utf-8' });
@@ -100,7 +100,7 @@ describe('ReposService', () => {
it('checkout the specified version', async () => { it('checkout the specified version', async () => {
await service.checkoutBranch(getTest1Project(), 'master'); await service.checkoutBranch(getTest1Project(), 'master');
const filePath = join( const filePath = join(
service.getWorkspaceRoot(getTest1Project(), 'master'), service.getWorkspaceRoot(getTest1Project()),
'README.md', 'README.md',
); );
const text = await readFile(filePath, { encoding: 'utf-8' }); const text = await readFile(filePath, { encoding: 'utf-8' });
@@ -112,7 +112,7 @@ describe('ReposService', () => {
it('should be checkout', async () => { it('should be checkout', async () => {
await service.checkoutCommit(getTest1Project(), '498c782685'); await service.checkoutCommit(getTest1Project(), '498c782685');
const filePath = join( const filePath = join(
service.getWorkspaceRoot(getTest1Project(), '498c782685'), service.getWorkspaceRoot(getTest1Project()),
'README.md', 'README.md',
); );
const text = await readFile(filePath, { encoding: 'utf-8' }); const text = await readFile(filePath, { encoding: 'utf-8' });
@@ -121,7 +121,7 @@ describe('ReposService', () => {
it('should be checkout right commit', async () => { it('should be checkout right commit', async () => {
await service.checkoutCommit(getTest1Project(), '7f7123fe5b'); await service.checkoutCommit(getTest1Project(), '7f7123fe5b');
const filePath = join( const filePath = join(
service.getWorkspaceRoot(getTest1Project(), '7f7123fe5b'), service.getWorkspaceRoot(getTest1Project()),
'README.md', 'README.md',
); );
const text = await readFile(filePath, { encoding: 'utf-8' }); const text = await readFile(filePath, { encoding: 'utf-8' });

View File

@@ -19,11 +19,10 @@ export class ReposService {
private readonly configService: ConfigService, private readonly configService: ConfigService,
) {} ) {}
getWorkspaceRoot(project: Project, subDir = ''): string { getWorkspaceRoot(project: Project): string {
return join( return join(
this.configService.get<string>('workspaces.root'), this.configService.get<string>('workspaces.root'),
project.name, project.name,
encodeURIComponent(subDir),
); );
} }
@@ -31,14 +30,14 @@ export class ReposService {
// TODO: 获取锁,失败抛错。 // TODO: 获取锁,失败抛错。
} }
async getGit(project: Project, subDir = 'default') { async getGit(project: Project) {
const workspaceRoot = this.getWorkspaceRoot(project, subDir); const workspaceRoot = this.getWorkspaceRoot(project);
await this.lockWorkspace(workspaceRoot); await this.lockWorkspace(workspaceRoot);
const firstInit = await access(workspaceRoot, F_OK) const firstInit = await access(workspaceRoot, F_OK)
.then(() => false) .then(() => false)
.catch(async () => { .catch(async () => {
await mkdir(workspaceRoot, { recursive: true }); await mkdir(workspaceRoot);
return true; return true;
}); });
const git = gitP(workspaceRoot); const git = gitP(workspaceRoot);
@@ -55,9 +54,10 @@ export class ReposService {
}); });
const git = await this.getGit(project); const git = await this.getGit(project);
await git.fetch(); await git.fetch();
return await git.log( return await git.log({
dto.branch ? ['--branches', dto.branch, '--'] : ['--all'], '--branches': dto.branch ?? '',
); '--remotes': DEFAULT_REMOTE_NAME,
});
} }
async listBranches(dto: ListBranchesArgs) { async listBranches(dto: ListBranchesArgs) {
@@ -69,7 +69,7 @@ export class ReposService {
} }
async checkoutBranch(project: Project, branch: string) { async checkoutBranch(project: Project, branch: string) {
const git = await this.getGit(project, branch); const git = await this.getGit(project);
try { try {
await git.fetch(DEFAULT_REMOTE_NAME, branch); await git.fetch(DEFAULT_REMOTE_NAME, branch);
} catch (err) { } catch (err) {
@@ -87,7 +87,7 @@ export class ReposService {
} }
async checkoutCommit(project: Project, commitNumber: string) { async checkoutCommit(project: Project, commitNumber: string) {
const git = await this.getGit(project, commitNumber); const git = await this.getGit(project);
try { try {
await git.fetch(DEFAULT_REMOTE_NAME); await git.fetch(DEFAULT_REMOTE_NAME);
} catch (err) { } catch (err) {

View File

@@ -0,0 +1,8 @@
import { Process, Processor } from '@nestjs/bull';
import { WORKSPACE_ACTION } from './repos.constants';
@Processor(WORKSPACE_ACTION)
export class WorkspaceActionConsumer {
@Process()
async dispatch() {}
}

View File

@@ -0,0 +1,5 @@
import { WorkspaceActions } from './workspace-actions.enum';
export class WorkspaceAction {
action: WorkspaceActions;
}

View File

@@ -0,0 +1,6 @@
export enum WorkspaceActions {
checkoutBranch = 'checkoutBranch',
checkoutCommit = 'checkoutCommit',
listLogs = 'listLogs',
listBranches = 'listBranches',
}