Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Enable streaming of build logs #459

Draft
wants to merge 40 commits into
base: master
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
40 commits
Select commit Hold shift + click to select a range
ff70a52
add getlogs method
munishchouhan Apr 17, 2024
6f3cd8f
fixed builderName
munishchouhan Apr 17, 2024
16481a0
Merge branch 'master' into 451-enable-streaming-of-build-logs
munishchouhan Apr 18, 2024
8af97ef
fixed DockerBuildStrategyTest
munishchouhan Apr 18, 2024
22a574e
Merge remote-tracking branch 'origin/451-enable-streaming-of-build-lo…
munishchouhan Apr 18, 2024
d228dd9
added BuildLogLocalServiceImpl
munishchouhan Apr 18, 2024
a5ba7d9
added getCurrentLogsPod(String name)
munishchouhan Apr 18, 2024
6cb5e0e
fixed logs color issue
munishchouhan Apr 18, 2024
eafcb94
added build in progress phase and dynamic logs loading
munishchouhan Apr 18, 2024
c335997
minor change
munishchouhan Apr 18, 2024
26fdb65
changed to inputstream
munishchouhan Apr 19, 2024
93491f5
strip ansi escape codes
munishchouhan Apr 19, 2024
7869323
added docs
munishchouhan Apr 19, 2024
b287a56
fixed tests
munishchouhan Apr 19, 2024
a294b64
added test
munishchouhan Apr 19, 2024
94f40e7
corrected docs
munishchouhan Apr 19, 2024
db5afde
corrected docs
munishchouhan Apr 19, 2024
f6bb71c
Merge branch 'master' into 451-enable-streaming-of-build-logs
munishchouhan Apr 22, 2024
e7313e7
Merge branch 'master' into 451-enable-streaming-of-build-logs
munishchouhan Apr 22, 2024
1def98a
updated doc [ci skip]
munishchouhan Apr 22, 2024
a43a160
Merge remote-tracking branch 'origin/451-enable-streaming-of-build-lo…
munishchouhan Apr 22, 2024
a559584
reverted doc changes
munishchouhan Apr 22, 2024
0c06d3c
Merge branch 'master' into 451-enable-streaming-of-build-logs
pditommaso Apr 23, 2024
153b0df
Merge branch 'master' into 451-enable-streaming-of-build-logs
munishchouhan Apr 23, 2024
96afbb9
Merge branch 'master' into 451-enable-streaming-of-build-logs
munishchouhan May 4, 2024
534da81
Merge branch 'master' into 451-enable-streaming-of-build-logs
munishchouhan May 15, 2024
4859154
Merge branch 'master' into 451-enable-streaming-of-build-logs
munishchouhan Jun 11, 2024
fd9ea32
fix tests
munishchouhan Jun 11, 2024
7dedb5f
Merge branch 'master' into 451-enable-streaming-of-build-logs
munishchouhan Jul 22, 2024
aeb386c
Merge branch 'master' into 451-enable-streaming-of-build-logs
munishchouhan Aug 14, 2024
0c53ec5
Merge branch 'master' into 451-enable-streaming-of-build-logs
munishchouhan Aug 20, 2024
59f25e8
Merge branch 'master' into 451-enable-streaming-of-build-logs
munishchouhan Aug 29, 2024
1511ddb
fixed errors
munishchouhan Aug 29, 2024
167a9ad
fixed errors
munishchouhan Aug 30, 2024
1752d55
master merged
munishchouhan Oct 17, 2024
e09c9ec
Merge branch 'master' into 451-enable-streaming-of-build-logs
munishchouhan Oct 22, 2024
d6b19e5
fixed error
munishchouhan Oct 22, 2024
ed403ee
fixed error
munishchouhan Oct 22, 2024
1c304b1
fixed tests
munishchouhan Oct 22, 2024
7b09548
Merge branch 'master' into 451-enable-streaming-of-build-logs
munishchouhan Nov 5, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -36,7 +36,9 @@ abstract class BuildStrategy {
private BuildConfig buildConfig

abstract void build(String jobName, BuildRequest req)


abstract InputStream getLogs(String buildId)

static final public String BUILDKIT_ENTRYPOINT = 'buildctl-daemonless.sh'

List<String> launchCmd(BuildRequest req) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -156,4 +156,16 @@ class DockerBuildStrategy extends BuildStrategy {
wrapper.add(buildConfig.singularityImage(platform))
return wrapper
}


@Override
InputStream getLogs(String jobName) {
def logCmd = ['docker', 'logs'] + jobName
log.info("Get build logs: ${logCmd.join(' ')}")
final proc = new ProcessBuilder()
.command(logCmd)
.redirectErrorStream(true)
.start()
return proc.inputStream
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -98,6 +98,12 @@ class KubeBuildStrategy extends BuildStrategy {
}
}

@Override
InputStream getLogs(String jobName) {
final pod = k8sService.getLatestPodForJob(jobName)
return k8sService.getCurrentLogsPod(pod.spec.containers.first().name)
}

protected String getBuildImage(BuildRequest buildRequest){
if( buildRequest.formatDocker() ) {
return buildConfig.buildkitImage
Expand Down
2 changes: 2 additions & 0 deletions src/main/groovy/io/seqera/wave/service/k8s/K8sService.groovy
Original file line number Diff line number Diff line change
Expand Up @@ -53,6 +53,8 @@ interface K8sService {

V1Job launchMirrorJob(String name, String containerImage, List<String> args, Path workDir, Path creds, MirrorConfig config)

InputStream getCurrentLogsPod(String name)

V1Pod getLatestPodForJob(String jobName)

}
19 changes: 19 additions & 0 deletions src/main/groovy/io/seqera/wave/service/k8s/K8sServiceImpl.groovy
Original file line number Diff line number Diff line change
Expand Up @@ -347,6 +347,25 @@ class K8sServiceImpl implements K8sService {
}
}

/**
* Fetch current available logs of a running pod
*
* @param name The pod name
* @return The logs as a string or when logs are not available or cannot be accessed
*/
@Override
InputStream getCurrentLogsPod(String name) {
try {
def logs = k8sClient.coreV1Api().readNamespacedPodLog(name, namespace, name, false, null, null, "false", false, null, null, null)
logs = logs ? logs.replaceAll("\u001B\\[[;\\d]*m", "") : null // strip ansi escape codes
return new ByteArrayInputStream(logs.getBytes())
} catch (Exception e) {
// logging trace here because errors are expected when the pod is not running
log.trace "Unable to fetch logs for pod: $name", e
return null
}
}

/**
* Delete a pod
*
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@

package io.seqera.wave.service.logs

import java.nio.charset.StandardCharsets
import java.util.concurrent.CompletableFuture
import java.util.concurrent.ExecutorService

Expand All @@ -27,6 +28,7 @@ import groovy.transform.CompileStatic
import groovy.util.logging.Slf4j
import io.micronaut.context.annotation.Requires
import io.micronaut.context.annotation.Value
import io.micronaut.http.MediaType
import io.micronaut.http.server.types.files.StreamedFile
import io.micronaut.objectstorage.ObjectStorageEntry
import io.micronaut.objectstorage.ObjectStorageOperations
Expand All @@ -35,6 +37,7 @@ import io.micronaut.runtime.event.annotation.EventListener
import io.micronaut.scheduling.TaskExecutors
import io.seqera.wave.service.builder.BuildEvent
import io.seqera.wave.service.builder.BuildRequest
import io.seqera.wave.service.builder.BuildStrategy
import io.seqera.wave.service.persistence.PersistenceService
import jakarta.annotation.PostConstruct
import jakarta.inject.Inject
Expand Down Expand Up @@ -64,6 +67,9 @@ class BuildLogServiceImpl implements BuildLogService {
@Inject
private PersistenceService persistenceService

@Inject
private BuildStrategy buildStrategy

@Nullable
@Value('${wave.build.logs.prefix}')
private String prefix
Expand Down Expand Up @@ -129,7 +135,14 @@ class BuildLogServiceImpl implements BuildLogService {
private StreamedFile fetchLogStream0(String buildId) {
if( !buildId ) return null
final Optional<ObjectStorageEntry<?>> result = objectStorageOperations.retrieve(logKey(buildId))
return result.isPresent() ? result.get().toStreamedFile() : null
return result.isPresent() ? result.get().toStreamedFile() : fetchLogStream1(buildId)
}

private StreamedFile fetchLogStream1(String buildId) {
def logStream = buildStrategy.getLogs(buildId)
if( !logStream )
return null
return logStream ? new StreamedFile(logStream, MediaType.APPLICATION_OCTET_STREAM_TYPE) : null
}

@Override
Expand Down
59 changes: 59 additions & 0 deletions src/main/resources/io/seqera/wave/build-view.hbs
Original file line number Diff line number Diff line change
Expand Up @@ -74,6 +74,21 @@
Container build failed
</h4>
</div>
{{else build_in_progress}}
<div style="color: #3c763d; background-color: #ffd300; padding: 15px; border: 1px solid transparent; border-radius: 4px;">
<h4 style="margin-top:0; color: inherit;">
Container build in progress
<div style="color: #3c763d; background-color: #dff0d8; padding: 15px; border: 1px solid transparent; border-radius: 4px;">
<h4 style="margin-top:0; color: inherit;">
Container build completed successfully!
</h4>
</div>
{{else build_failed}}
<div style="color: #a94442; background-color: #f2dede; padding: 15px; border: 1px solid transparent; border-radius: 4px;">
<h4 style="margin-top:0; color: inherit;">
Container build failed
</h4>
</div>
{{else build_in_progress}}
{{! build is not completed, show a spinning icon }}
<style>
Expand Down Expand Up @@ -209,6 +224,11 @@
{{/if}}

{{#if build_log_data}}
<h3>Build logs</h3>
<pre id="buildLogs" style="white-space: pre-wrap; overflow-x: auto; overflow-y: auto; background-color: #ededed; padding: 15px; border-radius: 4px; margin-bottom:30px;">{{build_log_data}}</pre>
{{#if build_log_truncated}}
<a href="{{build_log_url}}" download>Click here to download the complete build log</a>
{{/if}}
<h3>Build logs</h3>
<div style="position: relative; margin-bottom: 30px;">
<button id="logCopyBtn" class="copy-btn2" onclick="copyToClipboard('logData', 'logCopyBtn')" title="Copy">
Expand All @@ -226,6 +246,45 @@
</div>
{{/if}}

<script>
function checkStatusAndFetchData() {
fetch("{{server_url}}/v1alpha1/builds/{{build_id}}/status", {method: 'GET'})
.then(response => {
if (!response.ok) {
clearInterval(interval)
return;
}
return response.json();
}).then(statusResponse => {
if (statusResponse.status === 'PENDING') {
fetchData();
}else{
clearInterval(interval)
}
}).catch(error => {
console.error('Error checking status:', error)
clearInterval(interval)
});
}
function fetchData() {
fetch('{{server_url}}/v1alpha1/builds/{{build_id}}/logs', {method: 'GET', cache: 'no-cache'})
.then(response => {
if (!response.ok) {
clearInterval(interval)
return;
}
return response.text();
}).then(logs => {
document.getElementById('buildLogs').innerHTML = logs;
}).catch(error => {
console.error('Error fetching build logs:', error)
clearInterval(interval)
});
}

const interval = setInterval(checkStatusAndFetchData, 2000);
</script>

<div class="footer" style="clear:both;width:100%;">
<hr class="footer-hr" style="height:0;overflow:visible;margin-top:30px;border:0;border-top:1px solid #eee;color:#999999;font-size:12px;line-height:18px;margin-bottom:30px;">
<img style="float:right; width: 150px;" src="">
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -77,6 +77,11 @@ class ContainerBuildServiceTest extends Specification {
// do nothing
log.debug "Running fake build job=$jobName - request=$request"
}

@Override
InputStream getLogs(String podName) {
return "logs for pod name"
}
}

@Primary
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -41,20 +41,21 @@ class DockerBuildStrategyTest extends Specification {
def work = Path.of('/work/foo')
when:
def cmd = service.cmdForBuildkit('build-job-name', work, null, null)
def name = 'build-job-name'
then:
cmd == ['docker',
'run',
'--detach',
'--name',
'build-job-name',
name,
'--privileged',
'-v', '/work/foo:/work/foo',
'--entrypoint',
'buildctl-daemonless.sh',
'moby/buildkit:v0.14.1-rootless']

when:
cmd = service.cmdForBuildkit('build-job-name', work, Path.of('/foo/creds.json'), ContainerPlatform.of('arm64'))
cmd = service.cmdForBuildkit(name, work, Path.of('/foo/creds.json'), ContainerPlatform.of('arm64'))
then:
cmd == ['docker',
'run',
Expand Down Expand Up @@ -96,6 +97,7 @@ class DockerBuildStrategyTest extends Specification {
def creds = Path.of('/work/creds.json')
and:
def req = new BuildRequest(
id: '89fb83ce6ec8627b',
containerId: '89fb83ce6ec8627b',
buildId: 'bd-89fb83ce6ec8627b_1',
workspace: Path.of('/work/foo'),
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -521,6 +521,25 @@ class K8sServiceImplTest extends Specification {
ctx.close()
}

def "should get pod logs"() {
given:
def k8sClient = Mock(K8sClient)
def k8sService = new K8sServiceImpl(k8sClient: k8sClient)
def name = "builder-pod"
def logs = "\u001B[31mINFO: Build is in progress"

when:
InputStream result = k8sService.getCurrentLogsPod(name)

then:
2 * k8sClient.coreV1Api() >> Mock(CoreV1Api)
1 * k8sClient.coreV1Api().readNamespacedPodLog(_, _, _, _, _, _, _, _, _, _, _) >> logs
and:
result instanceof ByteArrayInputStream
String resultString = result.text
resultString == "INFO: Build is in progress"
}

def "getLatestPodForJob should return the latest pod when multiple pods are present"() {
given:
def jobName = "test-job"
Expand Down