Skip to content

Commit

Permalink
[server] Stream imagebuild logs from headless workspace directly
Browse files Browse the repository at this point in the history
  • Loading branch information
geropl committed Jan 31, 2022
1 parent fcfbb46 commit 351b1ab
Show file tree
Hide file tree
Showing 4 changed files with 118 additions and 12 deletions.
22 changes: 20 additions & 2 deletions components/dashboard/src/start/StartWorkspace.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -418,11 +418,29 @@ function ImageBuildView(props: ImageBuildViewProps) {
const logsEmitter = new EventEmitter();

useEffect(() => {
const watchBuild = () => getGitpodService().server.watchWorkspaceImageBuildLogs(props.workspaceId);
let registered = false;
const watchBuild = () => {
if (registered) {
return;
}

getGitpodService().server.watchWorkspaceImageBuildLogs(props.workspaceId)
.then(() => registered = true)
.catch(err => {

if (err?.code === ErrorCodes.HEADLESS_LOG_NOT_YET_AVAILABLE) {
// wait, and then retry
setTimeout(watchBuild, 5000);
}
})
}
watchBuild();

const toDispose = getGitpodService().registerClient({
notifyDidOpenConnection: () => watchBuild(),
notifyDidOpenConnection: () => {
registered = false; // new connection, we're not registered anymore
watchBuild();
},
onWorkspaceImageBuildLogs: (info: WorkspaceImageBuild.StateInfo, content?: WorkspaceImageBuild.LogContent) => {
if (!content) {
return;
Expand Down
3 changes: 3 additions & 0 deletions components/gitpod-protocol/src/messaging/error.ts
Original file line number Diff line number Diff line change
Expand Up @@ -78,4 +78,7 @@ export namespace ErrorCodes {

// 630 Snapshot Error
export const SNAPSHOT_ERROR = 630;

// 640 Headless logs are not available (yet)
export const HEADLESS_LOG_NOT_YET_AVAILABLE = 640;
}
87 changes: 77 additions & 10 deletions components/server/src/workspace/gitpod-server-impl.ts
Original file line number Diff line number Diff line change
Expand Up @@ -46,7 +46,7 @@ import { WorkspaceDeletionService } from './workspace-deletion-service';
import { WorkspaceFactory } from './workspace-factory';
import { WorkspaceStarter } from './workspace-starter';
import { HeadlessLogUrls } from "@gitpod/gitpod-protocol/lib/headless-workspace-log";
import { HeadlessLogService } from "./headless-log-service";
import { HeadlessLogService, HeadlessLogEndpoint } from "./headless-log-service";
import { InvalidGitpodYMLError } from "./config-provider";
import { ProjectsService } from "../projects/projects-service";
import { LocalMessageBroker } from "../messaging/local-message-broker";
Expand All @@ -58,6 +58,7 @@ import { ClientMetadata } from '../websocket/websocket-connection-manager';
import { ConfigurationService } from '../config/configuration-service';
import { ProjectEnvVar } from '@gitpod/gitpod-protocol/src/protocol';
import { InstallationAdminSettings } from '@gitpod/gitpod-protocol';
import { Deferred } from '@gitpod/gitpod-protocol/lib/util/deferred';

// shortcut
export const traceWI = (ctx: TraceContext, wi: Omit<LogContext, "userId">) => TraceContext.setOWI(ctx, wi); // userId is already taken care of in WebsocketConnectionManager
Expand Down Expand Up @@ -1112,24 +1113,90 @@ export class GitpodServerImpl implements GitpodServerWithTracing, Disposable {
traceWI(ctx, { workspaceId });

const user = this.checkAndBlockUser("watchWorkspaceImageBuildLogs", undefined, { workspaceId });
const logCtx: LogContext = { userId: user.id, workspaceId };

const { instance, workspace } = await this.internGetCurrentWorkspaceInstance(ctx, workspaceId);
if (!this.client) {
const client = this.client;
if (!client) {
return;
}

const logCtx: LogContext = { userId: user.id, workspaceId };
let { instance, workspace } = await this.internGetCurrentWorkspaceInstance(ctx, workspaceId);
if (!instance) {
log.debug(logCtx, `No running instance for workspaceId.`);
return;
}
traceWI(ctx, { instanceId: instance.id });
if (!workspace.imageNameResolved) {
log.debug(logCtx, `No imageNameResolved set for workspaceId, cannot watch logs.`);
return;
}
const teamMembers = await this.getTeamMembersByProject(workspace.projectId);
await this.guardAccess({ kind: "workspaceInstance", subject: instance, workspace, teamMembers }, "get");
if (!this.client) {

// wait for up to 20s for imageBuildLogInfo to appear due to:
// - db-sync round-trip times
// - but also: wait until the image build actually started (image pull!), and log info is available!
for (let i = 0; i < 10; i++) {
if (!instance || instance.status.phase !== 'preparing') {
log.debug(logCtx, `imagebuild logs: instance is not/no longer in 'preparing' state`, { phase: instance?.status.phase });
return;
}
if (workspace.imageBuildInfo?.log) {
break;
}
await new Promise(resolve => setTimeout(resolve, 2000));

({ instance, workspace } = await this.internGetCurrentWorkspaceInstance(ctx, workspaceId));
if (!workspace) {
log.warn(logCtx, `no workspace for workspaceId.`);
return;
}
}

if (!workspace.imageBuildInfo?.log) {
// during roll-out this is our fall-back case.
// Afterwards we might want to do some spinning-lock and re-check for a certain period (30s?) to give db-sync
// a change to move the imageBuildLogInfo across the globe.

log.warn(logCtx, "imageBuild logs: fallback!");
ctx.span?.setTag("workspace.imageBuild.logs.fallback", true);
await this.deprecatedDoWatchWorkspaceImageBuildLogs(ctx, logCtx, workspace);
return;
}
const logInfo = workspace.imageBuildInfo.log;

const aborted = new Deferred<boolean>();
try {
const logEndpoint: HeadlessLogEndpoint = {
url: logInfo.url,
headers: logInfo.headers,
};
let lineCount = 0;
await this.headlessLogService.streamImageBuildLog(logCtx, logEndpoint, async (chunk) => {
if (aborted.isResolved) {
return;
}

try {
chunk = chunk.replace("\n", WorkspaceImageBuild.LogLine.DELIMITER);
lineCount += chunk.split(WorkspaceImageBuild.LogLine.DELIMITER_REGEX).length;

client.onWorkspaceImageBuildLogs(undefined as any, {
text: chunk,
isDiff: true,
upToLine: lineCount
});
} catch (err) {
log.error("error while streaming imagebuild logs", err);
aborted.resolve(true);
}
}, aborted);
} catch (err) {
log.error(logCtx, "cannot watch imagebuild logs for workspaceId", err);
throw new ResponseError(ErrorCodes.HEADLESS_LOG_NOT_YET_AVAILABLE, "cannot watch imagebuild logs for workspaceId");
} finally {
aborted.resolve(false);
}
}

protected async deprecatedDoWatchWorkspaceImageBuildLogs(ctx: TraceContext, logCtx: LogContext, workspace: Workspace) {
if (!workspace.imageNameResolved) {
log.debug(logCtx, `No imageNameResolved set for workspaceId, cannot watch logs.`);
return;
}

Expand Down
18 changes: 18 additions & 0 deletions components/server/src/workspace/headless-log-service.ts
Original file line number Diff line number Diff line change
Expand Up @@ -261,6 +261,24 @@ export class HeadlessLogService {
await this.retryOnError(doStream, "stream workspace logs", doContinue, aborted);
}

/**
* Streaming imagebuild logs is different to other headless workspaces (prebuilds) because we do not store them as "workspace" (incl. status, etc.), but have a special field "workspace.imageBuildLogInfo".
* @param logCtx
* @param logEndpoint
* @param sink
* @param aborted
*/
async streamImageBuildLog(logCtx: LogContext, logEndpoint: HeadlessLogEndpoint, sink: (chunk: string) => Promise<void>, aborted: Deferred<boolean>): Promise<void> {
const tasks = await this.supervisorListTasks(logCtx, logEndpoint);
if (tasks.length === 0) {
throw new Error(`imagebuild logs: not tasks found for endpoint ${logEndpoint.url}!`);
}

// we're just looking at the first stream; image builds just have one stream atm
const task = tasks[0];
await this.streamWorkspaceLog(logCtx, logEndpoint, task.getTerminal(), sink, () => Promise.resolve(true), aborted);
}

/**
* Retries op while the passed WorkspaceInstance is still starting. Retries are stopped if either:
* - `op` calls `retry(false)` and an err is thrown, it is re-thrown by this method
Expand Down

0 comments on commit 351b1ab

Please sign in to comment.