Source: Sleepmeter/engine.js

  1. /*
  2. * Copyright 2020-2022 Sleepdiary Developers <sleepdiary@pileofstuff.org>
  3. *
  4. * Permission is hereby granted, free of charge, to any person
  5. * obtaining a copy of this software and associated documentation
  6. * files (the "Software"), to deal in the Software without
  7. * restriction, including without limitation the rights to use, copy,
  8. * modify, merge, publish, distribute, sublicense, and/or sell copies
  9. * of the Software, and to permit persons to whom the Software is
  10. * furnished to do so, subject to the following conditions:
  11. *
  12. * The above copyright notice and this permission notice shall be
  13. * included in all copies or substantial portions of the Software.
  14. *
  15. * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
  16. * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
  17. * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
  18. * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
  19. * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
  20. * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
  21. * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
  22. * SOFTWARE.
  23. */
  24. "use strict";
  25. /**
  26. * @public
  27. * @unrestricted
  28. * @augments DiaryBase
  29. *
  30. * @example
  31. * let diary = new_sleep_diary(contents_of_my_file));
  32. *
  33. * console.log( diary.custom_aids );
  34. * -> [
  35. * { "custom_aid_id": "CUSTOM_0001", "class": "READING", "name": "The Cat in the Hat" },
  36. * ...
  37. * ]
  38. *
  39. * console.log( diary.custom_hindrances );
  40. * -> [
  41. * { "custom_hindrance_id": "CUSTOM_0001", "class": "OBLIGATION", "name": "Appointment" },
  42. * ...
  43. * ]
  44. *
  45. * console.log( diary.custom_tags );
  46. * -> [
  47. * { "custom_tag_id": "CUSTOM_0001", "name": "ate cheese before bed" },
  48. * ...
  49. * ]
  50. *
  51. * // Print the complete list of records
  52. * console.log(diary.records);
  53. * -> [
  54. * {
  55. *
  56. * // normalised event times - use these where possible:
  57. * "start" : 12345678, // based on "bedtime"
  58. * "end" : 23456789, // based on "wake"
  59. * "duration" : 11111111, // "wake" minus "sleep" and minus "holes"
  60. *
  61. * "wake" : {
  62. * "string": "2001-02-03 04:05+0600",
  63. * "year" : 2001,
  64. * "month" : 2,
  65. * "day" : 3,
  66. * "hour" : 4,
  67. * "minute": 5,
  68. * "offset": 6
  69. * },
  70. * "sleep" : {
  71. * "string": "2001-02-03 04:04+0600",
  72. * "year" : 2001,
  73. * "month" : 2,
  74. * "day" : 3,
  75. * "hour" : 4,
  76. * "minute": 4,
  77. * "offset": 6
  78. * },
  79. * "bedtime" : {
  80. * "string": "2001-02-03 04:00+0600",
  81. * "year" : 2001,
  82. * "month" : 2,
  83. * "day" : 3,
  84. * "hour" : 4,
  85. * "minute": 0,
  86. * "offset": 6
  87. * },
  88. * "holes" : [ { "wake": 0, "sleep": 1 } ],
  89. * "type" : "NAP",
  90. * "dreams" : [
  91. * { "type": "GOOD", "mood": 5, "themes": [ "PRECOGNITIVE", "LUCID" ] }
  92. * ],
  93. * "aids" : [ "CPAP", "CUSTOM_0001" ],
  94. * "hindrances" : [ "ALARM_CLOCK", "CUSTOM_0001" ],
  95. * "tags" : [ "OUT_OF_TOWN", "CUSTOM_0001" ],
  96. * "quality" : 5,
  97. * "notes" : "sleep notes"
  98. * },
  99. * ...
  100. * ]
  101. *
  102. */
  103. class DiarySleepmeter extends DiaryBase {
  104. /**
  105. * @param {Object} file - file contents
  106. * @param {Function=} serialiser - function to serialise output
  107. */
  108. constructor(file,serialiser) {
  109. super(file,serialiser); // call the SleepDiary constructor
  110. /*
  111. * Several parts of the file format have a list of valid values.
  112. * This section defines regular expressions for each of those.
  113. */
  114. const sleep_aid_list = [
  115. "ALCOHOL",
  116. "AMBIEN",
  117. "AMBIEN_CR",
  118. "AROMATHERAPY",
  119. "BENADRYL",
  120. "CHAMOMILE",
  121. "CIRCADIN",
  122. "CPAP",
  123. "DOZILE",
  124. "EAR_PLUGS",
  125. "EXERCISE",
  126. "GABA",
  127. "IMOVANE",
  128. "LUNESTA",
  129. "MAGNESIUM",
  130. "MARIJUANA",
  131. "MEDITATION",
  132. "MELATONIN",
  133. "MILK",
  134. "MUSIC",
  135. "NYQUIL",
  136. "READING",
  137. "RESTAVIT",
  138. "ROZEREM",
  139. "SEX",
  140. "SOUND_MACHINE",
  141. "ST_JOHNS_WORT",
  142. "TV",
  143. "TYLENOL",
  144. "TYLENOL_PM",
  145. "UNISOM",
  146. "UNISOM2",
  147. "VALERIAN",
  148. "ZIMOVANE"
  149. ];
  150. const sleep_hindrance_list = [
  151. "ALARM_CLOCK",
  152. "ANGER",
  153. "ANXIETY",
  154. "ARGUMENT",
  155. "BABY_CRYING",
  156. "BATHROOM_BREAK",
  157. "TOO_BRIGHT",
  158. "BUNKMATE_SNORING",
  159. "CAFFEINE",
  160. "TOO_COLD",
  161. "DOG_BARKING",
  162. "FIRE_ANTS",
  163. "HEARTBURN",
  164. "TOO_HOT",
  165. "HUNGER",
  166. "LOUD_NEIGHBOR",
  167. "MIND_RACING",
  168. "PAIN",
  169. "PHONE_RANG",
  170. "RESTLESS_LEGS",
  171. "SCARY_MOVIE",
  172. "SICK",
  173. "SQUIRRELS_ON_ROOF",
  174. "STORM",
  175. "STRESS",
  176. "SUGAR",
  177. "VIDEO_GAME",
  178. "WIND"
  179. ];
  180. const sleep_tag_list = [
  181. "ALONE",
  182. "BUNKMATE",
  183. "CAMPING",
  184. "COUCH",
  185. "GOING_FISHING",
  186. "HOTEL",
  187. "OUT_OF_TOWN",
  188. "PASSED_OUT_DRUNK",
  189. "SCHOOL_NIGHT",
  190. "SLEEP_TALKING",
  191. "SLEEP_WALKING",
  192. "SLEPT_AT_FRIENDS_PLACE",
  193. "SLEPT_IN_CAR",
  194. "WORK_NIGHT"
  195. ];
  196. const custom = "CUSTOM_[0-9]*";
  197. const sleep_aid = sleep_aid_list .join('|') + "|" + custom;
  198. const sleep_hindrance = sleep_hindrance_list.join('|') + "|" + custom;
  199. const sleep_tag = sleep_tag_list .join('|') + "|" + custom;
  200. var sleep_tag_map = {};
  201. [ sleep_aid_list, sleep_hindrance_list, sleep_tag_list ].forEach(
  202. (list,n) => list.forEach( tag => sleep_tag_map[tag] = n )
  203. );
  204. const sleep_aid_class =
  205. "AIRWAY" +
  206. "|BEVERAGE" +
  207. "|DRUG" +
  208. "|EXERTION" +
  209. "|HERBAL" +
  210. "|READING" +
  211. "|RELAXATION" +
  212. "|SENSORY_DEPRIVATION" +
  213. "|SOUND"
  214. ;
  215. const sleep_hindrance_class =
  216. "ENVIRONMENTAL" +
  217. "|MENTAL" +
  218. "|NOISE" +
  219. "|OBLIGATION" +
  220. "|PHYSICAL" +
  221. "|STIMULANT"
  222. ;
  223. const sleep_type =
  224. "NIGHT_SLEEP" +
  225. "|NAP"
  226. ;
  227. const dream_type =
  228. "UNKNOWN" +
  229. "|GOOD" +
  230. "|EROTIC" +
  231. "|NEUTRAL" +
  232. "|STRANGE" +
  233. "|CREEPY" +
  234. "|TROUBLING" +
  235. "|NIGHTMARE"
  236. ;
  237. const dream_theme =
  238. "CHASE" +
  239. "|COMPENSATORY" +
  240. "|DAILY_LIFE" +
  241. "|DEATH" +
  242. "|EPIC" +
  243. "|FALLING" +
  244. "|FALSE_AWAKENING" +
  245. "|FLYING" +
  246. "|LUCID" +
  247. "|MURDER" +
  248. "|MUTUAL" +
  249. "|NAKED_IN_PUBLIC" +
  250. "|ORGASMIC" +
  251. "|PHYSIOLOGICAL" +
  252. "|PRECOGNITIVE" +
  253. "|PROGRESSIVE" +
  254. "|RECURRING" +
  255. "|RELIGIOUS" +
  256. "|SIGNAL" +
  257. "|TEETH" +
  258. "|TEST" +
  259. "|MONEY"
  260. ;
  261. /*
  262. * Some parts of the file format have a slightly more complex stricture.
  263. * This section defines regular expressions for each of those.
  264. */
  265. // DateTimes are of the form "yyyy-MM-dd hh:mm+ZZZ" (with quotes)
  266. const datetime_type =
  267. "\"" +
  268. "([0-9][0-9]*)-([0-9][0-9]?)-([0-9][0-9]?) " +
  269. "([0-9][0-9]?):([0-9][0-9])?" +
  270. "([-+])([0-9][0-9]*)([0-9][0-9])" +
  271. "\""
  272. ;
  273. // any integer greater than zero:
  274. const positive_integer = "[1-9][0-9]*";
  275. // any integer in the range 0..10 (inclusive):
  276. const zero_to_ten = "[0-9]|10";
  277. // any integer in the range -5..5 (inclusive):
  278. const negative_five_to_five = "-[1-5]|[0-5]";
  279. // a quoted string:
  280. const free_text = "\"(.|\n)*\"";
  281. /*
  282. * Fields in a single record
  283. */
  284. const custom_field = custom;
  285. const sleep_aid_class_field = sleep_aid_class;
  286. const sleep_hindrance_class_field = sleep_hindrance_class;
  287. const name_field = free_text;
  288. const wake_field = datetime_type;
  289. const sleep_field = datetime_type;
  290. const bedtime_field = datetime_type;
  291. const holes_field = "|" +
  292. "(" + positive_integer + ")-(" + positive_integer + ")" +
  293. "(\\|(" + positive_integer + ")-(" + positive_integer + "))*"
  294. ;
  295. const type_field = sleep_type;
  296. const dream = "(" + dream_type + "):(" + negative_five_to_five + ")(:(" + dream_theme + "))*";
  297. const dreams_field = "NONE|" +
  298. "(" + dream + ")" +
  299. "(\\|(" + dream + "))*";
  300. const aids_field = "NONE|" +
  301. "(" + sleep_aid + ")" +
  302. "(\\|(" + sleep_aid + "))*";
  303. const hindrances_field = "NONE|" +
  304. "(" + sleep_hindrance + ")" +
  305. "(\\|(" + sleep_hindrance + "))*";
  306. const tags_field = "NONE|" +
  307. "(" + sleep_tag + ")" +
  308. "(\\|(" + sleep_tag + "))*";
  309. const quality_field = zero_to_ten;
  310. const notes_field = free_text;
  311. /*
  312. * Complete records
  313. */
  314. const custom_aid =
  315. "^(" + custom_field + ")," +
  316. "(" + sleep_aid_class_field + ")," +
  317. "(" + name_field + ")$"
  318. ;
  319. const custom_hindrance =
  320. "^(" + custom_field + ")," +
  321. "(" + sleep_hindrance_class_field + ")," +
  322. "(" + name_field + ")$"
  323. ;
  324. const custom_tag =
  325. "^(" + custom_field + ")," +
  326. "(" + name_field + ")$"
  327. ;
  328. const diary_entry =
  329. "^(" + wake_field + ")," +
  330. "(" + sleep_field + ")," +
  331. "(" + bedtime_field + ")," +
  332. "(" + holes_field + ")," +
  333. "(" + type_field + ")," +
  334. "(" + dreams_field + ")," +
  335. "(" + aids_field + ")," +
  336. "(" + hindrances_field + ")," +
  337. "(" + tags_field + ")," +
  338. "(" + quality_field + ")," +
  339. "(" + notes_field + ")$"
  340. ;
  341. /*
  342. * Section headers
  343. */
  344. const custom_aid_section_header = "custom_aid_id,class,name";
  345. const custom_hindrance_section_header = "custom_hindrance_id,class,name";
  346. const custom_tag_section_header = "custom_tag_id,name";
  347. const diary_section_header = "wake,sleep,bedtime,holes,type,dreams,aid,hindrances,tags,quality,notes";
  348. /*
  349. * Compiled regular expressions
  350. */
  351. const custom_re = new RegExp("^" + custom + "$");
  352. const custom_aid_re = new RegExp(custom_aid);
  353. const custom_hindrance_re = new RegExp(custom_hindrance);
  354. const custom_tag_re = new RegExp(custom_tag);
  355. const diary_entry_re = new RegExp(diary_entry);
  356. const datetime_type_re = new RegExp(datetime_type);
  357. /*
  358. * Parsers for individual data types
  359. */
  360. function parse_free_text(string) {
  361. return string.substr( 1, string.length-2 );
  362. }
  363. function parse_time(string) {
  364. return new Date( string.substr( 1, string.length-2 ) ).getTime();
  365. }
  366. function parse_datetime(string) {
  367. let data = string.match(datetime_type_re);
  368. return {
  369. "string" : string,
  370. "year" : parseInt(data[1],10),
  371. "month" : parseInt(data[2],10),
  372. "day" : parseInt(data[3],10),
  373. "hour" : parseInt(data[4],10),
  374. "minute" : parseInt(data[5],10),
  375. "offset" : (
  376. (data[6]=='-'?-1:1) * (
  377. parseInt(data[7],10) * 60 +
  378. parseInt(data[8],10)
  379. )
  380. ),
  381. };
  382. }
  383. function parse_timestamp(timestamp,timezone) {
  384. const date = DiaryBase.date(timestamp,timezone),
  385. offset = date.offset()
  386. ;
  387. return {
  388. "string": (
  389. '"' +
  390. date["year"]() +
  391. '-' +
  392. DiaryBase.zero_pad( date["month"]() ) +
  393. '-' +
  394. DiaryBase.zero_pad( date["day" ] () ) +
  395. ' ' +
  396. DiaryBase.zero_pad( date["hour"]() ) +
  397. ':' +
  398. DiaryBase.zero_pad( date["minute"]() ) +
  399. ( offset < 0 ? '-' : '+' ) +
  400. DiaryBase.zero_pad(Math.abs(Math.round(offset/60))) +
  401. DiaryBase.zero_pad(Math.abs( offset%60 )) +
  402. '"'
  403. ),
  404. "year" : date["year"](),
  405. "month" : date["month"](),
  406. "day" : date["day"](),
  407. "hour" : date["hour"](),
  408. "minute" : date["minute"](),
  409. "offset" : offset,
  410. };
  411. }
  412. function date_to_timestamp(date,offset) {
  413. const offset_hours = Math.floor(offset/60),
  414. offset_minutes = Math.floor(offset%60)
  415. ;
  416. date = new Date(
  417. ( date["getTime"] ? date["getTime"]() : date )
  418. + ((offset_hours*60)+offset_minutes)*60*1000
  419. );
  420. return {
  421. "string": (
  422. '"' +
  423. date["getUTCFullYear"]() +
  424. '-' +
  425. DiaryBase.zero_pad( date["getUTCMonth"]()+1 ) +
  426. '-' +
  427. DiaryBase.zero_pad( date["getUTCDate" ] () ) +
  428. ' ' +
  429. DiaryBase.zero_pad( date["getUTCHours"]() ) +
  430. ':' +
  431. DiaryBase.zero_pad( date["getUTCMinutes"]() ) +
  432. ( offset[0] == '-' ? '' : '+' ) +
  433. DiaryBase.zero_pad(Math.floor(offset/60),2) +
  434. DiaryBase.zero_pad(Math.floor(offset%60),2) +
  435. '"'
  436. ),
  437. "year" : date["getUTCFullYear"](),
  438. "month" : date["getUTCMonth" ]()+1,
  439. "day" : date["getUTCDate" ](),
  440. "hour" : date["getUTCHours" ](),
  441. "minute" : date["getUTCMinutes" ](),
  442. "offset" : offset,
  443. };
  444. }
  445. function parse_holes(string) {
  446. let ret = [];
  447. if ( string != "" ) {
  448. let sub_records = string.split('|');
  449. for ( let n=0; n!=sub_records.length; ++n ) {
  450. let sub_sub_records = sub_records[n].split('-');
  451. ret.push({
  452. "wake" : parseInt(sub_sub_records[0],10),
  453. "sleep": parseInt(sub_sub_records[1],10),
  454. });
  455. }
  456. }
  457. return ret;
  458. }
  459. function parse_dreams(string) {
  460. let ret = [];
  461. if ( string.length && string != "NONE" ) {
  462. let sub_records = string.split('|');
  463. for ( let n=0; n!=sub_records.length; ++n ) {
  464. let sub_sub_records = sub_records[n].split(':');
  465. ret.push({
  466. "type" : sub_sub_records[0],
  467. "mood" : parseInt(sub_sub_records[1],10),
  468. "themes": sub_sub_records.slice(2),
  469. });
  470. }
  471. }
  472. return ret;
  473. }
  474. function parse_list(string) {
  475. if ( !string.length || string == "NONE" ) {
  476. return [];
  477. } else {
  478. return string.split('|');
  479. }
  480. }
  481. function parse_quality(string) {
  482. return parseInt(string,10);
  483. }
  484. /*
  485. * Parsers for complete records
  486. */
  487. function parse_custom_aid( record ) {
  488. let data = record.match(custom_aid_re);
  489. return {
  490. "custom_aid_id": data[1] ,
  491. "class" : data[2] ,
  492. "name" : parse_free_text(data[3]),
  493. };
  494. }
  495. function parse_custom_hindrance( record ) {
  496. let data = record.match(custom_hindrance_re);
  497. return {
  498. "custom_hindrance_id": data[1] ,
  499. "class" : data[2] ,
  500. "name" : parse_free_text(data[3]),
  501. };
  502. }
  503. function parse_custom_tag( record ) {
  504. let data = record.match(custom_tag_re);
  505. return {
  506. "custom_tag_id": data[1] ,
  507. "name" : parse_free_text(data[2]),
  508. };
  509. }
  510. function parse_diary_entry( record ) {
  511. // parse records:
  512. let match = record.match(diary_entry_re),
  513. holes = parse_holes(match[28])
  514. ;
  515. return {
  516. "start" : parse_time( match[19] ),
  517. "end" : parse_time( match[ 1] ),
  518. "duration" : holes.reduce(
  519. (prev,hole) => prev + hole["sleep"] - hole["wake"],
  520. parse_time( match[ 1] ) -
  521. parse_time( match[10] )
  522. ),
  523. "wake" : parse_datetime (match[ 1]),
  524. "sleep" : parse_datetime (match[10]),
  525. "bedtime" : parse_datetime (match[19]),
  526. "holes" : holes,
  527. "type" : match[34] ,
  528. "dreams" : parse_dreams (match[35]),
  529. "aids" : parse_list (match[47]),
  530. "hindrances": parse_list (match[51]),
  531. "tags" : parse_list (match[55]),
  532. "quality" : parse_quality (match[59]),
  533. "notes" : parse_free_text (match[60]),
  534. };
  535. }
  536. /*
  537. * Split a document into sections, and the sections into records.
  538. *
  539. * We just parse the records to strings at this stage,
  540. * so it's easier to manage multi-line strings.
  541. */
  542. function parse_sections(text) {
  543. let expect_header = true,
  544. current_list,
  545. current_re,
  546. ret = {
  547. custom_aids: [],
  548. custom_hindrances: [],
  549. custom_tags: [],
  550. records: []
  551. }
  552. ;
  553. /*
  554. * we mainly process the document one line at a time,
  555. * but occasionally we need to merge lines together.
  556. */
  557. let lines = text.split("\n");
  558. for ( let n=0; n!=lines.length; ++n ) {
  559. let line = lines[n];
  560. if ( expect_header ) {
  561. /*
  562. * This line is probably a header,
  563. * but could just be a weird multi-line string
  564. */
  565. switch ( line ) {
  566. case custom_aid_section_header:
  567. current_list = ret.custom_aids;
  568. current_re = custom_aid_re;
  569. break;
  570. case custom_hindrance_section_header:
  571. current_list = ret.custom_hindrances;
  572. current_re = custom_hindrance_re;
  573. break;
  574. case custom_tag_section_header:
  575. current_list = ret.custom_tags;
  576. current_re = custom_tag_re;
  577. break;
  578. case diary_section_header:
  579. current_list = ret.records;
  580. current_re = diary_entry_re;
  581. break;
  582. default:
  583. if ( current_list ) {
  584. /*
  585. * the previous line was blank, but this is
  586. * not a header. We assume this must be part
  587. * of a strange multi-line string.
  588. */
  589. current_list[current_list.length-1] += "\n";
  590. --n;
  591. } else {
  592. return this.corrupt(file);
  593. }
  594. }
  595. expect_header = false;
  596. } else if ( lines[n] == "" ) {
  597. /*
  598. * This line looks like a section footer,
  599. * but could just be a weird multi-line string
  600. */
  601. expect_header = true;
  602. } else {
  603. /*
  604. * This line is (part of) a record
  605. */
  606. while ( line.substr(line.length-1) != '"' ) {
  607. /*
  608. * Lines that do not end in a quote must be part
  609. * of a multi-line string
  610. */
  611. if ( ++n == lines.length ) {
  612. return this.corrupt(file);
  613. }
  614. line += "\n" + lines[n];
  615. }
  616. if ( current_re.test(line) ) {
  617. // this is probably a complete record
  618. current_list.push(line);
  619. } else if ( current_list.length ) {
  620. // the previous line actually ended with a multi-line string:
  621. current_list[current_list.length-1] += "\n" + line;
  622. } else {
  623. return this.corrupt(file);
  624. }
  625. }
  626. }
  627. return ret;
  628. }
  629. /**
  630. * Spreadsheet manager
  631. * @protected
  632. * @type {Spreadsheet}
  633. */
  634. this["spreadsheet"] = new Spreadsheet(this,[
  635. {
  636. "sheet" : "Records",
  637. "member" : "records",
  638. "cells": [
  639. {
  640. "members": [ "start", "start_offset" ],
  641. "formats": [ "time", null ],
  642. "export": (array_element,row,offset) => {
  643. row[offset ] = Spreadsheet.create_cell( new Date( array_element["start"] ) );
  644. row[offset+1] = Spreadsheet.create_cell( array_element["bedtime"]["offset"] );
  645. return true;
  646. },
  647. "import": (array_element,row,offset) => {
  648. array_element["start"] = row[offset]["value"].getTime();
  649. return array_element["bedtime"] = date_to_timestamp(
  650. row[offset ]["value"],
  651. row[offset+1]["value"],
  652. )
  653. },
  654. },
  655. {
  656. "members": [ "end", "end_offset" ],
  657. "formats": [ "time", null ],
  658. "export": (array_element,row,offset) => {
  659. row[offset ] = Spreadsheet.create_cell( new Date( array_element["end"] ) );
  660. row[offset+1] = Spreadsheet.create_cell( array_element["wake"]["offset"] );
  661. return true;
  662. },
  663. "import": (array_element,row,offset) => {
  664. array_element["end"] = row[offset]["value"].getTime();
  665. return array_element["wake"] = date_to_timestamp(
  666. row[offset ]["value"],
  667. row[offset+1]["value"],
  668. );
  669. },
  670. },
  671. {
  672. "member": "duration",
  673. "type" : "duration",
  674. },
  675. {
  676. "members": [ "sleep", "sleep_offset" ],
  677. "formats": [ "time", null ],
  678. "export": (array_element,row,offset) => {
  679. row[offset ] = Spreadsheet.create_cell( new Date( array_element["sleep"]["string"].substr( 1, array_element["sleep"]["string"].length-2 ) ) );
  680. row[offset+1] = Spreadsheet.create_cell( array_element["sleep"]["offset"] );
  681. return true;
  682. },
  683. "import": (array_element,row,offset) =>
  684. array_element["sleep"] = date_to_timestamp(
  685. row[offset ]["value"],
  686. row[offset+1]["value"]
  687. ),
  688. },
  689. {
  690. "members": [ "holes" ],
  691. "regexp" : /^([0-9]*-[0-9]*(\|[0-9]*-[0-9]*)*)?$/,
  692. "export": (array_element,row,offset) => {
  693. row[offset] = Spreadsheet.create_cell( array_element["holes"].map( hole => hole["wake"]+'-'+hole["sleep"] ).join('|') );
  694. return true;
  695. },
  696. "import": (array_element,row,offset) => {
  697. array_element["holes"] = parse_holes(row[offset]["value"]);
  698. return true;
  699. },
  700. },
  701. {
  702. "member": "type",
  703. "regexp": /^(NIGHT_SLEEP|NAP)$/,
  704. },
  705. {
  706. "members": [ "dreams" ],
  707. "export": (array_element,row,offset) => {
  708. row[offset] = Spreadsheet.create_cell(
  709. array_element["dreams"]
  710. .map(
  711. dream => [ dream["type"], dream["mood"] ].concat(dream["themes"]).join(':')
  712. ).join('|')
  713. );
  714. return true;
  715. },
  716. "import": (array_element,row,offset) => array_element["dreams"] = parse_dreams(row[offset]["value"]),
  717. },
  718. {
  719. "members": [ "aids" ],
  720. "export": (array_element,row,offset) => {
  721. row[offset] = Spreadsheet.create_cell( array_element["aids"].join("|") )
  722. return true;
  723. },
  724. "import": (array_element,row,offset) => {
  725. array_element["aids"] = parse_list(row[offset]["value"]);
  726. return true;
  727. },
  728. },
  729. {
  730. "members": [ "hindrances" ],
  731. "export": (array_element,row,offset) => {
  732. row[offset] = Spreadsheet.create_cell( array_element["hindrances"].join("|") );
  733. return true;
  734. },
  735. "import": (array_element,row,offset) => {
  736. array_element["hindrances"] = parse_list(row[offset]["value"]);
  737. return true;
  738. },
  739. },
  740. {
  741. "members": [ "tags" ],
  742. "export": (array_element,row,offset) => {
  743. row[offset] = Spreadsheet.create_cell(array_element["tags"].join("|"))
  744. return true;
  745. },
  746. "import": (array_element,row,offset) => {
  747. array_element["tags"] = parse_list(row[offset]["value"]);
  748. return true;
  749. },
  750. },
  751. {
  752. "member": "quality",
  753. "regexp": /^[0-9]*$/,
  754. },
  755. {
  756. "member": "notes",
  757. },
  758. ]
  759. },
  760. {
  761. "sheet" : "Custom Aids",
  762. "member": "custom_aids",
  763. "cells": [
  764. {
  765. "member": "custom_aid_id",
  766. "regexp": custom_re,
  767. },
  768. {
  769. "member": "class",
  770. "regexp": new RegExp( "^(" + sleep_aid_class + ")$" ),
  771. },
  772. {
  773. "member": "name",
  774. },
  775. ],
  776. },
  777. {
  778. "sheet" : "Custom Hindrances",
  779. "member": "custom_hindrances",
  780. "cells": [
  781. {
  782. "member": "custom_hindrance_id",
  783. "regexp": custom_re,
  784. },
  785. {
  786. "member": "class",
  787. "regexp": new RegExp( "^(" + sleep_hindrance_class + ")$" ),
  788. },
  789. {
  790. "member": "name",
  791. },
  792. ],
  793. },
  794. {
  795. "sheet" : "Custom Tags",
  796. "member": "custom_tags",
  797. "cells": [
  798. {
  799. "member": "custom_tag_id",
  800. "regexp": custom_re,
  801. },
  802. {
  803. "member": "name",
  804. },
  805. ],
  806. },
  807. ]);
  808. let custom_aids = [],
  809. custom_hindrances = [],
  810. custom_tags = [],
  811. records = []
  812. ;
  813. switch ( file["file_format"]() ) {
  814. case "string":
  815. /*
  816. * Parse the complete document
  817. */
  818. if ( file["contents"].search(diary_section_header) == -1 ) return this.invalid(file);
  819. let sections = parse_sections(file["contents"]);
  820. custom_aids = sections.custom_aids .map( parse_custom_aid );
  821. custom_hindrances = sections.custom_hindrances.map( parse_custom_hindrance );
  822. custom_tags = sections.custom_tags .map( parse_custom_tag );
  823. records = sections.records .map( parse_diary_entry );
  824. break;
  825. default:
  826. if ( this.initialise_from_common_formats(file) ) return;
  827. let bedtimes = {};
  828. file["to"]("Standard")["records"].forEach(
  829. record => {
  830. switch ( record["status"] ) {
  831. case "in bed":
  832. bedtimes[record["end"]] = [ record["start"], record["start_timezone"] ];
  833. break;
  834. case "asleep":
  835. let tags = [ [], [], [] ],
  836. bedtime = bedtimes[record["start"]] ||
  837. [ record["start"], record["start_timezone"] ]
  838. ;
  839. (record["tags"]||[]).forEach(
  840. tag => {
  841. if ( sleep_tag_map.hasOwnProperty(tag) ) {
  842. tags[sleep_tag_map[tag]].push(tag);
  843. } else {
  844. let our_tag = custom_tags.find(
  845. custom_tag => tag == custom_tag["name"]
  846. );
  847. if ( our_tag ) {
  848. tags[2].push(our_tag["custom_tag_id"]);
  849. } else {
  850. const id = "CUSTOM_" + DiaryBase.zero_pad(custom_tags.length+1,4);
  851. custom_tags.push({
  852. "custom_tag_id": id,
  853. "name" : tag,
  854. });
  855. tags[2].push(id);
  856. }
  857. }
  858. }
  859. );
  860. records.push({
  861. "start" : bedtime[0],
  862. "end" : record["end" ],
  863. "duration" : record["duration"],
  864. "wake" : parse_timestamp(record["end" ],record["end_timezone"]),
  865. "sleep" : parse_timestamp(record["start"],record["start_timezone"]),
  866. "bedtime" : parse_timestamp( bedtime[0], bedtime[1] ),
  867. "holes" : [],
  868. "type" : record["is_primary_sleep"] ? "NIGHT_SLEEP" : "NAP",
  869. "dreams" : [],
  870. "aids" : tags[0],
  871. "hindrances" : tags[1],
  872. "tags" : tags[2],
  873. "quality" : 5,
  874. "notes" : (record["comments"]||[]).join('; ')
  875. });
  876. break;
  877. }
  878. }
  879. );
  880. break;
  881. }
  882. /**
  883. * Things that aid sleep
  884. */
  885. this["custom_aids" ] = custom_aids;
  886. /**
  887. * Things that hinder sleep
  888. */
  889. this["custom_hindrances"] = custom_hindrances;
  890. /**
  891. * Arbitrary tags describing the sleep experience
  892. */
  893. this["custom_tags" ] = custom_tags;
  894. /**
  895. * Individual records from the sleep diary
  896. */
  897. this["records" ] = records;
  898. }
  899. ["to"](to_format) {
  900. switch ( to_format ) {
  901. case "output":
  902. let contents = "";
  903. if ( this["custom_aids" ].length ) {
  904. contents += "custom_aid_id,class,name\n";
  905. this["custom_aids" ].forEach(
  906. aid => contents += `${aid["custom_aid_id"]},${aid["class"]},"${aid["name"]}"\n`
  907. );
  908. contents += "\n";
  909. }
  910. if ( this["custom_hindrances"].length ) {
  911. contents += "custom_hindrance_id,class,name\n";
  912. this["custom_hindrances"].forEach(
  913. hindrance => contents += `${hindrance["custom_hindrance_id"]},${hindrance["class"]},"${hindrance["name"]}"\n`
  914. );
  915. contents += "\n";
  916. }
  917. if ( this["custom_tags"].length ) {
  918. contents += "custom_tag_id,name\n";
  919. this["custom_tags"].forEach(
  920. tag => contents += `${tag["custom_tag_id"]},"${tag["name"]}"\n`
  921. );
  922. contents += "\n";
  923. }
  924. contents += "wake,sleep,bedtime,holes,type,dreams,aid,hindrances,tags,quality,notes\n";
  925. this["records"].forEach(
  926. rec => contents += [
  927. rec["wake" ]["string"],
  928. rec["sleep" ]["string"],
  929. rec["bedtime"]["string"],
  930. rec["holes"].map( hole => `${hole["wake"]}-${hole["sleep"]}` ).join('|'),
  931. rec["type"],
  932. rec["dreams"].map(
  933. dream => [ dream["type"], dream["mood"] ].concat(dream["themes"]).join(':')
  934. ).join('|') || 'NONE',
  935. rec["aids" ].join('|') || 'NONE',
  936. rec["hindrances"].join('|') || 'NONE',
  937. rec["tags" ].join('|') || 'NONE',
  938. rec["quality"],
  939. '"' + rec["notes"] + '"'
  940. ].join(',') + "\n"
  941. );
  942. return this.serialise({
  943. "file_format": () => "string",
  944. "contents": contents
  945. });
  946. case "Standard":
  947. let custom_aid_map = {},
  948. custom_hindrance_map = {},
  949. custom_tag_map = {},
  950. records = []
  951. ;
  952. this["custom_aids"].forEach(
  953. aid => custom_aid_map[aid["custom_aid_id"]] = aid["name"]
  954. );
  955. this["custom_hindrances"].forEach(
  956. hindrance => custom_hindrance_map[hindrance["custom_hindrance_id"]] = hindrance["name"]
  957. );
  958. this["custom_tags"].forEach(
  959. tag => custom_tag_map[tag["custom_tag_id"]] = tag["name"]
  960. );
  961. function parse_timezone( datetime ) {
  962. let offset = Math.round(datetime["offset"]/60);
  963. // Note: offsets and Etc/GMT times are reversed,
  964. // so negative offsets are GMT+ and positive offsets are GMT-
  965. if ( offset < 0 ) return "Etc/GMT+" + Math.abs(offset);
  966. if ( offset > 0 ) return "Etc/GMT-" + offset ;
  967. return "Etc/GMT";
  968. }
  969. this["records"].forEach(
  970. record => {
  971. var sleep = record["sleep"],
  972. sleep_time = new Date(sleep["string"].substr(1,sleep["string"].length-2)).getTime()
  973. ;
  974. if ( record["start"] < sleep_time ) {
  975. records.push({
  976. "status" : "in bed",
  977. "start" : record["start"],
  978. "end" : sleep_time,
  979. "start_timezone": parse_timezone(record["bedtime"]),
  980. "end_timezone": parse_timezone(record["sleep"]),
  981. });
  982. }
  983. let tags = [];
  984. record["aids" ].forEach( h => tags.push( custom_aid_map [h] || h ) );
  985. record["hindrances"].forEach( h => tags.push( custom_hindrance_map[h] || h ) );
  986. record["tags" ].forEach( h => tags.push( custom_tag_map [h] || h ) );
  987. records.push(Object.assign(
  988. {
  989. "status" : "asleep",
  990. "start" : (
  991. ( record["start"] === undefined )
  992. ? ( sleep_time || undefined )
  993. : Math.max( record["start"], sleep_time )
  994. ),
  995. "end" : record["end"],
  996. "start_timezone" : parse_timezone(record["sleep"]),
  997. "end_timezone" : parse_timezone(record["wake"]),
  998. "tags" : tags,
  999. "comments" : record["notes"].length ? [ record["notes"] ] : [],
  1000. },
  1001. ( record["duration"] === undefined ) ? {} : { "duration" : record["duration"] },
  1002. ( record["type"] == "NIGHT_SLEEP" ) ? { "is_primary_sleep": true } : {},
  1003. ));
  1004. }
  1005. );
  1006. return new DiaryStandard({ "records": records }, this.serialiser);
  1007. default:
  1008. return super["to"](to_format);
  1009. }
  1010. }
  1011. ["merge"](other) {
  1012. let custom_aid_map = {},
  1013. custom_hindrance_map = {},
  1014. custom_tag_map = {}
  1015. ;
  1016. other = other["to"](this["file_format"]());
  1017. // Map custom aid ids in the other file to equivalents in this file:
  1018. if ( this["custom_aids"].length && other["custom_aids"].length ) {
  1019. other["custom_aids"].forEach(
  1020. their_aid => {
  1021. let our_aid = this["custom_aids"].find(
  1022. our_aid => (
  1023. our_aid["class"] == their_aid["class"] &&
  1024. our_aid["name" ] == their_aid["name" ]
  1025. )
  1026. );
  1027. if ( !our_aid ) {
  1028. // create a new ID, e.g. "CUSTOM_0001":
  1029. let id = "CUSTOM_0001";
  1030. for (
  1031. let n=2;
  1032. this["custom_aids"].some( aid => aid["custom_aid_id"] == id );
  1033. ++n
  1034. ) {
  1035. id = "CUSTOM_" + DiaryBase.zero_pad(n,4);
  1036. }
  1037. our_aid = {
  1038. "custom_aid_id": id,
  1039. "class" : their_aid["class"],
  1040. "name" : their_aid["name" ],
  1041. };
  1042. this["custom_aids"].push(our_aid);
  1043. }
  1044. custom_aid_map[their_aid["custom_aid_id"]] = our_aid["custom_aid_id"];
  1045. }
  1046. );
  1047. } else {
  1048. other["custom_aids"].forEach(
  1049. their_aid => custom_aid_map[their_aid["custom_aid_id"]] = their_aid["custom_aid_id"]
  1050. );
  1051. }
  1052. // Map custom hindrance ids in the other file to equivalents in this file:
  1053. if ( this["custom_hindrances"].length && other["custom_hindrances"].length ) {
  1054. other["custom_hindrances"].forEach(
  1055. their_hindrance => {
  1056. let our_hindrance = this["custom_hindrances"].find(
  1057. our_hindrance => (
  1058. our_hindrance["class"] == their_hindrance["class"] &&
  1059. our_hindrance["name" ] == their_hindrance["name" ]
  1060. )
  1061. );
  1062. if ( !our_hindrance ) {
  1063. // create a new ID, e.g. "CUSTOM_0001":
  1064. let id = "CUSTOM_0001";
  1065. for (
  1066. let n=2;
  1067. this["custom_hindrances"].some( hindrance => hindrance["custom_hindrance_id"] == id );
  1068. ++n
  1069. ) {
  1070. id = "CUSTOM_" + DiaryBase.zero_pad(n,4);
  1071. }
  1072. our_hindrance = {
  1073. "custom_hindrance_id": id,
  1074. "class" : their_hindrance["class"],
  1075. "name" : their_hindrance["name" ],
  1076. };
  1077. this["custom_hindrances"].push(our_hindrance);
  1078. }
  1079. custom_hindrance_map[their_hindrance["custom_hindrance_id"]] = our_hindrance["custom_hindrance_id"];
  1080. }
  1081. );
  1082. } else {
  1083. other["custom_hindrances"].forEach(
  1084. their_hindrance => custom_hindrance_map[their_hindrance["custom_hindrance_id"]] = their_hindrance["custom_hindrance_id"]
  1085. );
  1086. }
  1087. // Map custom tag ids in the other file to equivalents in this file:
  1088. if ( this["custom_tags"].length && other["custom_tags"].length ) {
  1089. other["custom_tags"].forEach(
  1090. their_tag => {
  1091. let our_tag = this["custom_tags"].find(
  1092. our_tag => our_tag["name"] == their_tag["name"]
  1093. );
  1094. if ( !our_tag ) {
  1095. // create a new ID, e.g. "CUSTOM_0001":
  1096. let id = "CUSTOM_0001";
  1097. for (
  1098. let n=2;
  1099. this["custom_tags"].some( tag => tag["custom_tag_id"] == id );
  1100. ++n
  1101. ) {
  1102. id = "CUSTOM_" + DiaryBase.zero_pad(n,4);
  1103. }
  1104. our_tag = {
  1105. "custom_tag_id": id,
  1106. "name" : their_tag["name" ],
  1107. };
  1108. this["custom_tags"].push(our_tag);
  1109. }
  1110. custom_tag_map[their_tag["custom_tag_id"]] = our_tag["custom_tag_id"];
  1111. }
  1112. );
  1113. } else {
  1114. other["custom_tags"].forEach(
  1115. their_tag => custom_tag_map[their_tag["custom_tag_id"]] = their_tag["custom_tag_id"]
  1116. );
  1117. }
  1118. // merge records:
  1119. this["records"] = this["records"].concat(
  1120. DiaryBase.unique(
  1121. this["records"],
  1122. other["records"],
  1123. r => [ r["wake"]["string"], r["sleep"]["string"], r["bedtime"]["string"] ].join()
  1124. )
  1125. .map(
  1126. record => Object.assign(
  1127. {},
  1128. record,
  1129. {
  1130. "aids" : record["aids"].map( aid => custom_aid_map[aid] || aid ),
  1131. "hindrances": record["hindrances"].map( hindrance => custom_hindrance_map[hindrance] || hindrance ),
  1132. "tags" : record["tags"].map( tag => custom_tag_map[tag] || tag ),
  1133. }))
  1134. )
  1135. .sort( (a,b) => a["wake"] - b["wake"] )
  1136. ;
  1137. return this;
  1138. }
  1139. ["file_format"]() { return "Sleepmeter"; }
  1140. ["format_info"]() {
  1141. return {
  1142. "name": "Sleepmeter",
  1143. "title": "Sleepmeter",
  1144. "url": "/src/Sleepmeter",
  1145. "statuses": [ "in bed", "asleep" ],
  1146. "extension": ".csv",
  1147. "logo": "http://www.squalllinesoftware.com/sites/squalllinesoftware.com/files/sleepmeter_logo_128x128.png",
  1148. "timezone": "offset",
  1149. }
  1150. }
  1151. }
  1152. DiaryBase.register(DiarySleepmeter);