Duplicating this data on every record of the builds table cost approximately 4G of duplication. Note that the database migration included took about 4h45m on an untuned server which uses very slow rotational disks in a RAID5 setup, with not a lot of RAM. I imagine in production it might take an hour or two, but not 4. If this should become a chunked migration, I can do that. Note: Because of the question about chunked migrations, I have NOT YET tested this migration thoroughly enough for merge.
539 lines
20 KiB
Plaintext
539 lines
20 KiB
Plaintext
[% WRAPPER layout.tt title="Build $id of job $project.name:$jobset.name:$job" %]
|
|
[% PROCESS common.tt %]
|
|
[% PROCESS "product-list.tt" %]
|
|
[% USE HTML %]
|
|
[% USE Date %]
|
|
|
|
[%
|
|
isAggregate = constituents.size > 0;
|
|
busy = 0;
|
|
building = 0;
|
|
FOR step IN steps;
|
|
IF step.busy;
|
|
busy = 1;
|
|
IF step.drvpath == build.drvpath; building = 1; END;
|
|
END;
|
|
END;
|
|
%]
|
|
|
|
[% BLOCK renderOutputs %]
|
|
[% start=1; FOREACH output IN outputs %]
|
|
[% IF !start %],<br/>[% END; start=0; output.path %]
|
|
[% END %]
|
|
[% END %]
|
|
|
|
[% BLOCK renderBuildSteps %]
|
|
<table class="table table-striped table-condensed clickable-rows">
|
|
<thead>
|
|
<tr><th>Nr</th><th>What</th><th>Duration</th><th>Machine</th><th>Status</th></tr>
|
|
</thead>
|
|
<tbody>
|
|
[% FOREACH step IN steps %]
|
|
[% IF ( type == "All" ) || ( type == "Failed" && step.busy == 0 && step.status != 0 ) || ( type == "Running" && step.busy != 0 ) %]
|
|
[% has_log = seen.${step.drvpath} ? 0 : buildStepLogExists(step);
|
|
seen.${step.drvpath} = 1;
|
|
log = c.uri_for('/build' build.id 'nixlog' step.stepnr); %]
|
|
<tr>
|
|
<td>[% step.stepnr %]</td>
|
|
<td>
|
|
[% IF step.type == 0 %]
|
|
Build of <tt>[% INCLUDE renderOutputs outputs=step.buildstepoutputs %]</tt>
|
|
[% ELSE %]
|
|
Substitution of <tt>[% INCLUDE renderOutputs outputs=step.buildstepoutputs %]</tt>
|
|
[% END %]
|
|
</td>
|
|
<td>
|
|
[% IF step.busy == 0;
|
|
IF step.stoptime;
|
|
INCLUDE renderDuration duration = step.stoptime - step.starttime;
|
|
ELSE;
|
|
%]<em>n/a</em>[%
|
|
END;
|
|
ELSIF build.finished;
|
|
INCLUDE renderDuration duration = build.stoptime - step.starttime;
|
|
ELSE;
|
|
INCLUDE renderDuration duration = curTime - step.starttime;
|
|
END %]
|
|
</td>
|
|
<td>[% IF step.busy != 0 || ((step.machine || step.starttime) && (step.status == 0 || step.status == 1 || step.status == 3 || step.status == 4 || step.status == 7)); INCLUDE renderMachineName machine=step.machine; ELSE; "<em>n/a</em>"; END %]</td>
|
|
<td class="step-status">
|
|
[% IF step.busy != 0 %]
|
|
[% IF step.busy == 1 %]
|
|
<strong>Preparing</strong>
|
|
[% ELSIF step.busy == 10 %]
|
|
<strong>Connecting</strong>
|
|
[% ELSIF step.busy == 20 %]
|
|
<strong>Sending inputs</strong>
|
|
[% ELSIF step.busy == 30 %]
|
|
<strong>Building</strong>
|
|
[% ELSIF step.busy == 40 %]
|
|
<strong>Receiving outputs</strong>
|
|
[% ELSIF step.busy == 50 %]
|
|
<strong>Post-processing</strong>
|
|
[% ELSE %]
|
|
<strong>Unknown state</strong>
|
|
[% END %]
|
|
[% ELSIF step.status == 0 %]
|
|
[% IF step.isnondeterministic %]
|
|
<span class="warn">Succeeded with non-determistic result</span>
|
|
[% ELSE %]
|
|
Succeeded
|
|
[% END %]
|
|
[% IF step.timesbuilt && step.timesbuilt > 1 %]
|
|
([% step.timesbuilt %] times)
|
|
[% END %]
|
|
[% ELSIF step.status == 3 %]
|
|
<span class="error">Aborted</span>[% IF step.errormsg %]: <em>[% HTML.escape(step.errormsg) %]</em>[% END %]
|
|
[% ELSIF step.status == 4 %]
|
|
<span class="error">Cancelled</span>
|
|
[% ELSIF step.status == 7 %]
|
|
<span class="error">Timed out</span>
|
|
[% ELSIF step.status == 8 %]
|
|
<span class="error">Cached failure</span>
|
|
[% ELSIF step.status == 9 %]
|
|
<span class="error">Unsupported system type</span>
|
|
[% ELSIF step.status == 10 %]
|
|
<span class="error">Log limit exceeded</span>
|
|
[% ELSIF step.status == 11 %]
|
|
<span class="error">Output limit exceeded</span>
|
|
[% ELSIF step.status == 12 %]
|
|
<span class="error">Non-determinism detected</span> [% IF step.timesbuilt %] after [% step.timesbuilt %] times[% END %]
|
|
[% ELSIF step.errormsg %]
|
|
<span class="error">Failed</span>: <em>[% HTML.escape(step.errormsg) %]</em>
|
|
[% ELSE %]
|
|
<span class="error">Failed</span>
|
|
[% END %]
|
|
[%%] [%+ IF has_log; INCLUDE renderLogLinks url=log inRow=1; END %]
|
|
[%+ IF step.propagatedfrom; %](propagated from [% INCLUDE renderBuildIdLink id=step.propagatedfrom.get_column('id') %])[% END %]
|
|
</td>
|
|
</tr>
|
|
[% END %]
|
|
[% END %]
|
|
</tbody>
|
|
</table>
|
|
[% END %]
|
|
|
|
<ul class="nav nav-tabs">
|
|
<li class="dropdown">
|
|
<a class="dropdown-toggle actions" data-toggle="dropdown" href="#">
|
|
Actions
|
|
<b class="caret"></b>
|
|
</a>
|
|
<ul class="dropdown-menu">
|
|
[% IF eval.nixexprinput || eval.flake %]
|
|
<li><a href="#reproduce" data-toggle="modal">Reproduce locally</a></li>
|
|
[% END %]
|
|
[% IF c.user_exists %]
|
|
[% IF available %]
|
|
[% IF build.keep %]
|
|
<li><a href="[% c.uri_for('/build' build.id 'keep' 0) %]">Unkeep</a></li>
|
|
[% ELSE %]
|
|
<li><a href="[% c.uri_for('/build' build.id 'keep' 1) %]">Keep</a></li>
|
|
[% END %]
|
|
[% END %]
|
|
[% IF build.finished %]
|
|
<li><a href="[% c.uri_for('/build' build.id 'restart') %]">Restart</a></li>
|
|
[% ELSE %]
|
|
<li><a href="[% c.uri_for('/build' build.id 'cancel') %]">Cancel</a></li>
|
|
<li><a href="[% c.uri_for('/build' build.id 'bump') %]">Bump up</a></li>
|
|
[% END %]
|
|
[% END %]
|
|
</ul>
|
|
</li>
|
|
|
|
<li class="active"><a href="#tabs-summary" data-toggle="tab">Summary</a></li>
|
|
[% IF isAggregate %]<li><a href="#tabs-constituents" data-toggle="tab">Constituents</a></li>[% END %]
|
|
<li><a href="#tabs-details" data-toggle="tab">Details</a></li>
|
|
<li><a href="#tabs-buildinputs" data-toggle="tab">Inputs</a></li>
|
|
[% IF steps.size() > 0 %]<li><a href="#tabs-buildsteps" data-toggle="tab">Build steps</a></li>[% END %]
|
|
[% IF build.dependents %]<li><a href="#tabs-usedby" data-toggle="tab">Used by</a></li>[% END%]
|
|
[% IF drvAvailable %]<li><a href="#tabs-build-deps" data-toggle="tab">Build dependencies</a></li>[% END %]
|
|
[% IF localStore && available %]<li><a href="#tabs-runtime-deps" data-toggle="tab">Runtime dependencies</a></li>[% END %]
|
|
</ul>
|
|
|
|
<div id="generic-tabs" class="tab-content">
|
|
|
|
<div id="tabs-summary" class="tab-pane active">
|
|
|
|
<table>
|
|
<tr>
|
|
<td>
|
|
[% INCLUDE renderBuildStatusIcon size=128 build=build %]
|
|
</td>
|
|
<td>
|
|
<table class="info-table">
|
|
<tr>
|
|
<th>Build ID:</th>
|
|
<td>[% build.id %]</td>
|
|
</tr>
|
|
<tr>
|
|
<th>Status:</th>
|
|
<td>
|
|
[% INCLUDE renderStatus build=build icon=0 busy=busy %]
|
|
[% IF isAggregate;
|
|
nrConstituents = 0;
|
|
nrFinished = 0;
|
|
nrFailedConstituents = 0;
|
|
FOREACH b IN constituents;
|
|
nrConstituents = nrConstituents + 1;
|
|
IF b.finished; nrFinished = nrFinished + 1; END;
|
|
IF b.finished && b.buildstatus != 0; nrFailedConstituents = nrFailedConstituents + 1; END;
|
|
END;
|
|
%];
|
|
[%+ IF nrFinished == nrConstituents && nrFailedConstituents == 0 %]
|
|
all [% nrConstituents %] constituent builds succeeded
|
|
[% ELSE %]
|
|
[% nrFailedConstituents %] out of [% nrConstituents %] constituent builds failed
|
|
[% IF nrFinished < nrConstituents %]
|
|
([% nrConstituents - nrFinished %] still pending)
|
|
[% END %]
|
|
[% END %]
|
|
[% END %]
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<th>System:</th>
|
|
<td><tt>[% build.system %]</tt></td>
|
|
</tr>
|
|
[% IF build.releasename %]
|
|
<tr>
|
|
<th>Release name:</th>
|
|
<td><tt>[% HTML.escape(build.releasename) %]</tt></td>
|
|
</tr>
|
|
[% ELSE %]
|
|
<tr>
|
|
<th>Nix name:</th>
|
|
<td><tt>[% build.nixname %]</tt></td>
|
|
</tr>
|
|
[% END %]
|
|
[% IF eval %]
|
|
<tr>
|
|
<th>Part of:</th>
|
|
<td>
|
|
<a href="[% c.uri_for(c.controller('JobsetEval').action_for('view'), [eval.id]) %]">evaluation [% eval.id %]</a>
|
|
[% IF nrEvals > 1 +%] (and <a href="[% c.uri_for('/build' build.id 'evals') %]">[% nrEvals - 1 %] others</a>)[% END %]
|
|
</td>
|
|
</tr>
|
|
[% END %]
|
|
[% IF build.iscachedbuild %]
|
|
<tr>
|
|
<th>Cached from:</th>
|
|
<td>[% IF cachedBuild; INCLUDE renderFullBuildLink build=cachedBuild; ELSE %]<em>unknown</em>[% END %]</td>
|
|
</tr>
|
|
[% END %]
|
|
[% actualBuild = build.iscachedbuild ? cachedBuild : build %]
|
|
[% IF (!isAggregate || !build.ischannel) && build.finished; %]
|
|
[% IF actualBuild %]
|
|
<tr>
|
|
<th>Duration:</th>
|
|
<td>[% INCLUDE renderDuration duration = actualBuild.stoptime - actualBuild.starttime %]</td>
|
|
</tr>
|
|
[% END %]
|
|
<tr>
|
|
<th>Finished at:</th>
|
|
<td>[% INCLUDE renderDateTime timestamp = build.stoptime; %]</td>
|
|
</tr>
|
|
[% END %]
|
|
[% IF (!build.finished && building) || (build.finished && (!isAggregate || !build.ischannel) && buildLogExists(build)) %]
|
|
<tr>
|
|
<th>Logfile:</th>
|
|
<td>
|
|
[% actualLog = cachedBuildStep ? c.uri_for('/build' cachedBuild.id 'nixlog' cachedBuildStep.stepnr) : c.uri_for('/build' build.id 'log') %]
|
|
<a class="btn btn-mini" href="[%actualLog%]">pretty</a>
|
|
<a class="btn btn-mini" href="[%actualLog%]/raw">raw</a>
|
|
<a class="btn btn-mini" href="[%actualLog%]/tail">tail</a>
|
|
</td>
|
|
</tr>
|
|
[% END %]
|
|
</table>
|
|
</td>
|
|
</tr>
|
|
</table>
|
|
|
|
[% IF build.ischannel || (build.buildproducts && !isAggregate) %]
|
|
|
|
<h3>Build products</h3>
|
|
|
|
[% IF !available %]
|
|
<p class="error">Note: this build is no longer available.</p>
|
|
[% END %]
|
|
|
|
[% INCLUDE renderProductList latestRoot=['/job' build.project.name build.jobset.name build.job 'latest'] %]
|
|
|
|
[% END %]
|
|
|
|
[% IF busy %]
|
|
<h3>Running build steps</h3>
|
|
[% INCLUDE renderBuildSteps type="Running" %]
|
|
[% END %]
|
|
|
|
[% IF build.finished %]
|
|
|
|
[% IF steps && build.buildstatus != 0 && build.buildstatus != 4 && build.buildstatus != 6 %]
|
|
<h3>Failed build steps</h3>
|
|
[% INCLUDE renderBuildSteps type="Failed" %]
|
|
[% END %]
|
|
|
|
[% IF otherEval %]
|
|
<h3>Changes</h3>
|
|
[% INCLUDE renderInputDiff inputs2=eval.jobsetevalinputs inputs1=otherEval.jobsetevalinputs %]
|
|
[% END %]
|
|
|
|
[% IF prevSuccessfulBuild %]
|
|
<h3>Previous builds</h3>
|
|
<table class="table table-striped table-condensed">
|
|
<thead>
|
|
<th>Last successful build [% INCLUDE renderDateTime timestamp = prevSuccessfulBuild.timestamp %]</th>
|
|
[% IF prevSuccessfulBuild && firstBrokenBuild && firstBrokenBuild.id != build.id %]
|
|
<th>First broken build [% INCLUDE renderDateTime timestamp = firstBrokenBuild.timestamp %]
|
|
</th>
|
|
[% END %]
|
|
<th>This build [% INCLUDE renderDateTime timestamp = build.timestamp %]
|
|
</th>
|
|
</thead>
|
|
<tr>
|
|
<td valign="center">[% INCLUDE renderBuildStatusIcon build=prevSuccessfulBuild size=32 %] [% INCLUDE renderBuildLink build=prevSuccessfulBuild %]</td>
|
|
[% IF prevSuccessfulBuild && firstBrokenBuild && firstBrokenBuild.id != build.id %]
|
|
<td valign="center">[% INCLUDE renderBuildStatusIcon build=firstBrokenBuild size=32 %] [% INCLUDE renderBuildLink build=firstBrokenBuild %]</td>
|
|
[% END %]
|
|
<td>[% INCLUDE renderBuildStatusIcon build=build size=32 %] [% INCLUDE renderBuildLink build=build %]</td>
|
|
</tr>
|
|
<tr>
|
|
<td></td>
|
|
[% IF prevSuccessfulBuild && firstBrokenBuild && firstBrokenBuild.id != build.id %]
|
|
<td>[% INCLUDE renderInputDiff inputs1=prevSuccessfulBuild.inputs inputs2=firstBrokenBuild.inputs %]</td>
|
|
[% END %]
|
|
</tr>
|
|
</table>
|
|
[% END %]
|
|
|
|
[% END %]
|
|
|
|
</div>
|
|
|
|
[% IF isAggregate %]
|
|
|
|
<div id="tabs-constituents" class="tab-pane">
|
|
|
|
<p>This build is an aggregate of the following builds:</p>
|
|
|
|
[% INCLUDE renderBuildList builds=constituents hideProjectName=1 hideJobsetName=1 %]
|
|
|
|
</div>
|
|
|
|
[% END %]
|
|
|
|
<div id="tabs-details" class="tab-pane">
|
|
|
|
<table class="info-table">
|
|
<tr>
|
|
<th>Queued at:</th>
|
|
<td>[% INCLUDE renderDateTime timestamp = build.timestamp %]</td>
|
|
</tr>
|
|
[% IF build.finished && !build.iscachedbuild %]
|
|
<tr>
|
|
<th>Build started:</th>
|
|
<td>[% INCLUDE renderDateTime timestamp = build.starttime %]</td>
|
|
</tr>
|
|
<tr>
|
|
<th>Build finished:</th>
|
|
<td>[% INCLUDE renderDateTime timestamp = build.stoptime %]</td>
|
|
</tr>
|
|
[% END %]
|
|
[% IF !build.finished %]
|
|
<tr>
|
|
<th>Priority:</th>
|
|
<td>[% build.priority %]</td>
|
|
</tr>
|
|
[% END %]
|
|
[% IF eval.nixexprinput %]
|
|
<tr>
|
|
<th>Nix expression:</th>
|
|
<td>file <tt>[% HTML.escape(eval.nixexprpath) %]</tt> in input <tt>[% HTML.escape(eval.nixexprinput) %]</tt></td>
|
|
</tr>
|
|
[% END %]
|
|
<tr>
|
|
<th>Nix name:</th>
|
|
<td><tt>[% build.nixname %]</tt></td>
|
|
</tr>
|
|
<tr>
|
|
<th>Short description:</th>
|
|
<td>[% IF build.description %][% HTML.escape(build.description) %][% ELSE %]<em>not given</em>[% END %]</td>
|
|
</tr>
|
|
<tr>
|
|
<th>License:</th>
|
|
<td>[% IF build.license %][% HTML.escape(build.license) %][% ELSE %]<em>not given</em>[% END %]</td>
|
|
</tr>
|
|
<tr>
|
|
<th>Homepage:</th>
|
|
<td>[% IF build.homepage %]<a [% HTML.attributes(href => build.homepage) %]>[% HTML.escape(build.homepage) %]</a>[% ELSE %]<em>not given</em>[% END %]</td>
|
|
</tr>
|
|
<tr>
|
|
<th>Maintainer(s):</th>
|
|
<td>[% IF build.maintainers %][% HTML.escape(build.maintainers) %][% ELSE %]<em>not given</em>[% END %]</td>
|
|
</tr>
|
|
<tr>
|
|
<th>System:</th>
|
|
<td><tt>[% build.system %]</tt></td>
|
|
</tr>
|
|
<tr>
|
|
<th>Derivation store path:</th>
|
|
<td><tt>[% build.drvpath %]</tt></td>
|
|
</tr>
|
|
<tr>
|
|
<th>Output store paths:</th>
|
|
<td><tt>[% INCLUDE renderOutputs outputs=build.buildoutputs %]</tt></td>
|
|
</tr>
|
|
[% chartsURL = c.uri_for('/job' build.project.name build.jobset.name build.job) _ "#tabs-charts" %]
|
|
[% IF build.finished && build.closuresize %]
|
|
<tr>
|
|
<th>Closure size:</th>
|
|
<td>[% mibs(build.closuresize / (1024 * 1024)) %] MiB
|
|
(<a href="[%chartsURL%]">history</a>)</td>
|
|
</tr>
|
|
[% END %]
|
|
[% IF build.finished && build.closuresize %]
|
|
<tr>
|
|
<th>Output size:</th>
|
|
<td>[% mibs(build.size / (1024 * 1024)) %] MiB
|
|
(<a href="[%chartsURL%]">history</a>)</td>
|
|
</tr>
|
|
[% END %]
|
|
[% IF build.finished && build.buildproducts %]
|
|
<tr>
|
|
<th>Availability:</th>
|
|
<td>
|
|
[% IF !available %]
|
|
<em>Build output is no longer available</em>
|
|
[% ELSIF build.keep %]
|
|
<em>Build output will be kept permanently</em>
|
|
[% ELSE %]
|
|
<em>Build output is available, but may be garbage-collected</em>
|
|
[% END %]
|
|
</td>
|
|
</tr>
|
|
[% END %]
|
|
</table>
|
|
|
|
[% IF build.finished && build.buildmetrics %]
|
|
<h3>Metrics</h3>
|
|
|
|
<table class="table table-small table-striped table-hover clickable-rows">
|
|
<thead>
|
|
<tr><th>Name</th><th>Value</th><th></th></tr>
|
|
</thead>
|
|
<tbody>
|
|
[% FOREACH metric IN build.buildmetrics %]
|
|
<tr>
|
|
<td><tt><a class="row-link" [% HTML.attributes(href => c.uri_for('/job' project.name jobset.name job 'metric' metric.name)) %]">[%HTML.escape(metric.name)%]</a></tt></td>
|
|
<td style="text-align: right">[%metric.value%]</td>
|
|
<td>[%metric.unit%]</td>
|
|
</tr>
|
|
[% END %]
|
|
</tbody>
|
|
</table>
|
|
[% END %]
|
|
|
|
</div>
|
|
|
|
<div id="tabs-buildinputs" class="tab-pane">
|
|
|
|
[% IF build.inputs && build.inputs.size > 0 %]
|
|
|
|
[% INCLUDE renderInputs inputs=build.inputs %]
|
|
|
|
[% ELSIF eval %]
|
|
|
|
[% INCLUDE renderInputs inputs=eval.jobsetevalinputs %]
|
|
|
|
[% END %]
|
|
|
|
</div>
|
|
|
|
[% IF steps %]
|
|
<div id="tabs-buildsteps" class="tab-pane">
|
|
[% INCLUDE renderBuildSteps type="All" %]
|
|
</div>
|
|
[% END %]
|
|
|
|
[% IF build.dependents %]
|
|
<div id="tabs-usedby" class="tab-pane">
|
|
|
|
<p>The following builds have used this build as an input:</p>
|
|
|
|
<table class="table table-condensed table-striped">
|
|
<thead>
|
|
<tr><th>Build</th><th>Input name</th><th>System</th><th>Timestamp</th></tr>
|
|
</thead>
|
|
<tbody>
|
|
[% FOREACH input IN build.dependents %]
|
|
<tr>
|
|
<td>[% INCLUDE renderFullBuildLink build=input.build %]</td>
|
|
<td><tt>[% input.name %]</tt></td>
|
|
<td><tt>[% input.build.system %]</tt></td>
|
|
<td>[% INCLUDE renderDateTime timestamp = input.build.timestamp %]</td>
|
|
</tr>
|
|
[% END %]
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
[% END %]
|
|
|
|
[% IF drvAvailable %]
|
|
[% INCLUDE makeLazyTab tabName="tabs-build-deps" uri=c.uri_for('/build' build.id 'build-deps') %]
|
|
[% END %]
|
|
|
|
[% IF available %]
|
|
[% INCLUDE makeLazyTab tabName="tabs-runtime-deps" uri=c.uri_for('/build' build.id 'runtime-deps') %]
|
|
[% END %]
|
|
|
|
</div>
|
|
|
|
|
|
<div id="reproduce" class="modal hide fade" tabindex="-1" role="dialog" aria-hidden="true">
|
|
[% url = c.uri_for('/build' build.id 'reproduce') %]
|
|
|
|
<div class="modal-header">
|
|
<button type="button" class="close" data-dismiss="modal" aria-hidden="true">×</button>
|
|
<h3>Reproduce this build</h3>
|
|
</div>
|
|
|
|
<div class="modal-body">
|
|
|
|
[% IF eval.flake %]
|
|
|
|
<p>If you have <a href='https://nixos.org/nix/download.html'>Nix
|
|
installed</a>, you can reproduce this build on your own machine by
|
|
running the following command:</p>
|
|
|
|
<pre>
|
|
<span class="shell-prompt"># </span>nix build [% HTML.escape(eval.flake) %]#hydraJobs.[% HTML.escape(job) %]
|
|
</pre>
|
|
|
|
[% ELSE %]
|
|
|
|
<p>If you have <a href='https://nixos.org/nix/download.html'>Nix
|
|
installed</a>, you can reproduce this build on your own machine by
|
|
downloading <a [% HTML.attributes(href => url) %]>a script</a>
|
|
that checks out all inputs of the build and then invokes Nix to
|
|
perform the build.</p>
|
|
|
|
<p>To download and execute the script from the command line, run the
|
|
following command:</p>
|
|
|
|
<pre>
|
|
<span class="shell-prompt"># </span>curl <a [% HTML.attributes(href => url) %]>[% HTML.escape(url) %]</a> | bash
|
|
</pre>
|
|
|
|
[% END %]
|
|
|
|
</div>
|
|
|
|
<div class="modal-footer">
|
|
<a href="#" class="btn btn-primary" data-dismiss="modal">Close</a>
|
|
</div>
|
|
</div>
|
|
|
|
|
|
[% END %]
|