Skip to content

Commit

Permalink
Add scan history page (#704)
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 18, 2024
1 parent 1394871 commit 89da55d
Show file tree
Hide file tree
Showing 9 changed files with 330 additions and 7 deletions.
71 changes: 65 additions & 6 deletions src/main/groovy/io/seqera/wave/controller/ViewController.groovy
Original file line number Diff line number Diff line change
Expand Up @@ -288,9 +288,28 @@ class ViewController {
return HttpResponse.<Map<String,Object>>ok(binding)
}

@View("scan-view")
@Get('/scans/{scanId}')
HttpResponse<Map<String,Object>> viewScan(String scanId) {
HttpResponse viewScan(String scanId) {
// check redirection for invalid suffix in the form `-nn`
final r1 = isScanInvalidSuffix(scanId)
if( r1 ) {
log.debug "Redirect to scan page [1]: $r1"
return HttpResponse.redirect(URI.create(r1))
}
// check all scans matching the pattern
if( isScanMissingSuffix(scanId) ) {
final scans = scanService.getAllScans(scanId)
if( !scans ) {
log.debug "Found not scan with id: $scanId"
throw new NotFoundException("Unknown container scan id '$scanId'")
}
if( scans.size()==1 ) {
log.debug "Redirect to scan page [2]: ${scans.first().id}"
return HttpResponse.temporaryRedirect(URI.create("/view/scans/${scans.first().id}"))
}
else
return HttpResponse.ok(new ModelAndView("scan-list", renderScansView(scans)))
}
final binding = new HashMap(10)
try {
final result = loadScanRecord(scanId)
Expand All @@ -306,15 +325,55 @@ class ViewController {

// return the response
binding.put('server_url', serverUrl)
return HttpResponse.<Map<String,Object>>ok(binding)
return HttpResponse.ok(new ModelAndView("scan-view", binding))
}

Map<String,?> renderScansView(List<WaveScanRecord> results) {
// create template binding
final bindingMap = new HashMap<String, ?>(3)
if( results ) {
bindingMap.put("scan_container_image", results[0].containerImage)
final binding = new ArrayList<Map<String,String>>()
for (def result : results) {
final bind = new HashMap(20)
bind.scan_id = result.id
bind.scan_status = result.status
bind.scan_time = formatTimestamp(result.startTime) ?: '-'
bind.scan_vuls_count = result.status == 'SUCCEEDED' ? result.vulnerabilities.size() : '-'
binding.add(bind)
}
bindingMap.put('scan_records', binding)
}
// result the main object
bindingMap.put('server_url', serverUrl)
return bindingMap
}

protected String isScanInvalidSuffix(String scanId) {
// check for scan id containing a -nn suffix
final check1 = DASH_SUFFIX_REGEX.matcher(scanId)
if( check1.matches() ) {
return "/view/scans/${check1.group(1)}_${check1.group(2)}"
}
return null
}

static final private Pattern SCAN_ID_REGEX = ~/((sc-)?[0-9a-z\-]{14,16})(?<!_\d{2})$/

protected boolean isScanMissingSuffix(String scanId) {
// check scan id missing the _nn suffix
return scanId
? SCAN_ID_REGEX.matcher(scanId).matches()
: false
}


/**
* Retrieve a {@link ScanEntry} object for the specified build ID
* Retrieve a {@link ScanEntry} object for the specified scan ID
*
* @param buildId The ID of the build for which load the scan result
* @param scanId The ID of the scan for which load the scan result
* @return The {@link ScanEntry} object associated with the specified build ID or throws the exception {@link NotFoundException} otherwise
* @throws NotFoundException If the a record for the specified build ID cannot be found
* @throws NotFoundException If the a record for the specified scan ID cannot be found
*/
protected WaveScanRecord loadScanRecord(String scanId) {
if( !scanService )
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -70,7 +70,7 @@ interface PersistenceService {
/**
* Retrieve all {@link WaveBuildRecord} object for the given container id
*
* @param containerId The container id for which the latest build record should be retrieved
* @param containerId The container id for which all the builds record should be retrieved
* @return The corresponding {@link WaveBuildRecord} object or {@code null} if no record is found
*/
List<WaveBuildRecord> allBuilds(String containerId)
Expand Down Expand Up @@ -146,4 +146,12 @@ interface PersistenceService {
*/
void saveMirrorResult(MirrorResult mirror)

/**
* Retrieve all {@link WaveScanRecord} object for the given partial scan id
*
* @param scanId The scan id for which all the scan records should be retrieved
* @return The corresponding {@link WaveScanRecord} object or {@code null} if no record is found
*/
List<WaveScanRecord> allScans(String scanId)

}
Original file line number Diff line number Diff line change
Expand Up @@ -108,6 +108,16 @@ class LocalPersistenceService implements PersistenceService {
scanStore.get(scanId)
}

@Override
List<WaveScanRecord> allScans(String scanId) {
final pattern = ~/^.*(sc-)?${scanId}_[0-9]+/
scanStore
.values()
.findAll( it-> pattern.matcher(it.id).matches() )
.sort { it.startTime }
.reverse()
}

@Override
MirrorResult loadMirrorResult(String mirrorId) {
mirrorStore.get(mirrorId)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -304,6 +304,22 @@ class SurrealPersistenceService implements PersistenceService {
return result
}

@Override
List<WaveScanRecord> allScans(String scanId) {
final query = """
select *
from wave_scan
where string::matches(type::string(id), '^wave_scan:⟨(sc-)?${scanId}_[0-9]+')
order by startTime desc
FETCH vulnerabilities
""".stripIndent()
final json = surrealDb.sqlAsString(getAuthorization(), query)
final type = new TypeReference<ArrayList<SurrealResult<WaveScanRecord>>>() {}
final data= json ? JacksonHelper.fromJson(json, type) : null
final result = data && data[0].result ? data[0].result : null
return result ? Arrays.asList(result) : null
}

// === mirror operations

/**
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -46,4 +46,6 @@ interface ContainerScanService {

ScanEntry getScanState(String scanId)

List<WaveScanRecord> getAllScans(String scanId)

}
Original file line number Diff line number Diff line change
Expand Up @@ -292,4 +292,8 @@ class ContainerScanServiceImpl implements ContainerScanService, JobHandler<ScanE
}
}

List<WaveScanRecord> getAllScans(String scanId){
persistenceService.allScans(scanId)
}

}
108 changes: 108 additions & 0 deletions src/main/resources/io/seqera/wave/scan-list.hbs
Original file line number Diff line number Diff line change
@@ -0,0 +1,108 @@
<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 scan data">
<title>Wave container scans</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 scan history</h1>
</div>
{{#if scan_records}}
<h3>Container</h3>
<table cellpadding="4" >
<tr>
<td>Image</td>
<td>{{scan_container_image}}</td>
</tr>
</table>
<div class="table-container">
<h3>History</h3>
<table cellpadding="4" >
<tr>
<th>Scan ID</th>
<th>Scan time</th>
<th>Status</th>
<th>Number of CVEs</th>
</tr>
{{#each scan_records}}
<tr>
<td><a href="{{server_url}}/view/scans/{{scan_id}}">{{scan_id}}</td>
<td>{{scan_time}}</td>
<td>{{scan_status}}</td>
<td>{{scan_vuls_count}}</td>
</tr>
{{/each}}
</table>
</div>
{{else}}
<pre> No Scan 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>
Loading

0 comments on commit 89da55d

Please sign in to comment.