schedulejs
Version:
Schedule tasks, meetings, appointments, etc
342 lines (295 loc) • 149 kB
HTML
make[1]: Entering directory `/home/bill/dev/schedule'
<!DOCTYPE html><html><head><title>Coverage</title><script>
headings = [];
onload = function(){
headings = document.querySelectorAll('h2');
};
onscroll = function(e){
var heading = find(window.scrollY);
if (!heading) return;
var links = document.querySelectorAll('#menu a')
, link;
for (var i = 0, len = links.length; i < len; ++i) {
link = links[i];
link.className = link.getAttribute('href') == '#' + heading.id
? 'active'
: '';
}
};
function find(y) {
var i = headings.length
, heading;
while (i--) {
heading = headings[i];
if (y >= heading.offsetTop) {
return heading;
}
}
}
</script>
<style>
body {
font: 14px/1.6 "Helvetica Neue", Helvetica, Arial, sans-serif;
margin: 0;
color: #2C2C2C;
border-top: 2px solid #ddd;
}
#coverage {
padding: 60px;
}
h1 a {
color: inherit;
font-weight: inherit;
}
h1 a:hover {
text-decoration: none;
}
.onload h1 {
opacity: 1;
}
h2 {
width: 80%;
margin-top: 80px;
margin-bottom: 0;
font-weight: 100;
letter-spacing: 1px;
border-bottom: 1px solid #eee;
}
a {
color: #8A6343;
font-weight: bold;
text-decoration: none;
}
a:hover {
text-decoration: underline;
}
ul {
margin-top: 20px;
padding: 0 15px;
width: 100%;
}
ul li {
float: left;
width: 40%;
margin-top: 5px;
margin-right: 60px;
list-style: none;
border-bottom: 1px solid #eee;
padding: 5px 0;
font-size: 12px;
}
ul::after {
content: '.';
height: 0;
display: block;
visibility: hidden;
clear: both;
}
code {
font: 12px monaco, monospace;
}
pre {
margin: 30px;
padding: 30px;
border: 1px solid #eee;
border-bottom-color: #ddd;
-webkit-border-radius: 2px;
-moz-border-radius: 2px;
-webkit-box-shadow: inset 0 0 10px #eee;
-moz-box-shadow: inset 0 0 10px #eee;
overflow-x: auto;
}
img {
margin: 30px;
padding: 1px;
-webkit-border-radius: 3px;
-moz-border-radius: 3px;
-webkit-box-shadow: 0 3px 10px #dedede, 0 1px 5px #888;
-moz-box-shadow: 0 3px 10px #dedede, 0 1px 5px #888;
max-width: 100%;
}
footer {
background: #eee;
width: 100%;
padding: 50px 0;
text-align: right;
border-top: 1px solid #ddd;
}
footer span {
display: block;
margin-right: 30px;
color: #888;
font-size: 12px;
}
#menu {
position: fixed;
font-size: 12px;
overflow-y: auto;
top: 0;
right: 0;
margin: 0;
height: 100%;
padding: 15px 0;
text-align: right;
border-left: 1px solid #eee;
-moz-box-shadow: 0 0 2px #888
, inset 5px 0 20px rgba(0,0,0,.5)
, inset 5px 0 3px rgba(0,0,0,.3);
-webkit-box-shadow: 0 0 2px #888
, inset 5px 0 20px rgba(0,0,0,.5)
, inset 5px 0 3px rgba(0,0,0,.3);
-webkit-font-smoothing: antialiased;
background: url("");
}
#menu::after {
display: block;
content: '';
padding-top: 80px;
}
#logo {
position: fixed;
bottom: 10px;
right: 10px;
background: rgba(255,255,255,.1);
font-size: 11px;
display: block;
width: 20px;
height: 20px;
line-height: 20px;
text-align: center;
-webkit-border-radius: 20px;
-moz-border-radius: 20px;
-webkit-box-shadow: 0 0 3px rgba(0,0,0,.2);
-moz-box-shadow: 0 0 3px rgba(0,0,0,.2);
color: inherit;
}
#menu li a {
display: block;
color: white;
padding: 0 35px 0 25px;
-webkit-transition: background 300ms;
-moz-transition: background 300ms;
}
#menu li {
position: relative;
list-style: none;
}
#menu a:hover,
#menu a.active {
text-decoration: none;
background: rgba(255,255,255,.1);
}
#menu li:hover .cov {
opacity: 1;
}
#menu li .dirname {
opacity: .60;
padding-right: 2px;
}
#menu li .basename {
opacity: 1;
}
#menu .cov {
background: rgba(0,0,0,.4);
position: absolute;
top: 0;
right: 8px;
font-size: 9px;
opacity: .6;
text-align: left;
width: 17px;
-webkit-border-radius: 10px;
-moz-border-radius: 10px;
padding: 2px 3px;
text-align: center;
}
#stats:nth-child(2n) {
display: inline-block;
margin-top: 15px;
border: 1px solid #eee;
padding: 10px;
-webkit-box-shadow: inset 0 0 2px #eee;
-moz-box-shadow: inset 0 0 2px #eee;
-webkit-border-radius: 5px;
-moz-border-radius: 5px;
}
#stats div {
float: left;
padding: 0 5px;
}
#stats::after {
display: block;
content: '';
clear: both;
}
#stats .sloc::after {
content: ' SLOC';
color: #b6b6b6;
}
#stats .percentage::after {
content: ' coverage';
color: #b6b6b6;
}
#stats .hits,
#stats .misses {
display: none;
}
.high {
color: #00d4b4;
}
.medium {
color: #e87d0d;
}
.low {
color: #d4081a;
}
.terrible {
color: #d4081a;
font-weight: bold;
}
table {
width: 80%;
margin-top: 10px;
border-collapse: collapse;
border: 1px solid #cbcbcb;
color: #363636;
-webkit-border-radius: 3px;
-moz-border-radius: 3px;
}
table thead {
display: none;
}
table td.line,
table td.hits {
width: 20px;
background: #eaeaea;
text-align: center;
font-size: 11px;
padding: 0 10px;
color: #949494;
}
table td.hits {
width: 10px;
padding: 2px 5px;
color: rgba(0,0,0,.2);
background: #f0f0f0;
}
tr.miss td.line,
tr.miss td.hits {
background: #e6c3c7;
}
tr.miss td {
background: #f8d5d8;
}
td.source {
padding-left: 15px;
line-height: 15px;
white-space: pre;
font: 12px monaco, monospace;
}
code .comment { color: #ddd }
code .init { color: #2F6FAD }
code .string { color: #5890AD }
code .keyword { color: #8A6343 }
code .number { color: #2F6FAD }
</style></head><body><div id="coverage"><h1 id="overview">Coverage</h1><div id="menu"><li><a href="#overview">overview</a></li><li><span class="cov terrible">4</span><a href="#compat/indexof.js"><span class="dirname">compat/</span><span class="basename">indexof.js</span></a></li><li><span class="cov low">33</span><a href="#compat/isarray.js"><span class="dirname">compat/</span><span class="basename">isarray.js</span></a></li><li><span class="cov high">100</span><a href="#core/create.js"><span class="dirname">core/</span><span class="basename">create.js</span></a></li><li><span class="cov high">100</span><a href="#core/dependency-graph.js"><span class="dirname">core/</span><span class="basename">dependency-graph.js</span></a></li><li><span class="cov high">100</span><a href="#core/resource-manager.js"><span class="dirname">core/</span><span class="basename">resource-manager.js</span></a></li><li><span class="cov high">100</span><a href="#core/resources.js"><span class="dirname">core/</span><span class="basename">resources.js</span></a></li><li><span class="cov high">84</span><a href="#core/tasks.js"><span class="dirname">core/</span><span class="basename">tasks.js</span></a></li><li><span class="cov high">100</span><a href="#date/date.js"><span class="dirname">date/</span><span class="basename">date.js</span></a></li><li><span class="cov high">100</span><a href="#date/timezone.js"><span class="dirname">date/</span><span class="basename">timezone.js</span></a></li><li><span class="cov high">100</span><a href="#sort/sort.js"><span class="dirname">sort/</span><span class="basename">sort.js</span></a></li><li><span class="cov high">100</span><a href="#sort/tasks.js"><span class="dirname">sort/</span><span class="basename">tasks.js</span></a></li><li><span class="cov high">100</span><a href="#util/functor.js"><span class="dirname">util/</span><span class="basename">functor.js</span></a></li><li><span class="cov high">100</span><a href="#util/memoized-range-fn.js"><span class="dirname">util/</span><span class="basename">memoized-range-fn.js</span></a></li><a id="logo" href="http://visionmedia.github.com/mocha/">m</a></div><div id="stats" class="high"><div class="percentage">91%</div><div class="sloc">384</div><div class="hits">352</div><div class="misses">32</div></div><div id="files"><div class="file"><h2 id="compat/indexof.js">compat/indexof.js</h2><div id="stats" class="terrible"><div class="percentage">4%</div><div class="sloc">23</div><div class="hits">1</div><div class="misses">22</div></div><table id="source"><thead><tr><th>Line</th><th>Hits</th><th>Source</th></tr></thead><tbody><tr><td class="line">1</td><td class="hits"></td><td class="source">// indexOf compares searchElement to elements of the Array using strict</td></tr><tr><td class="line">2</td><td class="hits"></td><td class="source">// equality (the same method used by the ===, or triple-equals, operator).</td></tr><tr><td class="line">3</td><td class="hits"></td><td class="source">//</td></tr><tr><td class="line">4</td><td class="hits"></td><td class="source">// https://developer.mozilla.org/en-US/docs/JavaScript/Reference/Global_Objects/Array/indexOf</td></tr><tr><td class="line">5</td><td class="hits"></td><td class="source">//</td></tr><tr class="hit"><td class="line">6</td><td class="hits">1</td><td class="source">if (!Array.prototype.indexOf) {</td></tr><tr class="miss"><td class="line">7</td><td class="hits">0</td><td class="source"> Array.prototype.indexOf = function (searchElement /*, fromIndex */ ) {</td></tr><tr class="miss"><td class="line">8</td><td class="hits">0</td><td class="source"> "use strict";</td></tr><tr class="miss"><td class="line">9</td><td class="hits">0</td><td class="source"> if (this == null) {</td></tr><tr class="miss"><td class="line">10</td><td class="hits">0</td><td class="source"> throw new TypeError();</td></tr><tr><td class="line">11</td><td class="hits"></td><td class="source"> }</td></tr><tr class="miss"><td class="line">12</td><td class="hits">0</td><td class="source"> var t = Object(this);</td></tr><tr class="miss"><td class="line">13</td><td class="hits">0</td><td class="source"> var len = t.length >>> 0;</td></tr><tr class="miss"><td class="line">14</td><td class="hits">0</td><td class="source"> if (len === 0) {</td></tr><tr class="miss"><td class="line">15</td><td class="hits">0</td><td class="source"> return -1;</td></tr><tr><td class="line">16</td><td class="hits"></td><td class="source"> }</td></tr><tr class="miss"><td class="line">17</td><td class="hits">0</td><td class="source"> var n = 0;</td></tr><tr class="miss"><td class="line">18</td><td class="hits">0</td><td class="source"> if (arguments.length > 1) {</td></tr><tr class="miss"><td class="line">19</td><td class="hits">0</td><td class="source"> n = Number(arguments[1]);</td></tr><tr class="miss"><td class="line">20</td><td class="hits">0</td><td class="source"> if (n != n) { // shortcut for verifying if it's NaN</td></tr><tr class="miss"><td class="line">21</td><td class="hits">0</td><td class="source"> n = 0;</td></tr><tr class="miss"><td class="line">22</td><td class="hits">0</td><td class="source"> } else if (n != 0 && n != Infinity && n != -Infinity) {</td></tr><tr class="miss"><td class="line">23</td><td class="hits">0</td><td class="source"> n = (n > 0 || -1) * Math.floor(Math.abs(n));</td></tr><tr><td class="line">24</td><td class="hits"></td><td class="source"> }</td></tr><tr><td class="line">25</td><td class="hits"></td><td class="source"> }</td></tr><tr class="miss"><td class="line">26</td><td class="hits">0</td><td class="source"> if (n >= len) {</td></tr><tr class="miss"><td class="line">27</td><td class="hits">0</td><td class="source"> return -1;</td></tr><tr><td class="line">28</td><td class="hits"></td><td class="source"> }</td></tr><tr class="miss"><td class="line">29</td><td class="hits">0</td><td class="source"> var k = n >= 0 ? n : Math.max(len - Math.abs(n), 0);</td></tr><tr class="miss"><td class="line">30</td><td class="hits">0</td><td class="source"> for (; k < len; k++) {</td></tr><tr class="miss"><td class="line">31</td><td class="hits">0</td><td class="source"> if (k in t && t[k] === searchElement) {</td></tr><tr class="miss"><td class="line">32</td><td class="hits">0</td><td class="source"> return k;</td></tr><tr><td class="line">33</td><td class="hits"></td><td class="source"> }</td></tr><tr><td class="line">34</td><td class="hits"></td><td class="source"> }</td></tr><tr class="miss"><td class="line">35</td><td class="hits">0</td><td class="source"> return -1;</td></tr><tr><td class="line">36</td><td class="hits"></td><td class="source"> }</td></tr><tr><td class="line">37</td><td class="hits"></td><td class="source">}</td></tr></tbody></table></div><div class="file"><h2 id="compat/isarray.js">compat/isarray.js</h2><div id="stats" class="low"><div class="percentage">33%</div><div class="sloc">3</div><div class="hits">1</div><div class="misses">2</div></div><table id="source"><thead><tr><th>Line</th><th>Hits</th><th>Source</th></tr></thead><tbody><tr><td class="line">1</td><td class="hits"></td><td class="source">// Returns true if an object is an array, false if it is not.</td></tr><tr><td class="line">2</td><td class="hits"></td><td class="source">//</td></tr><tr><td class="line">3</td><td class="hits"></td><td class="source">// From https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/isArray</td></tr><tr><td class="line">4</td><td class="hits"></td><td class="source"> </td></tr><tr><td class="line">5</td><td class="hits"></td><td class="source"> </td></tr><tr class="hit"><td class="line">6</td><td class="hits">1</td><td class="source">if(!Array.isArray) {</td></tr><tr class="miss"><td class="line">7</td><td class="hits">0</td><td class="source"> Array.isArray = function (vArg) {</td></tr><tr class="miss"><td class="line">8</td><td class="hits">0</td><td class="source"> return Object.prototype.toString.call(vArg) === "[object Array]";</td></tr><tr><td class="line">9</td><td class="hits"></td><td class="source"> };</td></tr><tr><td class="line">10</td><td class="hits"></td><td class="source">}</td></tr></tbody></table></div><div class="file"><h2 id="core/create.js">core/create.js</h2><div id="stats" class="high"><div class="percentage">100%</div><div class="sloc">64</div><div class="hits">64</div><div class="misses">0</div></div><table id="source"><thead><tr><th>Line</th><th>Hits</th><th>Source</th></tr></thead><tbody><tr><td class="line">1</td><td class="hits"></td><td class="source">/**</td></tr><tr><td class="line">2</td><td class="hits"></td><td class="source">* Schedule create</td></tr><tr><td class="line">3</td><td class="hits"></td><td class="source">* (c) 2013 Bill, BunKat LLC.</td></tr><tr><td class="line">4</td><td class="hits"></td><td class="source">*</td></tr><tr><td class="line">5</td><td class="hits"></td><td class="source">* Creates a schedule for each task that respects the task schedules, task</td></tr><tr><td class="line">6</td><td class="hits"></td><td class="source">* dependencies, resource schedules, project schedule, and start date provided.</td></tr><tr><td class="line">7</td><td class="hits"></td><td class="source">*</td></tr><tr><td class="line">8</td><td class="hits"></td><td class="source">* Schedule is freely distributable under the MIT license.</td></tr><tr><td class="line">9</td><td class="hits"></td><td class="source">* For all details and documentation:</td></tr><tr><td class="line">10</td><td class="hits"></td><td class="source">* http://github.com/bunkat/schedule</td></tr><tr><td class="line">11</td><td class="hits"></td><td class="source">*/</td></tr><tr><td class="line">12</td><td class="hits"></td><td class="source"> </td></tr><tr class="hit"><td class="line">13</td><td class="hits">1</td><td class="source">schedule.create = function(tasks, resources, sched, startDate) {</td></tr><tr><td class="line">14</td><td class="hits"></td><td class="source"> </td></tr><tr class="hit"><td class="line">15</td><td class="hits">22</td><td class="source"> var taskGraph = schedule.dependencyGraph(tasks),</td></tr><tr><td class="line">16</td><td class="hits"></td><td class="source"> resMgr = schedule.resourceManager(resources, startDate),</td></tr><tr><td class="line">17</td><td class="hits"></td><td class="source"> scheduledTasks = {};</td></tr><tr><td class="line">18</td><td class="hits"></td><td class="source"> </td></tr><tr><td class="line">19</td><td class="hits"></td><td class="source"> /**</td></tr><tr><td class="line">20</td><td class="hits"></td><td class="source"> * Main function, coordinates the process of creating a schedule.</td></tr><tr><td class="line">21</td><td class="hits"></td><td class="source"> */</td></tr><tr class="hit"><td class="line">22</td><td class="hits">22</td><td class="source"> function generateSchedule() {</td></tr><tr class="hit"><td class="line">23</td><td class="hits">22</td><td class="source"> var range, failedTasks = [];</td></tr><tr><td class="line">24</td><td class="hits"></td><td class="source"> </td></tr><tr><td class="line">25</td><td class="hits"></td><td class="source"> // add required resources not supplied in resources array, the project</td></tr><tr><td class="line">26</td><td class="hits"></td><td class="source"> // schedule and all task schedules to the resource manager (these will</td></tr><tr><td class="line">27</td><td class="hits"></td><td class="source"> // be treated as resources to calculate valid reservations with)</td></tr><tr class="hit"><td class="line">28</td><td class="hits">22</td><td class="source"> resMgr.addResource(taskGraph.resources, '', startDate);</td></tr><tr class="hit"><td class="line">29</td><td class="hits">22</td><td class="source"> resMgr.addResource([{id: '_proj', schedule: sched}], '', startDate);</td></tr><tr class="hit"><td class="line">30</td><td class="hits">22</td><td class="source"> resMgr.addResource(tasks, '_task', startDate);</td></tr><tr><td class="line">31</td><td class="hits"></td><td class="source"> </td></tr><tr class="hit"><td class="line">32</td><td class="hits">22</td><td class="source"> forwardPass(taskGraph.roots);</td></tr><tr class="hit"><td class="line">33</td><td class="hits">22</td><td class="source"> range = getSummary(tasks, failedTasks);</td></tr><tr class="hit"><td class="line">34</td><td class="hits">22</td><td class="source"> backwardPass(taskGraph.leaves, range[1]);</td></tr><tr><td class="line">35</td><td class="hits"></td><td class="source"> </td></tr><tr class="hit"><td class="line">36</td><td class="hits">22</td><td class="source"> return {</td></tr><tr><td class="line">37</td><td class="hits"></td><td class="source"> scheduledTasks: scheduledTasks,</td></tr><tr><td class="line">38</td><td class="hits"></td><td class="source"> failedTasks: failedTasks.length ? failedTasks : null,</td></tr><tr><td class="line">39</td><td class="hits"></td><td class="source"> success: failedTasks.length === 0,</td></tr><tr><td class="line">40</td><td class="hits"></td><td class="source"> start: range[0],</td></tr><tr><td class="line">41</td><td class="hits"></td><td class="source"> end: range[1]</td></tr><tr><td class="line">42</td><td class="hits"></td><td class="source"> };</td></tr><tr><td class="line">43</td><td class="hits"></td><td class="source"> }</td></tr><tr><td class="line">44</td><td class="hits"></td><td class="source"> </td></tr><tr><td class="line">45</td><td class="hits"></td><td class="source"> /**</td></tr><tr><td class="line">46</td><td class="hits"></td><td class="source"> * Schedules each task as their dependencies are met, tracking dependency</td></tr><tr><td class="line">47</td><td class="hits"></td><td class="source"> * end dates in the dependencies map.</td></tr><tr><td class="line">48</td><td class="hits"></td><td class="source"> */</td></tr><tr class="hit"><td class="line">49</td><td class="hits">22</td><td class="source"> function forwardPass(roots) {</td></tr><tr class="hit"><td class="line">50</td><td class="hits">22</td><td class="source"> var readyTasks = roots.slice(0),</td></tr><tr><td class="line">51</td><td class="hits"></td><td class="source"> dependencies = {}; // holds count and earliest start date of dependencies</td></tr><tr><td class="line">52</td><td class="hits"></td><td class="source"> </td></tr><tr class="hit"><td class="line">53</td><td class="hits">22</td><td class="source"> for(var i = 0, len = roots.length; i < len; i++) {</td></tr><tr class="hit"><td class="line">54</td><td class="hits">46</td><td class="source"> dependencies[roots[i]] = [0, startDate.getTime()];</td></tr><tr><td class="line">55</td><td class="hits"></td><td class="source"> }</td></tr><tr><td class="line">56</td><td class="hits"></td><td class="source"> </td></tr><tr class="hit"><td class="line">57</td><td class="hits">22</td><td class="source"> while(readyTasks.length) {</td></tr><tr class="hit"><td class="line">58</td><td class="hits">69</td><td class="source"> schedule.sort.tasks(taskGraph, readyTasks);</td></tr><tr><td class="line">59</td><td class="hits"></td><td class="source"> </td></tr><tr class="hit"><td class="line">60</td><td class="hits">69</td><td class="source"> var task = taskGraph.tasks[readyTasks.pop()],</td></tr><tr><td class="line">61</td><td class="hits"></td><td class="source"> start = dependencies[task.id][1],</td></tr><tr><td class="line">62</td><td class="hits"></td><td class="source"> end = forwardPassTask(task, start);</td></tr><tr><td class="line">63</td><td class="hits"></td><td class="source"> </td></tr><tr class="hit"><td class="line">64</td><td class="hits">69</td><td class="source"> if(end && task.requiredBy) {</td></tr><tr class="hit"><td class="line">65</td><td class="hits">20</td><td class="source"> updateDependencies(readyTasks, dependencies, task.requiredBy, end);</td></tr><tr class="hit"><td class="line">66</td><td class="hits">20</td><td class="source"> resMgr.optimize(getMinStart(dependencies)); // clean up expired exceptions</td></tr><tr><td class="line">67</td><td class="hits"></td><td class="source"> }</td></tr><tr><td class="line">68</td><td class="hits"></td><td class="source"> }</td></tr><tr><td class="line">69</td><td class="hits"></td><td class="source"> }</td></tr><tr><td class="line">70</td><td class="hits"></td><td class="source"> </td></tr><tr><td class="line">71</td><td class="hits"></td><td class="source"> /**</td></tr><tr><td class="line">72</td><td class="hits"></td><td class="source"> * Finds the next available time that all of a tasks constraints are met and</td></tr><tr><td class="line">73</td><td class="hits"></td><td class="source"> * makes the appropriate resource reservations. A task may be scheduled in a</td></tr><tr><td class="line">74</td><td class="hits"></td><td class="source"> * single contiguous block or multiple blocks of time.</td></tr><tr><td class="line">75</td><td class="hits"></td><td class="source"> */</td></tr><tr class="hit"><td class="line">76</td><td class="hits">22</td><td class="source"> function forwardPassTask(task, start) {</td></tr><tr class="hit"><td class="line">77</td><td class="hits">69</td><td class="source"> var resAll = ['_proj', '_task' + task.id],</td></tr><tr><td class="line">78</td><td class="hits"></td><td class="source"> resources = task.resources ? resAll.concat(task.resources) : resAll,</td></tr><tr><td class="line">79</td><td class="hits"></td><td class="source"> duration = task.duration,</td></tr><tr><td class="line">80</td><td class="hits"></td><td class="source"> next = start,</td></tr><tr><td class="line">81</td><td class="hits"></td><td class="source"> scheduledTask = {schedule: [], duration: task.duration};</td></tr><tr><td class="line">82</td><td class="hits"></td><td class="source"> </td></tr><tr class="hit"><td class="line">83</td><td class="hits">69</td><td class="source"> while(duration) {</td></tr><tr class="hit"><td class="line">84</td><td class="hits">73</td><td class="source"> var r = resMgr.makeReservation(resources, next, task.minSchedule || 1, duration);</td></tr><tr class="hit"><td class="line">85</td><td class="hits">75</td><td class="source"> if(!r.success) return undefined;</td></tr><tr><td class="line">86</td><td class="hits"></td><td class="source"> </td></tr><tr class="hit"><td class="line">87</td><td class="hits">71</td><td class="source"> scheduledTask.earlyStart = scheduledTask.earlyStart || r.start;</td></tr><tr class="hit"><td class="line">88</td><td class="hits">71</td><td class="source"> scheduledTask.schedule.push(r);</td></tr><tr class="hit"><td class="line">89</td><td class="hits">71</td><td class="source"> duration -= r.duration;</td></tr><tr class="hit"><td class="line">90</td><td class="hits">71</td><td class="source"> next = r.end;</td></tr><tr><td class="line">91</td><td class="hits"></td><td class="source"> }</td></tr><tr><td class="line">92</td><td class="hits"></td><td class="source"> </td></tr><tr class="hit"><td class="line">93</td><td class="hits">67</td><td class="source"> scheduledTask.earlyFinish = next;</td></tr><tr class="hit"><td class="line">94</td><td class="hits">67</td><td class="source"> scheduledTasks[task.id] = scheduledTask;</td></tr><tr><td class="line">95</td><td class="hits"></td><td class="source"> </td></tr><tr class="hit"><td class="line">96</td><td class="hits">67</td><td class="source"> return next;</td></tr><tr><td class="line">97</td><td class="hits"></td><td class="source"> }</td></tr><tr><td class="line">98</td><td class="hits"></td><td class="source"> </td></tr><tr><td class="line">99</td><td class="hits"></td><td class="source"> /**</td></tr><tr><td class="line">100</td><td class="hits"></td><td class="source"> * Finds the start and end date of the schedule and adds any tasks that were</td></tr><tr><td class="line">101</td><td class="hits"></td><td class="source"> * scheduled to the failedTasks array.</td></tr><tr><td class="line">102</td><td class="hits"></td><td class="source"> */</td></tr><tr class="hit"><td class="line">103</td><td class="hits">22</td><td class="source"> function getSummary(tasks, failedTasks) {</td></tr><tr class="hit"><td class="line">104</td><td class="hits">22</td><td class="source"> var start, end;</td></tr><tr><td class="line">105</td><td class="hits"></td><td class="source"> </td></tr><tr class="hit"><td class="line">106</td><td class="hits">22</td><td class="source"> for(var i = 0, len = tasks.length; i < len; i++) {</td></tr><tr class="hit"><td class="line">107</td><td class="hits">72</td><td class="source"> var t = scheduledTasks[tasks[i].id];</td></tr><tr class="hit"><td class="line">108</td><td class="hits">72</td><td class="source"> if(t) {</td></tr><tr class="hit"><td class="line">109</td><td class="hits">67</td><td class="source"> start = !start || t.earlyStart < start ? t.earlyStart : start;</td></tr><tr class="hit"><td class="line">110</td><td class="hits">67</td><td class="source"> end = !end || t.earlyFinish > end ? t.earlyFinish : end;</td></tr><tr><td class="line">111</td><td class="hits"></td><td class="source"> }</td></tr><tr><td class="line">112</td><td class="hits"></td><td class="source"> else {</td></tr><tr class="hit"><td class="line">113</td><td class="hits">5</td><td class="source"> failedTasks.push(tasks[i].id);</td></tr><tr><td class="line">114</td><td class="hits"></td><td class="source"> }</td></tr><tr><td class="line">115</td><td class="hits"></td><td class="source"> }</td></tr><tr><td class="line">116</td><td class="hits"></td><td class="source"> </td></tr><tr class="hit"><td class="line">117</td><td class="hits">22</td><td class="source"> return [start, end];</td></tr><tr><td class="line">118</td><td class="hits"></td><td class="source"> }</td></tr><tr><td class="line">119</td><td class="hits"></td><td class="source"> </td></tr><tr><td class="line">120</td><td class="hits"></td><td class="source"> /**</td></tr><tr><td class="line">121</td><td class="hits"></td><td class="source"> * As tasks are scheduled, the information is tracked in the dependencies</td></tr><tr><td class="line">122</td><td class="hits"></td><td class="source"> * array. As a tasks dependencies are all met, the task is pushed onto the</td></tr><tr><td class="line">123</td><td class="hits"></td><td class="source"> * readyTasks array which means it is available to be scheduled.</td></tr><tr><td class="line">124</td><td class="hits"></td><td class="source"> */</td></tr><tr class="hit"><td class="line">125</td><td class="hits">22</td><td class="source"> function updateDependencies(readyTasks, dependencies, tasks, end) {</td></tr><tr class="hit"><td class="line">126</td><td class="hits">20</td><td class="source"> for(var i = 0, len = tasks.length; i < len; i++) {</td></tr><tr class="hit"><td class="line">127</td><td class="hits">34</td><td class="source"> var tid = tasks[i],</td></tr><tr><td class="line">128</td><td class="hits"></td><td class="source"> dependsOn = taskGraph.tasks[tid].dependsOn,</td></tr><tr><td class="line">129</td><td class="hits"></td><td class="source"> metDeps = dependencies[tid] || (dependencies[tid] = [0, 0]);</td></tr><tr><td class="line">130</td><td class="hits"></td><td class="source"> </td></tr><tr class="hit"><td class="line">131</td><td class="hits">34</td><td class="source"> metDeps[0] += 1;</td></tr><tr class="hit"><td class="line">132</td><td class="hits">34</td><td class="source"> metDeps[1] = end > metDeps[1] ? end : metDeps[1];</td></tr><tr><td class="line">133</td><td class="hits"></td><td class="source"> </td></tr><tr class="hit"><td class="line">134</td><td class="hits">34</td><td class="source"> if(!dependsOn || metDeps[0] >= dependsOn.length) {</td></tr><tr class="hit"><td class="line">135</td><td class="hits">23</td><td class="source"> readyTasks.push(tid);</td></tr><tr><td class="line">136</td><td class="hits"></td><td class="source"> }</td></tr><tr><td class="line">137</td><td class="hits"></td><td class="source"> }</td></tr><tr><td class="line">138</td><td class="hits"></td><td class="source"> }</td></tr><tr><td class="line">139</td><td class="hits"></td><td class="source"> </td></tr><tr><td class="line">140</td><td class="hits"></td><td class="source"> /**</td></tr><tr><td class="line">141</td><td class="hits"></td><td class="source"> * Finds the earliest time that any of the remaining tasks could be scheduled</td></tr><tr><td class="line">142</td><td class="hits"></td><td class="source"> * for. It is used to optimize the resource manager since nothing can be</td></tr><tr><td class="line">143</td><td class="hits"></td><td class="source"> * scheduled before this time.</td></tr><tr><td class="line">144</td><td class="hits"></td><td class="source"> */</td></tr><tr class="hit"><td class="line">145</td><td class="hits">22</td><td class="source"> function getMinStart(dependencies) {</td></tr><tr class="hit"><td class="line">146</td><td class="hits">20</td><td class="source"> var min;</td></tr><tr class="hit"><td class="line">147</td><td class="hits">20</td><td class="source"> for(var id in dependencies) {</td></tr><tr class="hit"><td class="line">148</td><td class="hits">89</td><td class="source"> if(!min || min > dependencies[id][1]) {</td></tr><tr class="hit"><td class="line">149</td><td class="hits">20</td><td class="source"> min = dependencies[id][1];</td></tr><tr><td class="line">150</td><td class="hits"></td><td class="source"> }</td></tr><tr><td class="line">151</td><td class="hits"></td><td class="source"> }</td></tr><tr class="hit"><td class="line">152</td><td class="hits">20</td><td class="source"> return min;</td></tr><tr><td class="line">153</td><td class="hits"></td><td class="source"> }</td></tr><tr><td class="line">154</td><td class="hits"></td><td class="source"> </td></tr><tr><td class="line">155</td><td class="hits"></td><td class="source"> /**</td></tr><tr><td class="line">156</td><td class="hits"></td><td class="source"> * Calculates when a task must be completed by before it ends up slipping</td></tr><tr><td class="line">157</td><td class="hits"></td><td class="source"> * one of its dependencies or the schedule. Tasks with zero float amount</td></tr><tr><td class="line">158</td><td class="hits"></td><td class="source"> * are in the critical path.</td></tr><tr><td class="line">159</td><td class="hits"></td><td class="source"> */</td></tr><tr class="hit"><td class="line">160</td><td class="hits">22</td><td class="source"> function backwardPass(tasks, finishDate) {</td></tr><tr class="hit"><td class="line">161</td><td class="hits">59</td><td class="source"> for(var i = 0, len = tasks.length; i < len; i++) {</td></tr><tr class="hit"><td class="line">162</td><td class="hits">114</td><td class="source"> var sTask = scheduledTasks[tasks[i]],</td></tr><tr><td class="line">163</td><td class="hits"></td><td class="source"> dependsOn = taskGraph.tasks[tasks[i]].dependsOn;</td></tr><tr><td class="line">164</td><td class="hits"></td><td class="source"> </td></tr><tr class="hit"><td class="line">165</td><td class="hits">114</td><td class="source"> if(sTask) {</td></tr><tr class="hit"><td class="line">166</td><td class="hits">112</td><td class="source"> sTask.lateFinish = finishDate;</td></tr><tr class="hit"><td class="line">167</td><td class="hits">112</td><td class="source"> sTask.floatAmt = (sTask.lateFinish - sTask.earlyFinish) / later.MIN;</td></tr><tr><td class="line">168</td><td class="hits"></td><td class="source"> </td></tr><tr class="hit"><td class="line">169</td><td class="hits">112</td><td class="source"> if(dependsOn) {</td></tr><tr class="hit"><td class="line">170</td><td class="hits">37</td><td class="source"> backwardPass(dependsOn, sTask.earlyStart);</td></tr><tr><td class="line">171</td><td class="hits"></td><td class="source"> }</td></tr><tr><td class="line">172</td><td class="hits"></td><td class="source"> }</td></tr><tr><td class="line">173</td><td class="hits"></td><td class="source"> }</td></tr><tr><td class="line">174</td><td class="hits"></td><td class="source"> }</td></tr><tr><td class="line">175</td><td class="hits"></td><td class="source"> </td></tr><tr class="hit"><td class="line">176</td><td class="hits">22</td><td class="source"> return generateSchedule();</td></tr><tr><td class="line">177</td><td class="hits"></td><td class="source">};</td></tr></tbody></table></div><div class="file"><h2 id="core/dependency-graph.js">core/dependency-graph.js</h2><div id="stats" class="high"><div class="percentage">100%</div><div class="sloc">79</div><div class="hits">79</div><div class="misses">0</div></div><table id="source"><thead><tr><th>Line</th><th>Hits</th><th>Source</th></tr></thead><tbody><tr><td class="line">1</td><td class="hits"></td><td class="source">/**</td></tr><tr><td class="line">2</td><td class="hits"></td><td class="source">* Dependency graph</td></tr><tr><td class="line">3</td><td class="hits"></td><td class="source">* (c) 2013 Bill, BunKat LLC.</td></tr><tr><td class="line">4</td><td class="hits"></td><td class="source">*</td></tr><tr><td class="line">5</td><td class="hits"></td><td class="source">* Generates a dependency graph from a set of tasks and finds the root nodes,</td></tr><tr><td class="line">6</td><td class="hits"></td><td class="source">* leaf nodes, depth, and optimistic float (time between when a schedule starts</td></tr><tr><td class="line">7</td><td class="hits"></td><td class="source">* and when it must start to prevent a schedule slip). This information is used</td></tr><tr><td class="line">8</td><td class="hits"></td><td class="source">* by the schedule generator to schedule tasks against an actual timeline.</td></tr><tr><td class="line">9</td><td class="hits"></td><td class="source">*</td></tr><tr><td class="line">10</td><td class="hits"></td><td class="source">* Schedule is freely distributable under the MIT license.</td></tr><tr><td class="line">11</td><td class="hits"></td><td class="source">* For all details and documentation:</td></tr><tr><td class="line">12</td><td class="hits"></td><td class="source">* http://github.com/bunkat/schedule</td></tr><tr><td class="line">13</td><td class="hits"></td><td class="source">*/</td></tr><tr><td class="line">14</td><td class="hits"></td><td class="source"> </td></tr><tr class="hit"><td class="line">15</td><td class="hits">1</td><td class="source">schedule.dependencyGraph = function(taskArr) {</td></tr><tr><td class="line">16</td><td class="hits"></td><td class="source"> </td></tr><tr><td class="line">17</td><td class="hits"></td><td class="source"> /**</td></tr><tr><td class="line">18</td><td class="hits"></td><td class="source"> * Starting point for creating the dependency graph, clones the tasks and</td></tr><tr><td class="line">19</td><td class="hits"></td><td class="source"> * then fills out the graph properties.</td></tr><tr><td class="line">20</td><td class="hits"></td><td class="source"> */</td></tr><tr class="hit"><td class="line">21</td><td class="hits">33</td><td class="source"> function createDependencyGraph(tasks) {</td></tr><tr class="hit"><td class="line">22</td><td class="hits">33</td><td class="source"> var graph = {</td></tr><tr><td class="line">23</td><td class="hits"></td><td class="source"> tasks: {},</td></tr><tr><td class="line">24</td><td class="hits"></td><td class="source"> roots: [],</td></tr><tr><td class="line">25</td><td class="hits"></td><td class="source"> leaves: [],</td></tr><tr><td class="line">26</td><td class="hits"></td><td class="source"> resources: [],</td></tr><tr><td class="line">27</td><td class="hits"></td><td class="source"> depth: 0,</td></tr><tr><td class="line">28</td><td class="hits"></td><td class="source"> end : 0</td></tr><tr><td class="line">29</td><td class="hits"></td><td class="source"> };</td></tr><tr><td class="line"