mirror of
https://github.com/hcengineering/platform.git
synced 2025-04-06 07:49:20 +00:00
UBERF-8469: Fix exit from github service (#6921)
Signed-off-by: Andrey Sobolev <haiodo@gmail.com>
This commit is contained in:
parent
568dd2f47e
commit
b5b4d23629
@ -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",
|
||||||
|
13
services/github/pod-github/run.sh
Executable file
13
services/github/pod-github/run.sh
Executable 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
|
@ -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) => {
|
||||||
|
@ -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()
|
||||||
|
@ -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()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -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,
|
||||||
|
@ -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
|
||||||
}
|
}
|
||||||
|
@ -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', {
|
||||||
|
@ -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',
|
||||||
|
@ -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,
|
||||||
|
@ -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) {
|
||||||
|
@ -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'
|
||||||
|
@ -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 },
|
||||||
|
Loading…
Reference in New Issue
Block a user