UBERF-8469: Fix exit from github service (#6921)

Signed-off-by: Andrey Sobolev <haiodo@gmail.com>
This commit is contained in:
Andrey Sobolev 2024-10-15 12:04:39 +07:00 committed by Andrey Sobolev
parent 568dd2f47e
commit b5b4d23629
No known key found for this signature in database
GPG Key ID: BD80F68D68D8F7F2
13 changed files with 89 additions and 9 deletions

View File

@ -17,12 +17,12 @@
"_phase:bundle": "rushx bundle", "_phase:bundle": "rushx bundle",
"_phase:docker-build": "rushx docker:build", "_phase:docker-build": "rushx docker:build",
"_phase:docker-staging": "rushx docker:staging", "_phase:docker-staging": "rushx docker:staging",
"bundle": "mkdir -p bundle && esbuild src/index.ts --bundle --platform=node --outfile=bundle/bundle.js --log-level=error --sourcemap=external", "bundle": "mkdir -p bundle && esbuild src/index.ts --keep-names --bundle --platform=node --outfile=bundle/bundle.js --log-level=error --sourcemap=external",
"docker:build": "../../../common/scripts/docker_build.sh hardcoreeng/github", "docker:build": "../../../common/scripts/docker_build.sh hardcoreeng/github",
"docker:staging": "../../../common/scripts/docker_tag.sh hardcoreeng/github staging", "docker:staging": "../../../common/scripts/docker_tag.sh hardcoreeng/github staging",
"docker:push": "../../../common/scripts/docker_tag.sh hardcoreeng/github", "docker:push": "../../../common/scripts/docker_tag.sh hardcoreeng/github",
"docker:tbuild": "rush bundle --to @hcengineering/pod-github && docker build -t hardcoreeng/github . --platform=linux/amd64 && ../../../common/scripts/docker_tag_push.sh hardcoreeng/github", "docker:tbuild": "rush bundle --to @hcengineering/pod-github && docker build -t hardcoreeng/github . --platform=linux/amd64 && ../../../common/scripts/docker_tag_push.sh hardcoreeng/github",
"run-local": "cross-env APP_ID=$(cat ../../../../uberflow_private/appid) PRIVATE_KEY=\"$(cat ../../../../uberflow_private/private-key.pem)\" CLIENT_ID=$(cat ../../../../uberflow_private/client-id) CLIENT_SECRET=$(cat ../../../../uberflow_private/client-secret) SERVER_SECRET=secret ACCOUNTS_URL=http://localhost:3000/ COLLABORATOR_URL=http://localhost:3078 MINIO_ACCESS_KEY=minioadmin MINIO_SECRET_KEY=minioadmin MINIO_ENDPOINT=localhost ts-node src/index.ts", "run-local": "ruh.sh",
"format": "format src", "format": "format src",
"_phase:build": "compile transpile src", "_phase:build": "compile transpile src",
"_phase:test": "jest --passWithNoTests --silent", "_phase:test": "jest --passWithNoTests --silent",

View File

@ -0,0 +1,13 @@
export APP_ID="$POD_GITHUB_APPID"
export CLIENT_ID="$POD_GITHUB_CLIENTID"
export CLIENT_SECRET="$POD_GITHUB_CLIENT_SECRET"
export PRIVATE_KEY="$POD_GITHUB_PRIVATE_KEY"
export SERVER_SECRET=secret
export ACCOUNTS_URL=http://localhost:3000
export COLLABORATOR_URL=http://localhost:3078
export MINIO_ACCESS_KEY=minioadminchmo
export MINIO_SECRET_KEY=minioadmin
export MINIO_ENDPOINT=localhost
export MONGO_URL=mongodb://localhost:27017
rush bundle --to @hcengineering/pod-github
node $@ bundle/bundle.js

View File

@ -2,14 +2,14 @@
// Copyright © 2023 Hardcore Engineering Inc. // Copyright © 2023 Hardcore Engineering Inc.
// //
import { MeasureMetricsContext, metricsToString, newMetrics } from '@hcengineering/core' import { Analytics } from '@hcengineering/analytics'
import { SplitLogger, configureAnalytics } from '@hcengineering/analytics-service' import { SplitLogger, configureAnalytics } from '@hcengineering/analytics-service'
import { MeasureMetricsContext, metricsToString, newMetrics } from '@hcengineering/core'
import { loadBrandingMap } from '@hcengineering/server-core'
import { writeFile } from 'fs/promises' import { writeFile } from 'fs/promises'
import { join } from 'path' import { join } from 'path'
import config from './config' import config from './config'
import { start } from './server' import { start } from './server'
import { Analytics } from '@hcengineering/analytics'
import { loadBrandingMap } from '@hcengineering/server-core'
// Load and inc startID, to have easy logs. // Load and inc startID, to have easy logs.
@ -39,11 +39,18 @@ const intTimer = setInterval(() => {
} }
}, 30000) }, 30000)
void start(metricsContext, loadBrandingMap(config.BrandingPath)) let doOnClose: () => Promise<void> = async () => {}
void start(metricsContext, loadBrandingMap(config.BrandingPath)).then((r) => {
doOnClose = r
})
const onClose = (): void => { const onClose = (): void => {
clearInterval(intTimer) clearInterval(intTimer)
metricsContext.info('Closed') metricsContext.info('Closed')
void doOnClose().then((r) => {
process.exit(0)
})
} }
process.on('uncaughtException', (e) => { process.on('uncaughtException', (e) => {

View File

@ -96,6 +96,7 @@ export class PlatformWorker {
async close (): Promise<void> { async close (): Promise<void> {
this.canceled = true this.canceled = true
clearInterval(this.periodicSyncInterval)
await Promise.all( await Promise.all(
[...this.clients.values()].map(async (worker) => { [...this.clients.values()].map(async (worker) => {
await worker.close() await worker.close()

View File

@ -18,7 +18,7 @@ import { decodeToken } from '@hcengineering/server-token'
/** /**
* @public * @public
*/ */
export async function start (ctx: MeasureContext, brandingMap: BrandingMap): Promise<void> { export async function start (ctx: MeasureContext, brandingMap: BrandingMap): Promise<() => Promise<void>> {
// Create an authenticated Octokit client authenticated as a GitHub App // Create an authenticated Octokit client authenticated as a GitHub App
ctx.info('Running Huly Github integration', { appId: config.AppID, clientID: config.ClientID }) ctx.info('Running Huly Github integration', { appId: config.AppID, clientID: config.ClientID })
@ -184,8 +184,13 @@ export async function start (ctx: MeasureContext, brandingMap: BrandingMap): Pro
} }
}) })
app.listen(port, () => { const server = app.listen(port, () => {
ctx.info(`Server is listening for events at: ${localWebhookUrl}`) ctx.info(`Server is listening for events at: ${localWebhookUrl}`)
ctx.info('Press Ctrl + C to quit.') ctx.info('Press Ctrl + C to quit.')
}) })
return async () => {
await worker.close()
server.close()
}
} }

View File

@ -457,6 +457,9 @@ export class CommentSyncManager implements DocSyncManager {
repositories: GithubIntegrationRepository[] repositories: GithubIntegrationRepository[]
): Promise<void> { ): Promise<void> {
for (const repo of repositories) { for (const repo of repositories) {
if (this.provider.isClosing()) {
break
}
const syncKey = `${repo._id}:comment` const syncKey = `${repo._id}:comment`
if (repo.githubProject === undefined || !repo.enabled || integration.synchronized.has(syncKey)) { if (repo.githubProject === undefined || !repo.enabled || integration.synchronized.has(syncKey)) {
if (!repo.enabled) { if (!repo.enabled) {
@ -487,6 +490,9 @@ export class CommentSyncManager implements DocSyncManager {
}) })
try { try {
for await (const data of i) { for await (const data of i) {
if (this.provider.isClosing()) {
break
}
const comments: CommentExternalData[] = data.data as any const comments: CommentExternalData[] = data.data as any
this.ctx.info('retrieve comments for', { this.ctx.info('retrieve comments for', {
repo: repo.name, repo: repo.name,

View File

@ -815,7 +815,7 @@ export abstract class IssueSyncManagerBase {
// Collect field update. // Collect field update.
for (const [k, v] of Object.entries(platformUpdate)) { for (const [k, v] of Object.entries(platformUpdate)) {
const mapping = target.mappings.find((it) => it.name === k) const mapping = target.mappings.filter((it) => it != null).find((it) => it.name === k)
if (mapping === undefined) { if (mapping === undefined) {
continue continue
} }

View File

@ -985,6 +985,9 @@ export class IssueSyncManager extends IssueSyncManagerBase implements DocSyncMan
let partsize = 50 let partsize = 50
try { try {
while (true) { while (true) {
if (this.provider.isClosing()) {
break
}
const idsPart = ids.splice(0, partsize) const idsPart = ids.splice(0, partsize)
if (idsPart.length === 0) { if (idsPart.length === 0) {
break break
@ -1080,6 +1083,9 @@ export class IssueSyncManager extends IssueSyncManagerBase implements DocSyncMan
repositories: GithubIntegrationRepository[] repositories: GithubIntegrationRepository[]
): Promise<void> { ): Promise<void> {
for (const repo of repositories) { for (const repo of repositories) {
if (this.provider.isClosing()) {
break
}
const prj = projects.find((it) => repo.githubProject === it._id) const prj = projects.find((it) => repo.githubProject === it._id)
if (prj === undefined) { if (prj === undefined) {
continue continue
@ -1128,6 +1134,9 @@ export class IssueSyncManager extends IssueSyncManagerBase implements DocSyncMan
) )
try { try {
for await (const data of i) { for await (const data of i) {
if (this.provider.isClosing()) {
break
}
const issues: IssueExternalData[] = data.repository.issues.nodes const issues: IssueExternalData[] = data.repository.issues.nodes
if (issues.some((issue) => issue.url === undefined && Object.keys(issue).length === 0)) { if (issues.some((issue) => issue.url === undefined && Object.keys(issue).length === 0)) {
this.ctx.error('empty document content', { this.ctx.error('empty document content', {

View File

@ -376,6 +376,9 @@ export class ProjectsSyncManager implements DocSyncManager {
repositories: GithubIntegrationRepository[] repositories: GithubIntegrationRepository[]
): Promise<void> { ): Promise<void> {
for (const prj of projects) { for (const prj of projects) {
if (this.provider.isClosing()) {
break
}
// Wait global project sync // Wait global project sync
await integration.syncLock.get(prj._id) await integration.syncLock.get(prj._id)
@ -449,6 +452,9 @@ export class ProjectsSyncManager implements DocSyncManager {
(it) => it.space === prj._id (it) => it.space === prj._id
) )
for (const m of milestones) { for (const m of milestones) {
if (this.provider.isClosing()) {
break
}
try { try {
let { projectStructure, wasUpdates } = await this.ctx.withLog( let { projectStructure, wasUpdates } = await this.ctx.withLog(
'update project structure', 'update project structure',

View File

@ -1433,6 +1433,9 @@ export class PullRequestSyncManager extends IssueSyncManagerBase implements DocS
repositories: GithubIntegrationRepository[] repositories: GithubIntegrationRepository[]
): Promise<void> { ): Promise<void> {
for (const repo of repositories) { for (const repo of repositories) {
if (this.provider.isClosing()) {
break
}
const prj = projects.find((it) => repo.githubProject === it._id) const prj = projects.find((it) => repo.githubProject === it._id)
if (prj === undefined) { if (prj === undefined) {
continue continue
@ -1518,6 +1521,9 @@ export class PullRequestSyncManager extends IssueSyncManagerBase implements DocS
} }
) )
for await (const data of pullRequestIterator) { for await (const data of pullRequestIterator) {
if (this.provider.isClosing()) {
break
}
const issues: PullRequestExternalData[] = data.repository.pullRequests.nodes const issues: PullRequestExternalData[] = data.repository.pullRequests.nodes
this.ctx.info('retrieve pull requests for', { this.ctx.info('retrieve pull requests for', {
repo: repo.name, repo: repo.name,

View File

@ -60,6 +60,9 @@ export class UsersSyncManager implements DocSyncManager {
repositories: GithubIntegrationRepository[] repositories: GithubIntegrationRepository[]
): Promise<void> { ): Promise<void> {
for (const repo of repositories) { for (const repo of repositories) {
if (this.provider.isClosing()) {
break
}
const syncKey = `${repo._id}:users` const syncKey = `${repo._id}:users`
if ( if (
repo.githubProject === undefined || repo.githubProject === undefined ||
@ -107,6 +110,9 @@ export class UsersSyncManager implements DocSyncManager {
) )
try { try {
for await (const data of assignableUsersIterator) { for await (const data of assignableUsersIterator) {
if (this.provider.isClosing()) {
break
}
const users: UserInfo[] = data.repository[key]?.nodes ?? [] const users: UserInfo[] = data.repository[key]?.nodes ?? []
for (const d of users) { for (const d of users) {
if (d.login !== undefined) { if (d.login !== undefined) {

View File

@ -129,6 +129,8 @@ export interface IntegrationManager {
getProjectRepositories: (space: Ref<Space>) => Promise<GithubIntegrationRepository[]> getProjectRepositories: (space: Ref<Space>) => Promise<GithubIntegrationRepository[]>
getRepositoryById: (ref?: Ref<GithubIntegrationRepository> | null) => Promise<GithubIntegrationRepository | undefined> getRepositoryById: (ref?: Ref<GithubIntegrationRepository> | null) => Promise<GithubIntegrationRepository | undefined>
isClosing: () => boolean
} }
export type ExternalSyncField = 'externalVersion' | 'derivedVersion' export type ExternalSyncField = 'externalVersion' | 'derivedVersion'

View File

@ -117,11 +117,18 @@ export class GithubWorker implements IntegrationManager {
personMapper: UsersSyncManager personMapper: UsersSyncManager
isClosing (): boolean {
return this.closing
}
async close (): Promise<void> { async close (): Promise<void> {
clearInterval(this.periodicTimer) clearInterval(this.periodicTimer)
this.closing = true this.closing = true
this.ctx.warn('Closing', { workspace: this.workspace.name })
this.triggerSync()
await this.syncPromise await this.syncPromise
this.ctx.warn('ClosingDone', { workspace: this.workspace.name })
await this.client.close() await this.client.close()
} }
@ -1347,6 +1354,9 @@ export class GithubWorker implements IntegrationManager {
} }
private async waitChanges (): Promise<void> { private async waitChanges (): Promise<void> {
if (this.closing) {
return
}
if (this.triggerRequests > 0 || this.updateRequests > 0) { if (this.triggerRequests > 0 || this.updateRequests > 0) {
this.ctx.info('Trigger check pending:', { this.ctx.info('Trigger check pending:', {
requests: this.triggerRequests, requests: this.triggerRequests,
@ -1404,6 +1414,9 @@ export class GithubWorker implements IntegrationManager {
async _performFullSync (): Promise<void> { async _performFullSync (): Promise<void> {
// Wait previous active sync // Wait previous active sync
for (const integration of this.integrations.values()) { for (const integration of this.integrations.values()) {
if (this.closing) {
break
}
await this.ctx.withLog( await this.ctx.withLog(
'external sync', 'external sync',
{ installation: integration.installationName, workspace: this.workspace.name }, { installation: integration.installationName, workspace: this.workspace.name },
@ -1442,6 +1455,9 @@ export class GithubWorker implements IntegrationManager {
// Cleanup broken synchronized documents // Cleanup broken synchronized documents
while (true) { while (true) {
if (this.closing) {
break
}
const withError = await derivedClient.findAll<any>( const withError = await derivedClient.findAll<any>(
github.class.DocSyncInfo, github.class.DocSyncInfo,
{ error: { $ne: null }, url: null }, { error: { $ne: null }, url: null },
@ -1458,6 +1474,9 @@ export class GithubWorker implements IntegrationManager {
} }
for (const { _class, mapper } of this.mappers) { for (const { _class, mapper } of this.mappers) {
if (this.closing) {
break
}
await this.ctx.withLog( await this.ctx.withLog(
'external sync', 'external sync',
{ _class: _class.join(', '), workspace: this.workspace.name }, { _class: _class.join(', '), workspace: this.workspace.name },