Skip to content

Commit

Permalink
Add build list view for containerId (#696)
Browse files Browse the repository at this point in the history


Signed-off-by: munishchouhan <[email protected]>
Signed-off-by: Paolo Di Tommaso <[email protected]>
Co-authored-by: Paolo Di Tommaso <[email protected]>
  • Loading branch information
munishchouhan and pditommaso authored Oct 16, 2024
1 parent fdf1be7 commit 3cabb83
Show file tree
Hide file tree
Showing 10 changed files with 379 additions and 43 deletions.
77 changes: 52 additions & 25 deletions src/main/groovy/io/seqera/wave/controller/ViewController.groovy
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,7 @@ import io.micronaut.http.annotation.Get
import io.micronaut.http.annotation.QueryValue
import io.micronaut.scheduling.TaskExecutors
import io.micronaut.scheduling.annotation.ExecuteOn
import io.micronaut.views.ModelAndView
import io.micronaut.views.View
import io.seqera.wave.exception.HttpResponseException
import io.seqera.wave.exception.NotFoundException
Expand Down Expand Up @@ -114,55 +115,81 @@ class ViewController {
return binding
}

@View("build-view")
@Get('/builds/{buildId}')
HttpResponse viewBuild(String buildId) {
// check redirection for invalid suffix in the form `-nn`
final r1 = shouldRedirect1(buildId)
final r1 = isBuildInvalidSuffix(buildId)
if( r1 ) {
log.debug "Redirect to build page [1]: $r1"
return HttpResponse.redirect(URI.create(r1))
}
// check redirection when missing the suffix `_nn`
final r2 = shouldRedirect2(buildId)
if( r2 ) {
log.debug "Redirect to build page [2]: $r2"
return HttpResponse.redirect(URI.create(r2))

// check all builds matching the pattern
if( isBuildMissingSuffix(buildId) ) {
final builds = buildService.getAllBuilds(buildId)
if( !builds ) {
log.debug "Found not build with id: $buildId"
throw new NotFoundException("Unknown container build id '$buildId'")
}
if( builds.size()==1 ) {
log.debug "Redirect to build page [2]: ${builds.first().buildId}"
return HttpResponse.temporaryRedirect(URI.create("/view/builds/${builds.first().buildId}"))
}
else
return HttpResponse.ok(new ModelAndView("build-list", renderBuildsView(builds)))
}

// go ahead with proper handling
final record = buildService.getBuildRecord(buildId)
if( !record )
throw new NotFoundException("Unknown container build id '$buildId'")
return HttpResponse.ok(renderBuildView(record))
return HttpResponse.ok(new ModelAndView("build-view", renderBuildView(record)))
}

static final private Pattern DASH_SUFFIX = ~/([0-9a-zA-Z\-]+)-(\d+)$/
static final private Pattern DASH_SUFFIX_REGEX = ~/([0-9a-zA-Z\-]+)-(\d+)$/

static final private Pattern MISSING_SUFFIX = ~/([0-9a-zA-Z\-]+)(?<!_\d{2})$/
static final private Pattern CONTAINER_ID_REGEX = ~/((bd-)?[0-9a-z\-]{16})(?<!_\d{2})$/

protected String shouldRedirect1(String buildId) {
protected String isBuildInvalidSuffix(String buildId) {
// check for build id containing a -nn suffix
final check1 = DASH_SUFFIX.matcher(buildId)
final check1 = DASH_SUFFIX_REGEX.matcher(buildId)
if( check1.matches() ) {
return "/view/builds/${check1.group(1)}_${check1.group(2)}"
}
return null
}

protected String shouldRedirect2(String buildId) {
protected boolean isBuildMissingSuffix(String buildId) {
// check build id missing the _nn suffix
if( !MISSING_SUFFIX.matcher(buildId).matches() )
return null

final rec = buildService.getLatestBuild(buildId)
if( !rec )
return null
if( !rec.buildId.contains(buildId) )
return null
if( rec.buildId==buildId )
return null

return "/view/builds/${rec.buildId}"
return buildId
? CONTAINER_ID_REGEX.matcher(buildId).matches()
: false
}

Map<String,?> renderBuildsView(List<WaveBuildRecord> results) {
// create template binding
final binding = new ArrayList<Map<String,String>>()
for (def result : results){
final bind = new HashMap(20)
bind.build_id = result.buildId
bind.build_image = result.targetImage
bind.build_status = getStatus(result)
bind.build_time = formatTimestamp(result.startTime, result.offsetId) ?: '-'
binding.add(bind)
}
// result the main object
return Map.of("build_records", binding, 'server_url', serverUrl)
}

protected static String getStatus(WaveBuildRecord result){
if( result.done() && result.succeeded() )
return "SUCCEEDED"
else if ( result.done() && !result.succeeded() )
return "FAILED"
else if (!result.done())
return "IN PROGRESS"
else
return "UNKNOWN"
}

Map<String,String> renderBuildView(WaveBuildRecord result) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -93,4 +93,12 @@ interface ContainerBuildService {
*/
WaveBuildRecord getLatestBuild(String containerId)

/**
* Retrieve the all build record available for the specified container id.
*
* @param containerId The ID of the container for which the build record needs to be retrieve
* @return The {@link WaveBuildRecord} associated with the corresponding Id, or {@code null} if it cannot be found
*/
List<WaveBuildRecord> getAllBuilds(String containerId)

}
Original file line number Diff line number Diff line change
Expand Up @@ -381,4 +381,12 @@ class ContainerBuildServiceImpl implements ContainerBuildService, JobHandler<Bui
return persistenceService.latestBuild(containerId)
}

/**
* {@inheritDoc}
*/
@Override
List<WaveBuildRecord> getAllBuilds(String containerId) {
return persistenceService.allBuilds(containerId)
}

}
Original file line number Diff line number Diff line change
Expand Up @@ -67,6 +67,14 @@ interface PersistenceService {
*/
WaveBuildRecord latestBuild(String containerId)

/**
* Retrieve all {@link WaveBuildRecord} object for the given container id
*
* @param containerId The container id for which the latest build record should be retrieved
* @return The corresponding {@link WaveBuildRecord} object or {@code null} if no record is found
*/
List<WaveBuildRecord> allBuilds(String containerId)

/**
* Store a {@link WaveContainerRecord} object in the Surreal wave_request table.
*
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -60,6 +60,15 @@ class LocalPersistenceService implements PersistenceService {
.reverse() [0]
}

@Override
List<WaveBuildRecord> allBuilds(String containerId) {
buildStore
.values()
.findAll( it-> it.buildId.contains(containerId) )
.sort { it.startTime }
.reverse()
}

@Override
WaveBuildRecord loadBuildSucceed(String targetImage, String digest) {
buildStore.values().find( (build) -> build.targetImage==targetImage && build.digest==digest && build.succeeded() )
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -182,6 +182,21 @@ class SurrealPersistenceService implements PersistenceService {
return result
}

@Override
List<WaveBuildRecord> allBuilds(String containerId) {
final query = """
select *
from wave_build
where buildId ~ '${containerId}'
order by startTime desc
""".stripIndent()
final json = surrealDb.sqlAsString(getAuthorization(), query)
final type = new TypeReference<ArrayList<SurrealResult<WaveBuildRecord>>>() {}
final data= json ? JacksonHelper.fromJson(json, type) : null
final result = data && data[0].result ? data[0].result : null
return result ? Arrays.asList(result) : null
}

@Override
void saveContainerRequest(WaveContainerRecord data) {
// note: use surreal sql in order to by-pass issue with large payload
Expand Down
100 changes: 100 additions & 0 deletions src/main/resources/io/seqera/wave/build-list.hbs
Original file line number Diff line number Diff line change
@@ -0,0 +1,100 @@
<html>
<head>
<meta charset="utf-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1">
<meta name="description" content="Wave container build data">
<title>Wave container builds</title>
<style>
@import url('https://fonts.googleapis.com/css2?family=Inter:ital,opsz,wght@0,14..32,100..900;1,14..32,100..900&family=Roboto+Mono:ital,wght@0,100..700;1,100..700&display=swap');
pre {
white-space: pre-wrap;
overflow-x: auto;
overflow-y: auto;
background-color: #ededed;
padding: 15px;
border-radius: 4px;
margin-bottom: 30px;
max-height: 400px;
max-width: 100%;
word-wrap: break-word;
}
.table-container table{
width: 100%;
border-collapse: collapse;
}
.table-container th, .table-container td {
padding: 10px;
text-align: left;
border-bottom: 1px solid #e5e7eb;
vertical-align: middle;
}
.table-container th {
font-weight: bold;
background-color: #f1f5f9;
}
.table-container td {
font-size: 14px;
}
.table-container a {
background-color: #f1f5f9;
text-decoration: none;
padding: 4px 8px;
border-radius: 4px;
display: inline-block;
}
.table-container a:hover {
text-decoration: underline;
}
</style>
</head>
<body>
<div style="font-family:'-apple-system', BlinkMacSystemFont, 'Segoe UI', Helvetica, Arial, sans-serif, 'Apple Color Emoji', 'Segoe UI Emoji', 'Segoe UI Symbol'; padding: 30px; max-width: 800px; margin: 0 auto;">
<div>
<img style="float:left; margin-top: 5px; margin-right: 10px; width: 30px;" src="">
<h1>Wave container build records</h1>
</div>
{{#if build_records}}
<div class="table-container">
<table cellpadding="4" >
<tr>
<th>Build ID</th>
<th>Container image</th>
<th>Build time</th>
<th>Status</th>
</tr>
{{#each build_records}}
<tr>
<td><a href="{{server_url}}/view/builds/{{build_id}}">{{build_id}}</td>
<td>{{build_image}}</td>
<td>{{build_time}}</td>
<td>{{build_status}}</td>
</tr>
{{/each}}
</table>
</div>
{{else}}
<pre> No Build Record Found</pre>
{{/if}}

<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="">
<p class="footer-text" style="font-family:'-apple-system', BlinkMacSystemFont, 'Segoe UI', Helvetica, Arial, sans-serif, 'Apple Color Emoji', 'Segoe UI Emoji', 'Segoe UI Symbol';font-weight:normal;margin:0;margin-bottom:15px;color:#999999;font-size:12px;line-height:18px;">
<a href="{{server_url}}">{{server_url}}</a><br>
Seqera<br>
Carrer de Marià Aguiló, 28<br>
08005 Barcelona<br>
</p>
</div>

</div>
</body>
</html>
4 changes: 4 additions & 0 deletions src/main/resources/io/seqera/wave/build-view.hbs
Original file line number Diff line number Diff line change
Expand Up @@ -123,7 +123,11 @@

<tr>
<td>Container image</td>
{{#if build_in_progress}}
<td>{{build_image}}</td>
{{else}}
<td><a href="{{inspect_url}}">{{build_image}}</a></td>
{{/if}}
</tr>

<tr>
Expand Down
Loading

0 comments on commit 3cabb83

Please sign in to comment.