diff --git a/coverage/clover.xml b/coverage/clover.xml
new file mode 100644
index 000000000..d0b191c6b
--- /dev/null
+++ b/coverage/clover.xml
@@ -0,0 +1,139 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/coverage/coverage-final.json b/coverage/coverage-final.json
new file mode 100644
index 000000000..275363f90
--- /dev/null
+++ b/coverage/coverage-final.json
@@ -0,0 +1,2 @@
+{"/Users/joeyklee-nyt/Source/joeyklee/ml5-library/src/CharRNN/index.js": {"path":"/Users/joeyklee-nyt/Source/joeyklee/ml5-library/src/CharRNN/index.js","statementMap":{"0":{"start":{"line":18,"column":18},"end":{"line":18,"column":43}},"1":{"start":{"line":19,"column":21},"end":{"line":19,"column":56}},"2":{"start":{"line":20,"column":28},"end":{"line":20,"column":39}},"3":{"start":{"line":36,"column":4},"end":{"line":36,"column":23}},"4":{"start":{"line":43,"column":4},"end":{"line":43,"column":20}},"5":{"start":{"line":44,"column":4},"end":{"line":44,"column":25}},"6":{"start":{"line":45,"column":4},"end":{"line":45,"column":20}},"7":{"start":{"line":46,"column":4},"end":{"line":46,"column":38}},"8":{"start":{"line":52,"column":4},"end":{"line":52,"column":34}},"9":{"start":{"line":53,"column":4},"end":{"line":53,"column":20}},"10":{"start":{"line":54,"column":4},"end":{"line":54,"column":23}},"11":{"start":{"line":55,"column":4},"end":{"line":55,"column":28}},"12":{"start":{"line":56,"column":4},"end":{"line":61,"column":6}},"13":{"start":{"line":63,"column":4},"end":{"line":63,"column":73}},"14":{"start":{"line":68,"column":4},"end":{"line":68,"column":32}},"15":{"start":{"line":72,"column":4},"end":{"line":72,"column":23}},"16":{"start":{"line":76,"column":4},"end":{"line":76,"column":22}},"17":{"start":{"line":80,"column":19},"end":{"line":80,"column":45}},"18":{"start":{"line":81,"column":17},"end":{"line":81,"column":47}},"19":{"start":{"line":82,"column":4},"end":{"line":99,"column":7}},"20":{"start":{"line":83,"column":6},"end":{"line":98,"column":7}},"21":{"start":{"line":84,"column":8},"end":{"line":89,"column":9}},"22":{"start":{"line":85,"column":10},"end":{"line":85,"column":68}},"23":{"start":{"line":86,"column":10},"end":{"line":86,"column":32}},"24":{"start":{"line":88,"column":10},"end":{"line":88,"column":66}},"25":{"start":{"line":90,"column":13},"end":{"line":98,"column":7}},"26":{"start":{"line":91,"column":8},"end":{"line":95,"column":9}},"27":{"start":{"line":92,"column":10},"end":{"line":92,"column":55}},"28":{"start":{"line":94,"column":10},"end":{"line":94,"column":54}},"29":{"start":{"line":97,"column":8},"end":{"line":97,"column":36}},"30":{"start":{"line":100,"column":4},"end":{"line":100,"column":31}},"31":{"start":{"line":101,"column":4},"end":{"line":101,"column":27}},"32":{"start":{"line":102,"column":4},"end":{"line":102,"column":16}},"33":{"start":{"line":106,"column":4},"end":{"line":113,"column":5}},"34":{"start":{"line":107,"column":23},"end":{"line":107,"column":60}},"35":{"start":{"line":108,"column":6},"end":{"line":108,"column":24}},"36":{"start":{"line":109,"column":6},"end":{"line":109,"column":48}},"37":{"start":{"line":110,"column":6},"end":{"line":110,"column":24}},"38":{"start":{"line":112,"column":6},"end":{"line":112,"column":17}},"39":{"start":{"line":117,"column":4},"end":{"line":117,"column":20}},"40":{"start":{"line":118,"column":4},"end":{"line":118,"column":38}},"41":{"start":{"line":119,"column":23},"end":{"line":119,"column":37}},"42":{"start":{"line":121,"column":17},"end":{"line":132,"column":5}},"43":{"start":{"line":122,"column":19},"end":{"line":130,"column":9}},"44":{"start":{"line":123,"column":8},"end":{"line":130,"column":9}},"45":{"start":{"line":131,"column":6},"end":{"line":131,"column":18}},"46":{"start":{"line":134,"column":4},"end":{"line":138,"column":5}},"47":{"start":{"line":134,"column":17},"end":{"line":134,"column":18}},"48":{"start":{"line":135,"column":6},"end":{"line":135,"column":81}},"49":{"start":{"line":136,"column":6},"end":{"line":136,"column":81}},"50":{"start":{"line":137,"column":6},"end":{"line":137,"column":31}},"51":{"start":{"line":140,"column":4},"end":{"line":140,"column":32}},"52":{"start":{"line":144,"column":4},"end":{"line":144,"column":21}},"53":{"start":{"line":145,"column":17},"end":{"line":145,"column":51}},"54":{"start":{"line":146,"column":19},"end":{"line":146,"column":58}},"55":{"start":{"line":147,"column":24},"end":{"line":147,"column":73}},"56":{"start":{"line":148,"column":21},"end":{"line":148,"column":63}},"57":{"start":{"line":149,"column":4},"end":{"line":151,"column":5}},"58":{"start":{"line":150,"column":6},"end":{"line":150,"column":34}},"59":{"start":{"line":153,"column":20},"end":{"line":153,"column":22}},"60":{"start":{"line":154,"column":22},"end":{"line":154,"column":38}},"61":{"start":{"line":155,"column":25},"end":{"line":155,"column":27}},"62":{"start":{"line":157,"column":4},"end":{"line":159,"column":7}},"63":{"start":{"line":158,"column":6},"end":{"line":158,"column":42}},"64":{"start":{"line":161,"column":16},"end":{"line":161,"column":31}},"65":{"start":{"line":162,"column":34},"end":{"line":162,"column":36}},"66":{"start":{"line":164,"column":4},"end":{"line":192,"column":5}},"67":{"start":{"line":164,"column":17},"end":{"line":164,"column":18}},"68":{"start":{"line":165,"column":27},"end":{"line":165,"column":63}},"69":{"start":{"line":166,"column":6},"end":{"line":166,"column":38}},"70":{"start":{"line":167,"column":21},"end":{"line":167,"column":44}},"71":{"start":{"line":169,"column":6},"end":{"line":174,"column":7}},"72":{"start":{"line":170,"column":25},"end":{"line":170,"column":64}},"73":{"start":{"line":171,"column":8},"end":{"line":171,"column":83}},"74":{"start":{"line":173,"column":8},"end":{"line":173,"column":81}},"75":{"start":{"line":176,"column":6},"end":{"line":176,"column":31}},"76":{"start":{"line":177,"column":6},"end":{"line":177,"column":31}},"77":{"start":{"line":179,"column":22},"end":{"line":179,"column":37}},"78":{"start":{"line":180,"column":29},"end":{"line":180,"column":81}},"79":{"start":{"line":181,"column":21},"end":{"line":181,"column":76}},"80":{"start":{"line":182,"column":22},"end":{"line":182,"column":60}},"81":{"start":{"line":183,"column":28},"end":{"line":183,"column":43}},"82":{"start":{"line":184,"column":6},"end":{"line":184,"column":90}},"83":{"start":{"line":186,"column":6},"end":{"line":191,"column":7}},"84":{"start":{"line":187,"column":8},"end":{"line":187,"column":36}},"85":{"start":{"line":189,"column":8},"end":{"line":189,"column":64}},"86":{"start":{"line":190,"column":8},"end":{"line":190,"column":28}},"87":{"start":{"line":194,"column":20},"end":{"line":194,"column":22}},"88":{"start":{"line":195,"column":4},"end":{"line":200,"column":7}},"89":{"start":{"line":196,"column":21},"end":{"line":196,"column":82}},"90":{"start":{"line":196,"column":57},"end":{"line":196,"column":81}},"91":{"start":{"line":197,"column":6},"end":{"line":199,"column":7}},"92":{"start":{"line":198,"column":8},"end":{"line":198,"column":28}},"93":{"start":{"line":201,"column":4},"end":{"line":201,"column":49}},"94":{"start":{"line":202,"column":4},"end":{"line":205,"column":6}},"95":{"start":{"line":212,"column":4},"end":{"line":212,"column":32}},"96":{"start":{"line":235,"column":4},"end":{"line":235,"column":17}},"97":{"start":{"line":236,"column":4},"end":{"line":236,"column":66}},"98":{"start":{"line":248,"column":34},"end":{"line":248,"column":36}},"99":{"start":{"line":249,"column":24},"end":{"line":249,"column":45}},"100":{"start":{"line":250,"column":20},"end":{"line":250,"column":35}},"101":{"start":{"line":251,"column":27},"end":{"line":251,"column":79}},"102":{"start":{"line":252,"column":19},"end":{"line":252,"column":74}},"103":{"start":{"line":253,"column":20},"end":{"line":253,"column":58}},"104":{"start":{"line":254,"column":26},"end":{"line":254,"column":41}},"105":{"start":{"line":255,"column":4},"end":{"line":255,"column":88}},"106":{"start":{"line":257,"column":19},"end":{"line":257,"column":66}},"107":{"start":{"line":258,"column":19},"end":{"line":258,"column":82}},"108":{"start":{"line":258,"column":55},"end":{"line":258,"column":81}},"109":{"start":{"line":259,"column":4},"end":{"line":259,"column":49}},"110":{"start":{"line":260,"column":4},"end":{"line":262,"column":5}},"111":{"start":{"line":261,"column":6},"end":{"line":261,"column":23}},"112":{"start":{"line":264,"column":15},"end":{"line":267,"column":7}},"113":{"start":{"line":264,"column":49},"end":{"line":267,"column":5}},"114":{"start":{"line":268,"column":4},"end":{"line":271,"column":6}},"115":{"start":{"line":282,"column":4},"end":{"line":282,"column":21}},"116":{"start":{"line":283,"column":17},"end":{"line":283,"column":38}},"117":{"start":{"line":284,"column":25},"end":{"line":284,"column":27}},"118":{"start":{"line":286,"column":4},"end":{"line":288,"column":7}},"119":{"start":{"line":287,"column":6},"end":{"line":287,"column":42}},"120":{"start":{"line":290,"column":16},"end":{"line":290,"column":31}},"121":{"start":{"line":291,"column":4},"end":{"line":305,"column":5}},"122":{"start":{"line":291,"column":17},"end":{"line":291,"column":18}},"123":{"start":{"line":292,"column":27},"end":{"line":292,"column":63}},"124":{"start":{"line":293,"column":6},"end":{"line":293,"column":38}},"125":{"start":{"line":294,"column":21},"end":{"line":294,"column":44}},"126":{"start":{"line":296,"column":6},"end":{"line":301,"column":7}},"127":{"start":{"line":297,"column":25},"end":{"line":297,"column":64}},"128":{"start":{"line":298,"column":8},"end":{"line":298,"column":83}},"129":{"start":{"line":300,"column":8},"end":{"line":300,"column":81}},"130":{"start":{"line":302,"column":6},"end":{"line":302,"column":31}},"131":{"start":{"line":303,"column":6},"end":{"line":303,"column":31}},"132":{"start":{"line":304,"column":6},"end":{"line":304,"column":30}},"133":{"start":{"line":306,"column":4},"end":{"line":308,"column":5}},"134":{"start":{"line":307,"column":6},"end":{"line":307,"column":17}},"135":{"start":{"line":312,"column":16},"end":{"line":312,"column":80}},"136":{"start":{"line":312,"column":48},"end":{"line":312,"column":80}}},"fnMap":{"0":{"name":"(anonymous_0)","decl":{"start":{"line":30,"column":2},"end":{"line":30,"column":3}},"loc":{"start":{"line":30,"column":35},"end":{"line":65,"column":3}},"line":30},"1":{"name":"(anonymous_1)","decl":{"start":{"line":67,"column":2},"end":{"line":67,"column":3}},"loc":{"start":{"line":67,"column":15},"end":{"line":69,"column":3}},"line":67},"2":{"name":"(anonymous_2)","decl":{"start":{"line":71,"column":2},"end":{"line":71,"column":3}},"loc":{"start":{"line":71,"column":18},"end":{"line":73,"column":3}},"line":71},"3":{"name":"(anonymous_3)","decl":{"start":{"line":75,"column":2},"end":{"line":75,"column":3}},"loc":{"start":{"line":75,"column":13},"end":{"line":77,"column":3}},"line":75},"4":{"name":"(anonymous_4)","decl":{"start":{"line":79,"column":2},"end":{"line":79,"column":3}},"loc":{"start":{"line":79,"column":30},"end":{"line":103,"column":3}},"line":79},"5":{"name":"(anonymous_5)","decl":{"start":{"line":82,"column":30},"end":{"line":82,"column":31}},"loc":{"start":{"line":82,"column":37},"end":{"line":99,"column":5}},"line":82},"6":{"name":"(anonymous_6)","decl":{"start":{"line":105,"column":2},"end":{"line":105,"column":3}},"loc":{"start":{"line":105,"column":24},"end":{"line":114,"column":3}},"line":105},"7":{"name":"(anonymous_7)","decl":{"start":{"line":116,"column":2},"end":{"line":116,"column":3}},"loc":{"start":{"line":116,"column":20},"end":{"line":141,"column":3}},"line":116},"8":{"name":"(anonymous_8)","decl":{"start":{"line":121,"column":17},"end":{"line":121,"column":18}},"loc":{"start":{"line":121,"column":22},"end":{"line":132,"column":5}},"line":121},"9":{"name":"(anonymous_9)","decl":{"start":{"line":122,"column":19},"end":{"line":122,"column":20}},"loc":{"start":{"line":123,"column":8},"end":{"line":130,"column":9}},"line":123},"10":{"name":"(anonymous_10)","decl":{"start":{"line":143,"column":2},"end":{"line":143,"column":3}},"loc":{"start":{"line":143,"column":34},"end":{"line":206,"column":3}},"line":143},"11":{"name":"(anonymous_11)","decl":{"start":{"line":157,"column":22},"end":{"line":157,"column":23}},"loc":{"start":{"line":157,"column":30},"end":{"line":159,"column":5}},"line":157},"12":{"name":"(anonymous_12)","decl":{"start":{"line":195,"column":20},"end":{"line":195,"column":21}},"loc":{"start":{"line":195,"column":28},"end":{"line":200,"column":5}},"line":195},"13":{"name":"(anonymous_13)","decl":{"start":{"line":196,"column":50},"end":{"line":196,"column":51}},"loc":{"start":{"line":196,"column":57},"end":{"line":196,"column":81}},"line":196},"14":{"name":"(anonymous_14)","decl":{"start":{"line":211,"column":2},"end":{"line":211,"column":3}},"loc":{"start":{"line":211,"column":10},"end":{"line":213,"column":3}},"line":211},"15":{"name":"(anonymous_15)","decl":{"start":{"line":234,"column":2},"end":{"line":234,"column":3}},"loc":{"start":{"line":234,"column":36},"end":{"line":237,"column":3}},"line":234},"16":{"name":"(anonymous_16)","decl":{"start":{"line":247,"column":2},"end":{"line":247,"column":3}},"loc":{"start":{"line":247,"column":32},"end":{"line":272,"column":3}},"line":247},"17":{"name":"(anonymous_17)","decl":{"start":{"line":258,"column":48},"end":{"line":258,"column":49}},"loc":{"start":{"line":258,"column":55},"end":{"line":258,"column":81}},"line":258},"18":{"name":"(anonymous_18)","decl":{"start":{"line":264,"column":43},"end":{"line":264,"column":44}},"loc":{"start":{"line":264,"column":49},"end":{"line":267,"column":5}},"line":264},"19":{"name":"(anonymous_19)","decl":{"start":{"line":281,"column":2},"end":{"line":281,"column":3}},"loc":{"start":{"line":281,"column":34},"end":{"line":309,"column":3}},"line":281},"20":{"name":"(anonymous_20)","decl":{"start":{"line":286,"column":17},"end":{"line":286,"column":18}},"loc":{"start":{"line":286,"column":25},"end":{"line":288,"column":5}},"line":286},"21":{"name":"(anonymous_21)","decl":{"start":{"line":312,"column":16},"end":{"line":312,"column":17}},"loc":{"start":{"line":312,"column":48},"end":{"line":312,"column":80}},"line":312}},"branchMap":{"0":{"loc":{"start":{"line":83,"column":6},"end":{"line":98,"column":7}},"type":"if","locations":[{"start":{"line":83,"column":6},"end":{"line":98,"column":7}},{"start":{"line":90,"column":13},"end":{"line":98,"column":7}}],"line":83},"1":{"loc":{"start":{"line":84,"column":8},"end":{"line":89,"column":9}},"type":"if","locations":[{"start":{"line":84,"column":8},"end":{"line":89,"column":9}},{"start":{"line":87,"column":15},"end":{"line":89,"column":9}}],"line":84},"2":{"loc":{"start":{"line":90,"column":13},"end":{"line":98,"column":7}},"type":"if","locations":[{"start":{"line":90,"column":13},"end":{"line":98,"column":7}},{"start":{"line":96,"column":13},"end":{"line":98,"column":7}}],"line":90},"3":{"loc":{"start":{"line":91,"column":8},"end":{"line":95,"column":9}},"type":"if","locations":[{"start":{"line":91,"column":8},"end":{"line":95,"column":9}},{"start":{"line":93,"column":15},"end":{"line":95,"column":9}}],"line":91},"4":{"loc":{"start":{"line":145,"column":17},"end":{"line":145,"column":51}},"type":"binary-expr","locations":[{"start":{"line":145,"column":17},"end":{"line":145,"column":29}},{"start":{"line":145,"column":33},"end":{"line":145,"column":51}}],"line":145},"5":{"loc":{"start":{"line":146,"column":19},"end":{"line":146,"column":58}},"type":"binary-expr","locations":[{"start":{"line":146,"column":19},"end":{"line":146,"column":34}},{"start":{"line":146,"column":38},"end":{"line":146,"column":58}}],"line":146},"6":{"loc":{"start":{"line":147,"column":24},"end":{"line":147,"column":73}},"type":"binary-expr","locations":[{"start":{"line":147,"column":24},"end":{"line":147,"column":44}},{"start":{"line":147,"column":48},"end":{"line":147,"column":73}}],"line":147},"7":{"loc":{"start":{"line":148,"column":21},"end":{"line":148,"column":63}},"type":"binary-expr","locations":[{"start":{"line":148,"column":21},"end":{"line":148,"column":37}},{"start":{"line":148,"column":41},"end":{"line":148,"column":63}}],"line":148},"8":{"loc":{"start":{"line":149,"column":4},"end":{"line":151,"column":5}},"type":"if","locations":[{"start":{"line":149,"column":4},"end":{"line":151,"column":5}},{"start":{},"end":{}}],"line":149},"9":{"loc":{"start":{"line":169,"column":6},"end":{"line":174,"column":7}},"type":"if","locations":[{"start":{"line":169,"column":6},"end":{"line":174,"column":7}},{"start":{"line":172,"column":13},"end":{"line":174,"column":7}}],"line":169},"10":{"loc":{"start":{"line":186,"column":6},"end":{"line":191,"column":7}},"type":"if","locations":[{"start":{"line":186,"column":6},"end":{"line":191,"column":7}},{"start":{"line":188,"column":13},"end":{"line":191,"column":7}}],"line":186},"11":{"loc":{"start":{"line":197,"column":6},"end":{"line":199,"column":7}},"type":"if","locations":[{"start":{"line":197,"column":6},"end":{"line":199,"column":7}},{"start":{},"end":{}}],"line":197},"12":{"loc":{"start":{"line":249,"column":24},"end":{"line":249,"column":45}},"type":"cond-expr","locations":[{"start":{"line":249,"column":35},"end":{"line":249,"column":39}},{"start":{"line":249,"column":42},"end":{"line":249,"column":45}}],"line":249},"13":{"loc":{"start":{"line":260,"column":4},"end":{"line":262,"column":5}},"type":"if","locations":[{"start":{"line":260,"column":4},"end":{"line":262,"column":5}},{"start":{},"end":{}}],"line":260},"14":{"loc":{"start":{"line":296,"column":6},"end":{"line":301,"column":7}},"type":"if","locations":[{"start":{"line":296,"column":6},"end":{"line":301,"column":7}},{"start":{"line":299,"column":13},"end":{"line":301,"column":7}}],"line":296},"15":{"loc":{"start":{"line":306,"column":4},"end":{"line":308,"column":5}},"type":"if","locations":[{"start":{"line":306,"column":4},"end":{"line":308,"column":5}},{"start":{},"end":{}}],"line":306},"16":{"loc":{"start":{"line":312,"column":17},"end":{"line":312,"column":33}},"type":"default-arg","locations":[{"start":{"line":312,"column":29},"end":{"line":312,"column":33}}],"line":312}},"s":{"0":1,"1":1,"2":1,"3":1,"4":1,"5":1,"6":1,"7":1,"8":1,"9":1,"10":1,"11":1,"12":1,"13":1,"14":0,"15":0,"16":0,"17":1,"18":1,"19":1,"20":8,"21":4,"22":2,"23":2,"24":2,"25":4,"26":2,"27":1,"28":1,"29":2,"30":1,"31":1,"32":1,"33":1,"34":1,"35":1,"36":1,"37":1,"38":0,"39":1,"40":1,"41":1,"42":1,"43":2,"44":108,"45":2,"46":1,"47":1,"48":2,"49":2,"50":2,"51":1,"52":2,"53":2,"54":2,"55":2,"56":2,"57":2,"58":2,"59":2,"60":2,"61":2,"62":2,"63":26,"64":2,"65":2,"66":2,"67":2,"68":54,"69":54,"70":54,"71":54,"72":54,"73":54,"74":0,"75":54,"76":54,"77":54,"78":54,"79":54,"80":54,"81":54,"82":54,"83":54,"84":24,"85":30,"86":30,"87":2,"88":2,"89":30,"90":1861,"91":30,"92":30,"93":2,"94":2,"95":2,"96":2,"97":2,"98":0,"99":0,"100":0,"101":0,"102":0,"103":0,"104":0,"105":0,"106":0,"107":0,"108":0,"109":0,"110":0,"111":0,"112":0,"113":0,"114":0,"115":0,"116":0,"117":0,"118":0,"119":0,"120":0,"121":0,"122":0,"123":0,"124":0,"125":0,"126":0,"127":0,"128":0,"129":0,"130":0,"131":0,"132":0,"133":0,"134":0,"135":1,"136":1},"f":{"0":1,"1":0,"2":0,"3":0,"4":1,"5":8,"6":1,"7":1,"8":2,"9":108,"10":2,"11":26,"12":30,"13":1861,"14":2,"15":2,"16":0,"17":0,"18":0,"19":0,"20":0,"21":1},"b":{"0":[4,4],"1":[2,2],"2":[2,2],"3":[1,1],"4":[2,1],"5":[2,1],"6":[2,1],"7":[2,2],"8":[2,0],"9":[54,0],"10":[24,30],"11":[30,0],"12":[0,0],"13":[0,0],"14":[0,0],"15":[0,0],"16":[0]},"_coverageSchema":"1a1c01bbd47fc00a2c39e90264f33305004495a9","hash":"2dcd353da921bd16b99f55884619d506852a5916"}
+}
diff --git a/coverage/lcov-report/base.css b/coverage/lcov-report/base.css
new file mode 100644
index 000000000..f418035b4
--- /dev/null
+++ b/coverage/lcov-report/base.css
@@ -0,0 +1,224 @@
+body, html {
+ margin:0; padding: 0;
+ height: 100%;
+}
+body {
+ font-family: Helvetica Neue, Helvetica, Arial;
+ font-size: 14px;
+ color:#333;
+}
+.small { font-size: 12px; }
+*, *:after, *:before {
+ -webkit-box-sizing:border-box;
+ -moz-box-sizing:border-box;
+ box-sizing:border-box;
+ }
+h1 { font-size: 20px; margin: 0;}
+h2 { font-size: 14px; }
+pre {
+ font: 12px/1.4 Consolas, "Liberation Mono", Menlo, Courier, monospace;
+ margin: 0;
+ padding: 0;
+ -moz-tab-size: 2;
+ -o-tab-size: 2;
+ tab-size: 2;
+}
+a { color:#0074D9; text-decoration:none; }
+a:hover { text-decoration:underline; }
+.strong { font-weight: bold; }
+.space-top1 { padding: 10px 0 0 0; }
+.pad2y { padding: 20px 0; }
+.pad1y { padding: 10px 0; }
+.pad2x { padding: 0 20px; }
+.pad2 { padding: 20px; }
+.pad1 { padding: 10px; }
+.space-left2 { padding-left:55px; }
+.space-right2 { padding-right:20px; }
+.center { text-align:center; }
+.clearfix { display:block; }
+.clearfix:after {
+ content:'';
+ display:block;
+ height:0;
+ clear:both;
+ visibility:hidden;
+ }
+.fl { float: left; }
+@media only screen and (max-width:640px) {
+ .col3 { width:100%; max-width:100%; }
+ .hide-mobile { display:none!important; }
+}
+
+.quiet {
+ color: #7f7f7f;
+ color: rgba(0,0,0,0.5);
+}
+.quiet a { opacity: 0.7; }
+
+.fraction {
+ font-family: Consolas, 'Liberation Mono', Menlo, Courier, monospace;
+ font-size: 10px;
+ color: #555;
+ background: #E8E8E8;
+ padding: 4px 5px;
+ border-radius: 3px;
+ vertical-align: middle;
+}
+
+div.path a:link, div.path a:visited { color: #333; }
+table.coverage {
+ border-collapse: collapse;
+ margin: 10px 0 0 0;
+ padding: 0;
+}
+
+table.coverage td {
+ margin: 0;
+ padding: 0;
+ vertical-align: top;
+}
+table.coverage td.line-count {
+ text-align: right;
+ padding: 0 5px 0 20px;
+}
+table.coverage td.line-coverage {
+ text-align: right;
+ padding-right: 10px;
+ min-width:20px;
+}
+
+table.coverage td span.cline-any {
+ display: inline-block;
+ padding: 0 5px;
+ width: 100%;
+}
+.missing-if-branch {
+ display: inline-block;
+ margin-right: 5px;
+ border-radius: 3px;
+ position: relative;
+ padding: 0 4px;
+ background: #333;
+ color: yellow;
+}
+
+.skip-if-branch {
+ display: none;
+ margin-right: 10px;
+ position: relative;
+ padding: 0 4px;
+ background: #ccc;
+ color: white;
+}
+.missing-if-branch .typ, .skip-if-branch .typ {
+ color: inherit !important;
+}
+.coverage-summary {
+ border-collapse: collapse;
+ width: 100%;
+}
+.coverage-summary tr { border-bottom: 1px solid #bbb; }
+.keyline-all { border: 1px solid #ddd; }
+.coverage-summary td, .coverage-summary th { padding: 10px; }
+.coverage-summary tbody { border: 1px solid #bbb; }
+.coverage-summary td { border-right: 1px solid #bbb; }
+.coverage-summary td:last-child { border-right: none; }
+.coverage-summary th {
+ text-align: left;
+ font-weight: normal;
+ white-space: nowrap;
+}
+.coverage-summary th.file { border-right: none !important; }
+.coverage-summary th.pct { }
+.coverage-summary th.pic,
+.coverage-summary th.abs,
+.coverage-summary td.pct,
+.coverage-summary td.abs { text-align: right; }
+.coverage-summary td.file { white-space: nowrap; }
+.coverage-summary td.pic { min-width: 120px !important; }
+.coverage-summary tfoot td { }
+
+.coverage-summary .sorter {
+ height: 10px;
+ width: 7px;
+ display: inline-block;
+ margin-left: 0.5em;
+ background: url(sort-arrow-sprite.png) no-repeat scroll 0 0 transparent;
+}
+.coverage-summary .sorted .sorter {
+ background-position: 0 -20px;
+}
+.coverage-summary .sorted-desc .sorter {
+ background-position: 0 -10px;
+}
+.status-line { height: 10px; }
+/* yellow */
+.cbranch-no { background: yellow !important; color: #111; }
+/* dark red */
+.red.solid, .status-line.low, .low .cover-fill { background:#C21F39 }
+.low .chart { border:1px solid #C21F39 }
+.highlighted,
+.highlighted .cstat-no, .highlighted .fstat-no, .highlighted .cbranch-no{
+ background: #C21F39 !important;
+}
+/* medium red */
+.cstat-no, .fstat-no, .cbranch-no, .cbranch-no { background:#F6C6CE }
+/* light red */
+.low, .cline-no { background:#FCE1E5 }
+/* light green */
+.high, .cline-yes { background:rgb(230,245,208) }
+/* medium green */
+.cstat-yes { background:rgb(161,215,106) }
+/* dark green */
+.status-line.high, .high .cover-fill { background:rgb(77,146,33) }
+.high .chart { border:1px solid rgb(77,146,33) }
+/* dark yellow (gold) */
+.status-line.medium, .medium .cover-fill { background: #f9cd0b; }
+.medium .chart { border:1px solid #f9cd0b; }
+/* light yellow */
+.medium { background: #fff4c2; }
+
+.cstat-skip { background: #ddd; color: #111; }
+.fstat-skip { background: #ddd; color: #111 !important; }
+.cbranch-skip { background: #ddd !important; color: #111; }
+
+span.cline-neutral { background: #eaeaea; }
+
+.coverage-summary td.empty {
+ opacity: .5;
+ padding-top: 4px;
+ padding-bottom: 4px;
+ line-height: 1;
+ color: #888;
+}
+
+.cover-fill, .cover-empty {
+ display:inline-block;
+ height: 12px;
+}
+.chart {
+ line-height: 0;
+}
+.cover-empty {
+ background: white;
+}
+.cover-full {
+ border-right: none !important;
+}
+pre.prettyprint {
+ border: none !important;
+ padding: 0 !important;
+ margin: 0 !important;
+}
+.com { color: #999 !important; }
+.ignore-none { color: #999; font-weight: normal; }
+
+.wrapper {
+ min-height: 100%;
+ height: auto !important;
+ height: 100%;
+ margin: 0 auto -48px;
+}
+.footer, .push {
+ height: 48px;
+}
diff --git a/coverage/lcov-report/block-navigation.js b/coverage/lcov-report/block-navigation.js
new file mode 100644
index 000000000..cc1213023
--- /dev/null
+++ b/coverage/lcov-report/block-navigation.js
@@ -0,0 +1,87 @@
+/* eslint-disable */
+var jumpToCode = (function init() {
+ // Classes of code we would like to highlight in the file view
+ var missingCoverageClasses = ['.cbranch-no', '.cstat-no', '.fstat-no'];
+
+ // Elements to highlight in the file listing view
+ var fileListingElements = ['td.pct.low'];
+
+ // We don't want to select elements that are direct descendants of another match
+ var notSelector = ':not(' + missingCoverageClasses.join('):not(') + ') > '; // becomes `:not(a):not(b) > `
+
+ // Selecter that finds elements on the page to which we can jump
+ var selector =
+ fileListingElements.join(', ') +
+ ', ' +
+ notSelector +
+ missingCoverageClasses.join(', ' + notSelector); // becomes `:not(a):not(b) > a, :not(a):not(b) > b`
+
+ // The NodeList of matching elements
+ var missingCoverageElements = document.querySelectorAll(selector);
+
+ var currentIndex;
+
+ function toggleClass(index) {
+ missingCoverageElements
+ .item(currentIndex)
+ .classList.remove('highlighted');
+ missingCoverageElements.item(index).classList.add('highlighted');
+ }
+
+ function makeCurrent(index) {
+ toggleClass(index);
+ currentIndex = index;
+ missingCoverageElements.item(index).scrollIntoView({
+ behavior: 'smooth',
+ block: 'center',
+ inline: 'center'
+ });
+ }
+
+ function goToPrevious() {
+ var nextIndex = 0;
+ if (typeof currentIndex !== 'number' || currentIndex === 0) {
+ nextIndex = missingCoverageElements.length - 1;
+ } else if (missingCoverageElements.length > 1) {
+ nextIndex = currentIndex - 1;
+ }
+
+ makeCurrent(nextIndex);
+ }
+
+ function goToNext() {
+ var nextIndex = 0;
+
+ if (
+ typeof currentIndex === 'number' &&
+ currentIndex < missingCoverageElements.length - 1
+ ) {
+ nextIndex = currentIndex + 1;
+ }
+
+ makeCurrent(nextIndex);
+ }
+
+ return function jump(event) {
+ if (
+ document.getElementById('fileSearch') === document.activeElement &&
+ document.activeElement != null
+ ) {
+ // if we're currently focused on the search input, we don't want to navigate
+ return;
+ }
+
+ switch (event.which) {
+ case 78: // n
+ case 74: // j
+ goToNext();
+ break;
+ case 66: // b
+ case 75: // k
+ case 80: // p
+ goToPrevious();
+ break;
+ }
+ };
+})();
+window.addEventListener('keydown', jumpToCode);
diff --git a/coverage/lcov-report/favicon.png b/coverage/lcov-report/favicon.png
new file mode 100644
index 000000000..669181783
Binary files /dev/null and b/coverage/lcov-report/favicon.png differ
diff --git a/coverage/lcov-report/index.html b/coverage/lcov-report/index.html
new file mode 100644
index 000000000..ddec29596
--- /dev/null
+++ b/coverage/lcov-report/index.html
@@ -0,0 +1,116 @@
+
+
+
+
+
+ Code coverage report for All files
+
+
+
+
+
+
+
+
+
+
+
+
All files
+
+
+
+ 69.34%
+ Statements
+ 95/137
+
+
+
+
+ 63.63%
+ Branches
+ 21/33
+
+
+
+
+ 63.63%
+ Functions
+ 14/22
+
+
+
+
+ 70%
+ Lines
+ 91/130
+
+
+
+
+
+ Press n or j to go to the next uncovered block, b , p or k for the previous block.
+
+
+
+ Filter:
+
+
+
+
+
+
+
+
+
+ File
+
+ Statements
+
+ Branches
+
+ Functions
+
+ Lines
+
+
+
+
+ index.js
+
+
+
+ 69.34%
+ 95/137
+ 63.63%
+ 21/33
+ 63.63%
+ 14/22
+ 70%
+ 91/130
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/coverage/lcov-report/index.js.html b/coverage/lcov-report/index.js.html
new file mode 100644
index 000000000..87e7cf884
--- /dev/null
+++ b/coverage/lcov-report/index.js.html
@@ -0,0 +1,1027 @@
+
+
+
+
+
+ Code coverage report for index.js
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ 69.34%
+ Statements
+ 95/137
+
+
+
+
+ 63.63%
+ Branches
+ 21/33
+
+
+
+
+ 63.63%
+ Functions
+ 14/22
+
+
+
+
+ 70%
+ Lines
+ 91/130
+
+
+
+
+
+ Press n or j to go to the next uncovered block, b , p or k for the previous block.
+
+
+
+ Filter:
+
+
+
+
+
+
+1
+2
+3
+4
+5
+6
+7
+8
+9
+10
+11
+12
+13
+14
+15
+16
+17
+18
+19
+20
+21
+22
+23
+24
+25
+26
+27
+28
+29
+30
+31
+32
+33
+34
+35
+36
+37
+38
+39
+40
+41
+42
+43
+44
+45
+46
+47
+48
+49
+50
+51
+52
+53
+54
+55
+56
+57
+58
+59
+60
+61
+62
+63
+64
+65
+66
+67
+68
+69
+70
+71
+72
+73
+74
+75
+76
+77
+78
+79
+80
+81
+82
+83
+84
+85
+86
+87
+88
+89
+90
+91
+92
+93
+94
+95
+96
+97
+98
+99
+100
+101
+102
+103
+104
+105
+106
+107
+108
+109
+110
+111
+112
+113
+114
+115
+116
+117
+118
+119
+120
+121
+122
+123
+124
+125
+126
+127
+128
+129
+130
+131
+132
+133
+134
+135
+136
+137
+138
+139
+140
+141
+142
+143
+144
+145
+146
+147
+148
+149
+150
+151
+152
+153
+154
+155
+156
+157
+158
+159
+160
+161
+162
+163
+164
+165
+166
+167
+168
+169
+170
+171
+172
+173
+174
+175
+176
+177
+178
+179
+180
+181
+182
+183
+184
+185
+186
+187
+188
+189
+190
+191
+192
+193
+194
+195
+196
+197
+198
+199
+200
+201
+202
+203
+204
+205
+206
+207
+208
+209
+210
+211
+212
+213
+214
+215
+216
+217
+218
+219
+220
+221
+222
+223
+224
+225
+226
+227
+228
+229
+230
+231
+232
+233
+234
+235
+236
+237
+238
+239
+240
+241
+242
+243
+244
+245
+246
+247
+248
+249
+250
+251
+252
+253
+254
+255
+256
+257
+258
+259
+260
+261
+262
+263
+264
+265
+266
+267
+268
+269
+270
+271
+272
+273
+274
+275
+276
+277
+278
+279
+280
+281
+282
+283
+284
+285
+286
+287
+288
+289
+290
+291
+292
+293
+294
+295
+296
+297
+298
+299
+300
+301
+302
+303
+304
+305
+306
+307
+308
+309
+310
+311
+312
+313
+314
+315
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+1x
+1x
+1x
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+1x
+
+
+
+
+
+
+1x
+1x
+1x
+1x
+
+
+
+
+
+1x
+1x
+1x
+1x
+1x
+
+
+
+
+
+
+1x
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+1x
+1x
+1x
+8x
+4x
+2x
+2x
+
+2x
+
+4x
+2x
+1x
+
+1x
+
+
+2x
+
+
+1x
+1x
+1x
+
+
+
+1x
+1x
+1x
+1x
+1x
+
+
+
+
+
+
+1x
+1x
+1x
+
+1x
+2x
+108x
+
+
+
+
+
+
+
+2x
+
+
+1x
+2x
+2x
+2x
+
+
+1x
+
+
+
+2x
+2x
+2x
+2x
+2x
+2x
+2x
+
+
+2x
+2x
+2x
+
+2x
+26x
+
+
+2x
+2x
+
+2x
+54x
+54x
+54x
+
+54x
+54x
+54x
+
+
+
+
+54x
+54x
+
+54x
+54x
+54x
+54x
+54x
+54x
+
+54x
+24x
+
+30x
+30x
+
+
+
+2x
+2x
+1861x
+30x
+30x
+
+
+2x
+2x
+
+
+
+
+
+
+
+
+
+2x
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+2x
+2x
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+1x
+
+
+ // Copyright (c) 2018 ml5
+//
+// This software is released under the MIT License.
+// https://opensource.org/licenses/MIT
+
+/* eslint prefer-destructuring: ["error", {AssignmentExpression: {array: false}}] */
+/* eslint no-await-in-loop: "off" */
+/*
+A LSTM Generator: Run inference mode for a pre-trained LSTM.
+*/
+
+import * as tf from "@tensorflow/tfjs";
+import axios from "axios";
+import sampleFromDistribution from "./../utils/sample";
+import CheckpointLoader from "../utils/checkpointLoader";
+import callCallback from "../utils/callcallback";
+
+const regexCell = /cell_[0-9]|lstm_[0-9]/gi;
+const regexWeights = /weights|weight|kernel|kernels|w/gi;
+const regexFullyConnected = /softmax/gi;
+
+class CharRNN {
+ /**
+ * Create a CharRNN.
+ * @param {String} modelPath - The path to the trained charRNN model.
+ * @param {function} callback - Optional. A callback to be called once
+ * the model has loaded. If no callback is provided, it will return a
+ * promise that will be resolved once the model has loaded.
+ */
+ constructor(modelPath, callback) {
+ /**
+ * Boolean value that specifies if the model has loaded.
+ * @type {boolean}
+ * @public
+ */
+ this.ready = false;
+
+ /**
+ * The pre-trained charRNN model.
+ * @type {model}
+ * @public
+ */
+ this.model = {};
+ this.cellsAmount = 0;
+ this.cells = [];
+ this.zeroState = { c: [], h: [] };
+ /**
+ * The vocabulary size (or total number of possible characters).
+ * @type {c: Array, h: Array}
+ * @public
+ */
+ this.state = { c: [], h: [] };
+ this.vocab = {};
+ this.vocabSize = 0;
+ this.probabilities = [];
+ this.defaults = {
+ seed: "a", // TODO: use no seed by default
+ length: 20,
+ temperature: 0.5,
+ stateful: false,
+ };
+
+ this.ready = callCallback(this.loadCheckpoints(modelPath), callback);
+ // this.then = this.ready.then.bind(this.ready);
+ }
+
+ re setState() {
+ this.state = this.zeroState;
+ }
+
+ se tState(state) {
+ this.state = state;
+ }
+
+ ge tState() {
+ return this.state;
+ }
+
+ async loadCheckpoints(path) {
+ const reader = new CheckpointLoader(path);
+ const vars = await reader.getAllVariables();
+ Object.keys(vars).forEach(key => {
+ if (key.match(regexCell)) {
+ if (key.match(regexWeights)) {
+ this.model[`Kernel_${key.match(/[0-9]/)[0]}`] = vars[key];
+ this.cellsAmount += 1;
+ } else {
+ this.model[`Bias_${key.match(/[0-9]/)[0]}`] = vars[key];
+ }
+ } else if (key.match(regexFullyConnected)) {
+ if (key.match(regexWeights)) {
+ this.model.fullyConnectedWeights = vars[key];
+ } else {
+ this.model.fullyConnectedBiases = vars[key];
+ }
+ } else {
+ this.model[key] = vars[key];
+ }
+ });
+ await this.loadVocab(path);
+ await this.initCells();
+ return this;
+ }
+
+ async loadVocab(path) {
+ try {
+ const { data } = await axios.get(`${path}/vocab.json`);
+ this.vocab = data;
+ this.vocabSize = Object.keys(data).length;
+ return this.vocab;
+ } catch (err) {
+ return err;
+ }
+ }
+
+ async initCells() {
+ this.cells = [];
+ this.zeroState = { c: [], h: [] };
+ const forgetBias = tf.tensor(1.0);
+
+ const lstm = i => {
+ const cell = (DATA, C, H) =>
+ tf.basicLSTMCell(
+ forgetBias,
+ this.model[`Kernel_${i}`],
+ this.model[`Bias_${i}`],
+ DATA,
+ C,
+ H,
+ );
+ return cell;
+ };
+
+ for (let i = 0; i < this.cellsAmount; i += 1) {
+ this.zeroState.c.push(tf.zeros([1, this.model[`Bias_${i}`].shape[0] / 4]));
+ this.zeroState.h.push(tf.zeros([1, this.model[`Bias_${i}`].shape[0] / 4]));
+ this.cells.push(lstm(i));
+ }
+
+ this.state = this.zeroState;
+ }
+
+ async generateInternal(options) {
+ await this.ready;
+ const seed = options.seed || this.defaults.seed;
+ const length = +options.length || this.defaults.length;
+ const temperature = +options.temperature || this.defaults.temperature;
+ const stateful = options.stateful || this.defaults.stateful;
+ if (!stateful) {
+ this.state = this.zeroState;
+ }
+
+ const results = [];
+ const userInput = Array.from(seed);
+ const encodedInput = [];
+
+ userInput.forEach(char => {
+ encodedInput.push(this.vocab[char]);
+ });
+
+ let input = encodedInput[0];
+ let probabilitiesNormalized = []; // will contain final probabilities (normalized)
+
+ for (let i = 0; i < userInput.length + length + -1; i += 1) {
+ const onehotBuffer = await tf.buffer([1, this.vocabSize]);
+ onehotBuffer.set(1.0, 0, input);
+ const onehot = onehotBuffer.toTensor();
+ let output;
+ if (this.model.embedding) {
+ const embedded = tf.matMul(onehot, this.model.embedding);
+ output = tf.multiRNNCell(this.cells, embedded, this.state.c, this.state.h);
+ } else E {
+ output = tf.multiRNNCell(this.cells, onehot, this.state.c, this.state.h);
+ }
+
+ this.state.c = output[0];
+ this.state.h = output[1];
+
+ const outputH = this.state.h[1];
+ const weightedResult = tf.matMul(outputH, this.model.fullyConnectedWeights);
+ const logits = tf.add(weightedResult, this.model.fullyConnectedBiases);
+ const divided = tf.div(logits, tf.tensor(temperature));
+ const probabilities = tf.exp(divided);
+ probabilitiesNormalized = await tf.div(probabilities, tf.sum(probabilities)).data();
+
+ if (i < userInput.length - 1) {
+ input = encodedInput[i + 1];
+ } else {
+ input = sampleFromDistribution(probabilitiesNormalized);
+ results.push(input);
+ }
+ }
+
+ let generated = "";
+ results.forEach(char => {
+ const mapped = Object.keys(this.vocab).find(key => this.vocab[key] === char);
+ if (mapped) {
+ generated += mapped;
+ }
+ });
+ this.probabilities = probabilitiesNormalized;
+ return {
+ sample: generated,
+ state: this.state,
+ };
+ }
+
+ /**
+ * Reset the model state.
+ */
+ reset() {
+ this.state = this.zeroState;
+ }
+
+ /**
+ * @typedef {Object} options
+ * @property {String} seed
+ * @property {number} length
+ * @property {number} temperature
+ */
+
+ // stateless
+ /**
+ * Generates content in a stateless manner, based on some initial text
+ * (known as a "seed"). Returns a string.
+ * @param {options} options - An object specifying the input parameters of
+ * seed, length and temperature. Default length is 20, temperature is 0.5
+ * and seed is a random character from the model. The object should look like
+ * this:
+ * @param {function} callback - Optional. A function to be called when the model
+ * has generated content. If no callback is provided, it will return a promise
+ * that will be resolved once the model has generated new content.
+ */
+ async generate(options, callback) {
+ this.reset();
+ return callCallback(this.generateInternal(options), callback);
+ }
+
+ // stateful
+ /**
+ * Predict the next character based on the model's current state.
+ * @param {number} temp
+ * @param {function} callback - Optional. A function to be called when the
+ * model finished adding the seed. If no callback is provided, it will
+ * return a promise that will be resolved once the prediction has been generated.
+ */
+ as ync predict(temp, callback) {
+ let probabilitiesNormalized = [];
+ const temperature = temp > 0 ? temp : 0.1;
+ const outputH = this.state.h[1];
+ const weightedResult = tf.matMul(outputH, this.model.fullyConnectedWeights);
+ const logits = tf.add(weightedResult, this.model.fullyConnectedBiases);
+ const divided = tf.div(logits, tf.tensor(temperature));
+ const probabilities = tf.exp(divided);
+ probabilitiesNormalized = await tf.div(probabilities, tf.sum(probabilities)).data();
+
+ const sample = sampleFromDistribution(probabilitiesNormalized);
+ const result = Object.keys(this.vocab).find(ke y => this.vocab[key] === sample) ;
+ this.probabilities = probabilitiesNormalized;
+ if (callback) {
+ callback(result);
+ }
+ /* eslint max-len: ["error", { "code": 180 }] */
+ const pm = Object.keys(this.vocab).map(c => ({
+ char: c,
+ probability: this.probabilities[this.vocab[c]],
+ }));
+ return {
+ sample: result,
+ probabilities: pm,
+ };
+ }
+
+ /**
+ * Feed a string of characters to the model state.
+ * @param {String} inputSeed - A string to feed the charRNN model state.
+ * @param {function} callback - Optional. A function to be called when
+ * the model finished adding the seed. If no callback is provided, it
+ * will return a promise that will be resolved once seed has been fed.
+ */
+ as ync feed(inputSeed, callback) {
+ await this.ready;
+ const seed = Array.from(inputSeed);
+ const encodedInput = [];
+
+ seed.forEach(ch ar => {
+ encodedInput.push(this.vocab[char]);
+ });
+
+ let input = encodedInput[0];
+ for (let i = 0; i < seed.length; i += 1) {
+ const onehotBuffer = await tf.buffer([1, this.vocabSize]);
+ onehotBuffer.set(1.0, 0, input);
+ const onehot = onehotBuffer.toTensor();
+ let output;
+ if (this.model.embedding) {
+ const embedded = tf.matMul(onehot, this.model.embedding);
+ output = tf.multiRNNCell(this.cells, embedded, this.state.c, this.state.h);
+ } else {
+ output = tf.multiRNNCell(this.cells, onehot, this.state.c, this.state.h);
+ }
+ this.state.c = output[0];
+ this.state.h = output[1];
+ input = encodedInput[i];
+ }
+ if (callback) {
+ callback();
+ }
+ }
+}
+
+const charRNN = (modelPath = "./", callback) => new CharRNN(modelPath, callback);
+
+export default charRNN;
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/coverage/lcov-report/prettify.css b/coverage/lcov-report/prettify.css
new file mode 100644
index 000000000..b317a7cda
--- /dev/null
+++ b/coverage/lcov-report/prettify.css
@@ -0,0 +1 @@
+.pln{color:#000}@media screen{.str{color:#080}.kwd{color:#008}.com{color:#800}.typ{color:#606}.lit{color:#066}.pun,.opn,.clo{color:#660}.tag{color:#008}.atn{color:#606}.atv{color:#080}.dec,.var{color:#606}.fun{color:red}}@media print,projection{.str{color:#060}.kwd{color:#006;font-weight:bold}.com{color:#600;font-style:italic}.typ{color:#404;font-weight:bold}.lit{color:#044}.pun,.opn,.clo{color:#440}.tag{color:#006;font-weight:bold}.atn{color:#404}.atv{color:#060}}pre.prettyprint{padding:2px;border:1px solid #888}ol.linenums{margin-top:0;margin-bottom:0}li.L0,li.L1,li.L2,li.L3,li.L5,li.L6,li.L7,li.L8{list-style-type:none}li.L1,li.L3,li.L5,li.L7,li.L9{background:#eee}
diff --git a/coverage/lcov-report/prettify.js b/coverage/lcov-report/prettify.js
new file mode 100644
index 000000000..b3225238f
--- /dev/null
+++ b/coverage/lcov-report/prettify.js
@@ -0,0 +1,2 @@
+/* eslint-disable */
+window.PR_SHOULD_USE_CONTINUATION=true;(function(){var h=["break,continue,do,else,for,if,return,while"];var u=[h,"auto,case,char,const,default,double,enum,extern,float,goto,int,long,register,short,signed,sizeof,static,struct,switch,typedef,union,unsigned,void,volatile"];var p=[u,"catch,class,delete,false,import,new,operator,private,protected,public,this,throw,true,try,typeof"];var l=[p,"alignof,align_union,asm,axiom,bool,concept,concept_map,const_cast,constexpr,decltype,dynamic_cast,explicit,export,friend,inline,late_check,mutable,namespace,nullptr,reinterpret_cast,static_assert,static_cast,template,typeid,typename,using,virtual,where"];var x=[p,"abstract,boolean,byte,extends,final,finally,implements,import,instanceof,null,native,package,strictfp,super,synchronized,throws,transient"];var R=[x,"as,base,by,checked,decimal,delegate,descending,dynamic,event,fixed,foreach,from,group,implicit,in,interface,internal,into,is,lock,object,out,override,orderby,params,partial,readonly,ref,sbyte,sealed,stackalloc,string,select,uint,ulong,unchecked,unsafe,ushort,var"];var r="all,and,by,catch,class,else,extends,false,finally,for,if,in,is,isnt,loop,new,no,not,null,of,off,on,or,return,super,then,true,try,unless,until,when,while,yes";var w=[p,"debugger,eval,export,function,get,null,set,undefined,var,with,Infinity,NaN"];var s="caller,delete,die,do,dump,elsif,eval,exit,foreach,for,goto,if,import,last,local,my,next,no,our,print,package,redo,require,sub,undef,unless,until,use,wantarray,while,BEGIN,END";var I=[h,"and,as,assert,class,def,del,elif,except,exec,finally,from,global,import,in,is,lambda,nonlocal,not,or,pass,print,raise,try,with,yield,False,True,None"];var f=[h,"alias,and,begin,case,class,def,defined,elsif,end,ensure,false,in,module,next,nil,not,or,redo,rescue,retry,self,super,then,true,undef,unless,until,when,yield,BEGIN,END"];var H=[h,"case,done,elif,esac,eval,fi,function,in,local,set,then,until"];var A=[l,R,w,s+I,f,H];var e=/^(DIR|FILE|vector|(de|priority_)?queue|list|stack|(const_)?iterator|(multi)?(set|map)|bitset|u?(int|float)\d*)/;var C="str";var z="kwd";var j="com";var O="typ";var G="lit";var L="pun";var F="pln";var m="tag";var E="dec";var J="src";var P="atn";var n="atv";var N="nocode";var M="(?:^^\\.?|[+-]|\\!|\\!=|\\!==|\\#|\\%|\\%=|&|&&|&&=|&=|\\(|\\*|\\*=|\\+=|\\,|\\-=|\\->|\\/|\\/=|:|::|\\;|<|<<|<<=|<=|=|==|===|>|>=|>>|>>=|>>>|>>>=|\\?|\\@|\\[|\\^|\\^=|\\^\\^|\\^\\^=|\\{|\\||\\|=|\\|\\||\\|\\|=|\\~|break|case|continue|delete|do|else|finally|instanceof|return|throw|try|typeof)\\s*";function k(Z){var ad=0;var S=false;var ac=false;for(var V=0,U=Z.length;V122)){if(!(al<65||ag>90)){af.push([Math.max(65,ag)|32,Math.min(al,90)|32])}if(!(al<97||ag>122)){af.push([Math.max(97,ag)&~32,Math.min(al,122)&~32])}}}}af.sort(function(av,au){return(av[0]-au[0])||(au[1]-av[1])});var ai=[];var ap=[NaN,NaN];for(var ar=0;arat[0]){if(at[1]+1>at[0]){an.push("-")}an.push(T(at[1]))}}an.push("]");return an.join("")}function W(al){var aj=al.source.match(new RegExp("(?:\\[(?:[^\\x5C\\x5D]|\\\\[\\s\\S])*\\]|\\\\u[A-Fa-f0-9]{4}|\\\\x[A-Fa-f0-9]{2}|\\\\[0-9]+|\\\\[^ux0-9]|\\(\\?[:!=]|[\\(\\)\\^]|[^\\x5B\\x5C\\(\\)\\^]+)","g"));var ah=aj.length;var an=[];for(var ak=0,am=0;ak=2&&ai==="["){aj[ak]=X(ag)}else{if(ai!=="\\"){aj[ak]=ag.replace(/[a-zA-Z]/g,function(ao){var ap=ao.charCodeAt(0);return"["+String.fromCharCode(ap&~32,ap|32)+"]"})}}}}return aj.join("")}var aa=[];for(var V=0,U=Z.length;V=0;){S[ac.charAt(ae)]=Y}}var af=Y[1];var aa=""+af;if(!ag.hasOwnProperty(aa)){ah.push(af);ag[aa]=null}}ah.push(/[\0-\uffff]/);V=k(ah)})();var X=T.length;var W=function(ah){var Z=ah.sourceCode,Y=ah.basePos;var ad=[Y,F];var af=0;var an=Z.match(V)||[];var aj={};for(var ae=0,aq=an.length;ae=5&&"lang-"===ap.substring(0,5);if(am&&!(ai&&typeof ai[1]==="string")){am=false;ap=J}if(!am){aj[ag]=ap}}var ab=af;af+=ag.length;if(!am){ad.push(Y+ab,ap)}else{var al=ai[1];var ak=ag.indexOf(al);var ac=ak+al.length;if(ai[2]){ac=ag.length-ai[2].length;ak=ac-al.length}var ar=ap.substring(5);B(Y+ab,ag.substring(0,ak),W,ad);B(Y+ab+ak,al,q(ar,al),ad);B(Y+ab+ac,ag.substring(ac),W,ad)}}ah.decorations=ad};return W}function i(T){var W=[],S=[];if(T.tripleQuotedStrings){W.push([C,/^(?:\'\'\'(?:[^\'\\]|\\[\s\S]|\'{1,2}(?=[^\']))*(?:\'\'\'|$)|\"\"\"(?:[^\"\\]|\\[\s\S]|\"{1,2}(?=[^\"]))*(?:\"\"\"|$)|\'(?:[^\\\']|\\[\s\S])*(?:\'|$)|\"(?:[^\\\"]|\\[\s\S])*(?:\"|$))/,null,"'\""])}else{if(T.multiLineStrings){W.push([C,/^(?:\'(?:[^\\\']|\\[\s\S])*(?:\'|$)|\"(?:[^\\\"]|\\[\s\S])*(?:\"|$)|\`(?:[^\\\`]|\\[\s\S])*(?:\`|$))/,null,"'\"`"])}else{W.push([C,/^(?:\'(?:[^\\\'\r\n]|\\.)*(?:\'|$)|\"(?:[^\\\"\r\n]|\\.)*(?:\"|$))/,null,"\"'"])}}if(T.verbatimStrings){S.push([C,/^@\"(?:[^\"]|\"\")*(?:\"|$)/,null])}var Y=T.hashComments;if(Y){if(T.cStyleComments){if(Y>1){W.push([j,/^#(?:##(?:[^#]|#(?!##))*(?:###|$)|.*)/,null,"#"])}else{W.push([j,/^#(?:(?:define|elif|else|endif|error|ifdef|include|ifndef|line|pragma|undef|warning)\b|[^\r\n]*)/,null,"#"])}S.push([C,/^<(?:(?:(?:\.\.\/)*|\/?)(?:[\w-]+(?:\/[\w-]+)+)?[\w-]+\.h|[a-z]\w*)>/,null])}else{W.push([j,/^#[^\r\n]*/,null,"#"])}}if(T.cStyleComments){S.push([j,/^\/\/[^\r\n]*/,null]);S.push([j,/^\/\*[\s\S]*?(?:\*\/|$)/,null])}if(T.regexLiterals){var X=("/(?=[^/*])(?:[^/\\x5B\\x5C]|\\x5C[\\s\\S]|\\x5B(?:[^\\x5C\\x5D]|\\x5C[\\s\\S])*(?:\\x5D|$))+/");S.push(["lang-regex",new RegExp("^"+M+"("+X+")")])}var V=T.types;if(V){S.push([O,V])}var U=(""+T.keywords).replace(/^ | $/g,"");if(U.length){S.push([z,new RegExp("^(?:"+U.replace(/[\s,]+/g,"|")+")\\b"),null])}W.push([F,/^\s+/,null," \r\n\t\xA0"]);S.push([G,/^@[a-z_$][a-z_$@0-9]*/i,null],[O,/^(?:[@_]?[A-Z]+[a-z][A-Za-z_$@0-9]*|\w+_t\b)/,null],[F,/^[a-z_$][a-z_$@0-9]*/i,null],[G,new RegExp("^(?:0x[a-f0-9]+|(?:\\d(?:_\\d+)*\\d*(?:\\.\\d*)?|\\.\\d\\+)(?:e[+\\-]?\\d+)?)[a-z]*","i"),null,"0123456789"],[F,/^\\[\s\S]?/,null],[L,/^.[^\s\w\.$@\'\"\`\/\#\\]*/,null]);return g(W,S)}var K=i({keywords:A,hashComments:true,cStyleComments:true,multiLineStrings:true,regexLiterals:true});function Q(V,ag){var U=/(?:^|\s)nocode(?:\s|$)/;var ab=/\r\n?|\n/;var ac=V.ownerDocument;var S;if(V.currentStyle){S=V.currentStyle.whiteSpace}else{if(window.getComputedStyle){S=ac.defaultView.getComputedStyle(V,null).getPropertyValue("white-space")}}var Z=S&&"pre"===S.substring(0,3);var af=ac.createElement("LI");while(V.firstChild){af.appendChild(V.firstChild)}var W=[af];function ae(al){switch(al.nodeType){case 1:if(U.test(al.className)){break}if("BR"===al.nodeName){ad(al);if(al.parentNode){al.parentNode.removeChild(al)}}else{for(var an=al.firstChild;an;an=an.nextSibling){ae(an)}}break;case 3:case 4:if(Z){var am=al.nodeValue;var aj=am.match(ab);if(aj){var ai=am.substring(0,aj.index);al.nodeValue=ai;var ah=am.substring(aj.index+aj[0].length);if(ah){var ak=al.parentNode;ak.insertBefore(ac.createTextNode(ah),al.nextSibling)}ad(al);if(!ai){al.parentNode.removeChild(al)}}}break}}function ad(ak){while(!ak.nextSibling){ak=ak.parentNode;if(!ak){return}}function ai(al,ar){var aq=ar?al.cloneNode(false):al;var ao=al.parentNode;if(ao){var ap=ai(ao,1);var an=al.nextSibling;ap.appendChild(aq);for(var am=an;am;am=an){an=am.nextSibling;ap.appendChild(am)}}return aq}var ah=ai(ak.nextSibling,0);for(var aj;(aj=ah.parentNode)&&aj.nodeType===1;){ah=aj}W.push(ah)}for(var Y=0;Y=S){ah+=2}if(V>=ap){Z+=2}}}var t={};function c(U,V){for(var S=V.length;--S>=0;){var T=V[S];if(!t.hasOwnProperty(T)){t[T]=U}else{if(window.console){console.warn("cannot override language handler %s",T)}}}}function q(T,S){if(!(T&&t.hasOwnProperty(T))){T=/^\s*]*(?:>|$)/],[j,/^<\!--[\s\S]*?(?:-\->|$)/],["lang-",/^<\?([\s\S]+?)(?:\?>|$)/],["lang-",/^<%([\s\S]+?)(?:%>|$)/],[L,/^(?:<[%?]|[%?]>)/],["lang-",/^]*>([\s\S]+?)<\/xmp\b[^>]*>/i],["lang-js",/^
+
+
+
+
+ DBSCAN Example
+ DBSCAN is a density-based clustering non-parametric algorithm : given a set of points in some space, it groups together points that are closely packed together (points with many nearby neighbors),
+ marking as outliers points that lie alone in low-density regions (whose nearest neighbors are too far away).
+ DBSCAN is one of the most common clustering algorithms and also most cited in scientific literature.
+
+
+
+
+
+