diff --git a/LICENSE b/LICENSE new file mode 100644 index 0000000..261eeb9 --- /dev/null +++ b/LICENSE @@ -0,0 +1,201 @@ + Apache License + Version 2.0, January 2004 + http://www.apache.org/licenses/ + + TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + + 1. Definitions. + + "License" shall mean the terms and conditions for use, reproduction, + and distribution as defined by Sections 1 through 9 of this document. + + "Licensor" shall mean the copyright owner or entity authorized by + the copyright owner that is granting the License. + + "Legal Entity" shall mean the union of the acting entity and all + other entities that control, are controlled by, or are under common + control with that entity. For the purposes of this definition, + "control" means (i) the power, direct or indirect, to cause the + direction or management of such entity, whether by contract or + otherwise, or (ii) ownership of fifty percent (50%) or more of the + outstanding shares, or (iii) beneficial ownership of such entity. + + "You" (or "Your") shall mean an individual or Legal Entity + exercising permissions granted by this License. + + "Source" form shall mean the preferred form for making modifications, + including but not limited to software source code, documentation + source, and configuration files. + + "Object" form shall mean any form resulting from mechanical + transformation or translation of a Source form, including but + not limited to compiled object code, generated documentation, + and conversions to other media types. + + "Work" shall mean the work of authorship, whether in Source or + Object form, made available under the License, as indicated by a + copyright notice that is included in or attached to the work + (an example is provided in the Appendix below). + + "Derivative Works" shall mean any work, whether in Source or Object + form, that is based on (or derived from) the Work and for which the + editorial revisions, annotations, elaborations, or other modifications + represent, as a whole, an original work of authorship. For the purposes + of this License, Derivative Works shall not include works that remain + separable from, or merely link (or bind by name) to the interfaces of, + the Work and Derivative Works thereof. + + "Contribution" shall mean any work of authorship, including + the original version of the Work and any modifications or additions + to that Work or Derivative Works thereof, that is intentionally + submitted to Licensor for inclusion in the Work by the copyright owner + or by an individual or Legal Entity authorized to submit on behalf of + the copyright owner. For the purposes of this definition, "submitted" + means any form of electronic, verbal, or written communication sent + to the Licensor or its representatives, including but not limited to + communication on electronic mailing lists, source code control systems, + and issue tracking systems that are managed by, or on behalf of, the + Licensor for the purpose of discussing and improving the Work, but + excluding communication that is conspicuously marked or otherwise + designated in writing by the copyright owner as "Not a Contribution." + + "Contributor" shall mean Licensor and any individual or Legal Entity + on behalf of whom a Contribution has been received by Licensor and + subsequently incorporated within the Work. + + 2. Grant of Copyright License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + copyright license to reproduce, prepare Derivative Works of, + publicly display, publicly perform, sublicense, and distribute the + Work and such Derivative Works in Source or Object form. + + 3. Grant of Patent License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + (except as stated in this section) patent license to make, have made, + use, offer to sell, sell, import, and otherwise transfer the Work, + where such license applies only to those patent claims licensable + by such Contributor that are necessarily infringed by their + Contribution(s) alone or by combination of their Contribution(s) + with the Work to which such Contribution(s) was submitted. If You + institute patent litigation against any entity (including a + cross-claim or counterclaim in a lawsuit) alleging that the Work + or a Contribution incorporated within the Work constitutes direct + or contributory patent infringement, then any patent licenses + granted to You under this License for that Work shall terminate + as of the date such litigation is filed. + + 4. Redistribution. You may reproduce and distribute copies of the + Work or Derivative Works thereof in any medium, with or without + modifications, and in Source or Object form, provided that You + meet the following conditions: + + (a) You must give any other recipients of the Work or + Derivative Works a copy of this License; and + + (b) You must cause any modified files to carry prominent notices + stating that You changed the files; and + + (c) You must retain, in the Source form of any Derivative Works + that You distribute, all copyright, patent, trademark, and + attribution notices from the Source form of the Work, + excluding those notices that do not pertain to any part of + the Derivative Works; and + + (d) If the Work includes a "NOTICE" text file as part of its + distribution, then any Derivative Works that You distribute must + include a readable copy of the attribution notices contained + within such NOTICE file, excluding those notices that do not + pertain to any part of the Derivative Works, in at least one + of the following places: within a NOTICE text file distributed + as part of the Derivative Works; within the Source form or + documentation, if provided along with the Derivative Works; or, + within a display generated by the Derivative Works, if and + wherever such third-party notices normally appear. The contents + of the NOTICE file are for informational purposes only and + do not modify the License. You may add Your own attribution + notices within Derivative Works that You distribute, alongside + or as an addendum to the NOTICE text from the Work, provided + that such additional attribution notices cannot be construed + as modifying the License. + + You may add Your own copyright statement to Your modifications and + may provide additional or different license terms and conditions + for use, reproduction, or distribution of Your modifications, or + for any such Derivative Works as a whole, provided Your use, + reproduction, and distribution of the Work otherwise complies with + the conditions stated in this License. + + 5. Submission of Contributions. Unless You explicitly state otherwise, + any Contribution intentionally submitted for inclusion in the Work + by You to the Licensor shall be under the terms and conditions of + this License, without any additional terms or conditions. + Notwithstanding the above, nothing herein shall supersede or modify + the terms of any separate license agreement you may have executed + with Licensor regarding such Contributions. + + 6. Trademarks. This License does not grant permission to use the trade + names, trademarks, service marks, or product names of the Licensor, + except as required for reasonable and customary use in describing the + origin of the Work and reproducing the content of the NOTICE file. + + 7. Disclaimer of Warranty. Unless required by applicable law or + agreed to in writing, Licensor provides the Work (and each + Contributor provides its Contributions) on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or + implied, including, without limitation, any warranties or conditions + of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A + PARTICULAR PURPOSE. You are solely responsible for determining the + appropriateness of using or redistributing the Work and assume any + risks associated with Your exercise of permissions under this License. + + 8. Limitation of Liability. In no event and under no legal theory, + whether in tort (including negligence), contract, or otherwise, + unless required by applicable law (such as deliberate and grossly + negligent acts) or agreed to in writing, shall any Contributor be + liable to You for damages, including any direct, indirect, special, + incidental, or consequential damages of any character arising as a + result of this License or out of the use or inability to use the + Work (including but not limited to damages for loss of goodwill, + work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses), even if such Contributor + has been advised of the possibility of such damages. + + 9. Accepting Warranty or Additional Liability. While redistributing + the Work or Derivative Works thereof, You may choose to offer, + and charge a fee for, acceptance of support, warranty, indemnity, + or other liability obligations and/or rights consistent with this + License. However, in accepting such obligations, You may act only + on Your own behalf and on Your sole responsibility, not on behalf + of any other Contributor, and only if You agree to indemnify, + defend, and hold each Contributor harmless for any liability + incurred by, or claims asserted against, such Contributor by reason + of your accepting any such warranty or additional liability. + + END OF TERMS AND CONDITIONS + + APPENDIX: How to apply the Apache License to your work. + + To apply the Apache License to your work, attach the following + boilerplate notice, with the fields enclosed by brackets "[]" + replaced with your own identifying information. (Don't include + the brackets!) The text should be enclosed in the appropriate + comment syntax for the file format. We also recommend that a + file or class name and description of purpose be included on the + same "printed page" as the copyright notice for easier + identification within third-party archives. + + Copyright [yyyy] [name of copyright owner] + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. diff --git a/README.md b/README.md index 928db2e..f2d9a63 100644 --- a/README.md +++ b/README.md @@ -1 +1,49 @@ -# mycroft-health \ No newline at end of file +Working on progress... + +* Blood Pressure +> track health blood pressure +Top? +> 110 +Bottom? +> 70 +For whom? +> me +Confirm? +> yes + +* Diabetes +> track health suger level 7.6 +Before or after meal? +> after +For whom? +> mother +Confirm? +> yes + +* Body Temperature +> track health temperature 99 +For whom? +> wife +Confirm? +> yes + +* Body Pain +> track health pain headache +For whom? +> me +Confirm? +> yes + +* Heartbeat +> track health heartbeat 75 +For whom? +> me +Confirm? +> yes + +* Generate report +> generate report for this month +Which catagory? +> pain +For whom? +> me diff --git a/__init__.py b/__init__.py new file mode 100644 index 0000000..a3e7b48 --- /dev/null +++ b/__init__.py @@ -0,0 +1,386 @@ +# Copyright 2019 S. M. Estiaque Ahmed +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +import os +import sqlite3 +from datetime import timedelta +from json2html import json2html + +# For test. +from sendgrid import SendGridAPIClient +from sendgrid.helpers.mail import Mail + +from adapt.intent import IntentBuilder +from mycroft.util.time import now_local +from mycroft.skills.core import MycroftSkill +from mycroft.skills.core import intent_handler +from mycroft.util.format import pronounce_number + + +class HealthSkill(MycroftSkill): + def __init__(self): + super(HealthSkill, self).__init__(name="HealthSkill") + self.log.error(self.root_dir) + self.home_dir = os.path.expanduser('~') + self.log.error(self.home_dir) + + @intent_handler(IntentBuilder("PressureIntent").require("Track") + .require("Health").require("Pressure")) + def handle_pressure_intent(self, message): + top = self.get_response("top.mh") + + if not top: + self.speak_dialog("error.input.mh") + return + + try: + top = float(top) + + except Exception: + self.speak_dialog("error.input.mh") + return + + bottom = self.get_response("bottom.mh") + + if not bottom: + self.speak_dialog("error.input.mh") + return + + try: + bottom = float(bottom) + + except Exception: + self.speak_dialog("error.input.mh") + return + + person = self.get_response("person.mh") + person = "" if not person else person + + self.log.error(top) + self.log.error(bottom) + self.log.error(person) + + pressure_data = [(now_local(), "pressure", str(top), "", person), + (now_local(), "pressure", str(bottom), "", person)] + + confirm = self.ask_yesno( + "confirm.pressure.mh", + {"top": pronounce_number(top), + "bottom": pronounce_number(bottom)}) + + if confirm == "yes": + if not self.insert_db(pressure_data): + self.speak_dialog("error.save.mh") + + @intent_handler(IntentBuilder("SugarIntent").require("Track") + .require("Health").require("Sugar").require("Value")) + def handle_sugar_intent(self, message): + value = message.data.get("Value") + + if not value: + self.speak_dialog("error.input.mh") + return + + try: + value = float(value) + + except Exception: + self.speak_dialog("error.input.mh") + return + + meal_status = self.get_response("meal.mh") + + if not meal_status: + return + + person = self.get_response("person.mh") + person = "" if not person else person + + self.log.error(value) + self.log.error(meal_status) + self.log.error(person) + + sugar_data = [(now_local(), "diabetes", str(value), + meal_status, person)] + + confirm = self.ask_yesno( + "confirm.sugar.mh", + {"meal": meal_status, + "value": pronounce_number(value)}) + + if confirm == "yes": + if not self.insert_db(sugar_data): + self.speak_dialog("error.save.mh") + + @intent_handler(IntentBuilder("TemperatureIntent").require("Track") + .require("Health").require("Temperature").require("Value")) + def handle_temperature_intent(self, message): + value = message.data.get("Value") + + if not value: + self.speak_dialog("error.input.mh") + return + + try: + value = float(value) + + except Exception: + self.speak_dialog("error.input.mh") + return + + person = self.get_response("person.mh") + person = "" if not person else person + + self.log.error(value) + self.log.error(person) + + temperature_data = [(now_local(), "temperature", + str(value), "", person)] + + confirm = self.ask_yesno( + "confirm.temperature.mh", + {"value": pronounce_number(value)}) + + if confirm == "yes": + if not self.insert_db(temperature_data): + self.speak_dialog("error.save.mh") + + @intent_handler(IntentBuilder("PainIntent").require("Track") + .require("Health").require("Pain").require("Value")) + def handle_pain_intent(self, message): + value = message.data.get("Value") + + if not value: + self.speak_dialog("error.input.mh") + return + + person = self.get_response("person.mh") + person = "" if not person else person + + self.log.error(value) + self.log.error(person) + + pain_data = [(now_local(), "pain", value, "", person)] + + confirm = self.ask_yesno( + "confirm.pain.mh", + {"value": value}) + + if confirm == "yes": + if not self.insert_db(pain_data): + self.speak_dialog("error.save.mh") + + @intent_handler(IntentBuilder("HeartbeatIntent").require("Track") + .require("Health").require("Heartbeat").require("Value")) + def handle_heartbeat_intent(self, message): + value = message.data.get("Value") + + if not value: + self.speak_dialog("error.input.mh") + return + + try: + value = int(value) + + except Exception: + self.speak_dialog("error.input.mh") + return + + person = self.get_response("person.mh") + person = "" if not person else person + + self.log.error(value) + self.log.error(person) + + heartbeat_data = [(now_local(), "heartbeat", str(value), + "", person)] + + confirm = self.ask_yesno( + "confirm.heartbeat.mh", + {"value": pronounce_number(value)}) + + if confirm == "yes": + if not self.insert_db(heartbeat_data): + self.speak_dialog("error.save.mh") + + @intent_handler(IntentBuilder("GenerateIntent").require("Generate") + .require("Health").require("Report").require("Month")) + # @intent_handler(IntentBuilder("GenerateIntent").require("Generate") + # .require("Health").require("Report")) + def handle_report_intent(self, message): + # month = message.data.get("Month") + # self.log.error(month) + + # For test. + month = "this" + + if not month: + self.speak_dialog("error.input.mh") + return + + catagory = self.get_response("catagory.mh") + + if not catagory or catagory not in ["pressure", "diabetes", "pain", + "temperature", "heartbeat"]: + self.speak_dialog("error.input.mh") + return + + person = self.get_response("person.mh") + person = "" if not person else person + + current_datetime = now_local() + + # Multi languages support should be implemented here. + if month == "this" or month == "current": + to_datetime = current_datetime + + elif month == "last" or month == "previous": + to_datetime = current_datetime.replace( + day=1, + hour=23, + minute=59, + second=59, + microsecond=0) - timedelta(days=1) + + from_datetime = to_datetime.replace( + day=1, + hour=0, + minute=0, + second=0, + microsecond=0) + + health_data_list = self.get_data(from_datetime, to_datetime, + catagory, person) + + health_data = [] + + for data in health_data_list: + health_data.append({ + 'Datetime': data[0].strftime("%m/%d/%Y %H:%M:%S"), + 'Catagory': data[1], + 'Value': data[2], + 'Parameter': data[3], + 'Person': data[4]}) + + table = json2html.convert(json=health_data) + + content = """ + """ + + content = """ + {0}

Health Data: {1} to {2}

+ {3}""".format( + content, + from_datetime.strftime("%m/%d/%Y %H:%M:%S"), + to_datetime.strftime("%m/%d/%Y %H:%M:%S"), + table) + + # self.send_email( + # "Health Report - {0}".format(catagory.upper()), + # content) + + # For test. + message = Mail( + from_email="Mycroft Health ", + to_emails="smearumi@gmail.com", + subject="Mycroft Health Report - {0}".format(catagory.upper()), + html_content=content) + + api_key = "SG.yJSmPOSlR8CDxWrEiHtkiw.08p0jt2Isa_F9we0" + api_key = "{0}-H22n2_prFhVBLuBxVcSOSiV4EM".format(api_key) + + try: + sg = SendGridAPIClient(api_key) + _ = sg.send(message) + + except Exception: + pass + + def db_connect(self): + self.log.error(self.root_dir) + + try: + connection = sqlite3.connect( + "{0}/mycroft-health-db.db".format( + self.home_dir), + detect_types=sqlite3.PARSE_DECLTYPES | + sqlite3.PARSE_COLNAMES) + + except Exception: + return None + + try: + cursor = connection.cursor() + + sql = """CREATE TABLE health_data (datetime TIMESTAMP NOT NULL, + catagory TEXT NOT NULL, value TEXT NOT NULL, + parameter TEXT, person TEXT)""" + + cursor.execute(sql) + connection.commit() + # connection.close() + + except Exception: + pass + + return connection + + def insert_db(self, data): + connection = self.db_connect() + + if not connection: + return + + sql = """INSERT INTO health_data VALUES (?, ?, ?, ?, ?)""" + + with connection: + try: + cursor = connection.cursor() + cursor.executemany(sql, data) + connection.commit() + + except Exception: + connection.rollback() + return False + + return True + + def get_data(self, from_datetime, to_datetime, catagory, person): + connection = self.db_connect() + + if not connection: + return + + sql = "SELECT * FROM health_data WHERE datetime > ? and datetime < ?" + sql = "{0} and catagory = ? and person = ?".format(sql) + data_tuple = (from_datetime, to_datetime, catagory, person) + + with connection: + try: + cursor = connection.cursor() + cursor.execute(sql, data_tuple) + return cursor.fetchall() + + except Exception: + return None + + def stop(self): + pass + + +def create_skill(): + return HealthSkill() diff --git a/dialog/en-us/bottom.mh.dialog b/dialog/en-us/bottom.mh.dialog new file mode 100644 index 0000000..fce83b7 --- /dev/null +++ b/dialog/en-us/bottom.mh.dialog @@ -0,0 +1,2 @@ +please provide the bottom value. +bottom value? diff --git a/dialog/en-us/catagory.mh.dialog b/dialog/en-us/catagory.mh.dialog new file mode 100644 index 0000000..2ca4a46 --- /dev/null +++ b/dialog/en-us/catagory.mh.dialog @@ -0,0 +1,2 @@ +Please select a catagory: pressure, temperature, diabetes, pain, heartbeat. +which catagory do you want to select? pressure, temperature, diabetes, pain, heartbeat? diff --git a/dialog/en-us/confirm.heartbeat.mh.dialog b/dialog/en-us/confirm.heartbeat.mh.dialog new file mode 100644 index 0000000..42930d1 --- /dev/null +++ b/dialog/en-us/confirm.heartbeat.mh.dialog @@ -0,0 +1,2 @@ +Heartbeat is: {{value}}. Say yes to confirm. +Heart rate is: {{value}}. Do you wish to say yes to confirm? diff --git a/dialog/en-us/confirm.pain.mh.dialog b/dialog/en-us/confirm.pain.mh.dialog new file mode 100644 index 0000000..5c208bf --- /dev/null +++ b/dialog/en-us/confirm.pain.mh.dialog @@ -0,0 +1,2 @@ +Pain in: {{value}}. Say yes to confirm. +Feeling pain in: {{value}}. Do you wish to say yes to confirm? diff --git a/dialog/en-us/confirm.pressure.mh.dialog b/dialog/en-us/confirm.pressure.mh.dialog new file mode 100644 index 0000000..5242162 --- /dev/null +++ b/dialog/en-us/confirm.pressure.mh.dialog @@ -0,0 +1 @@ +Pressure readings, top: {{top}} and bottom: {{bottom}}. Do you wish to say yes to confirm? diff --git a/dialog/en-us/confirm.sugar.mh.dialog b/dialog/en-us/confirm.sugar.mh.dialog new file mode 100644 index 0000000..99511a7 --- /dev/null +++ b/dialog/en-us/confirm.sugar.mh.dialog @@ -0,0 +1 @@ +Diabetes reading {{meal}}: {{value}}. Do you wish to say yes to confirm? diff --git a/dialog/en-us/confirm.temperature.mh.dialog b/dialog/en-us/confirm.temperature.mh.dialog new file mode 100644 index 0000000..60bf942 --- /dev/null +++ b/dialog/en-us/confirm.temperature.mh.dialog @@ -0,0 +1 @@ +Temperature reading: {{value}}. Do you wish to say yes to confirm? diff --git a/dialog/en-us/error.input.mh.dialog b/dialog/en-us/error.input.mh.dialog new file mode 100644 index 0000000..d10e4a8 --- /dev/null +++ b/dialog/en-us/error.input.mh.dialog @@ -0,0 +1,2 @@ +Error: Input error. +Input error occurs. diff --git a/dialog/en-us/error.save.mh.dialog b/dialog/en-us/error.save.mh.dialog new file mode 100644 index 0000000..a85a751 --- /dev/null +++ b/dialog/en-us/error.save.mh.dialog @@ -0,0 +1,2 @@ +Error: I can't save data. +Error occurs during saving data. diff --git a/dialog/en-us/meal.mh b/dialog/en-us/meal.mh new file mode 100644 index 0000000..48f409e --- /dev/null +++ b/dialog/en-us/meal.mh @@ -0,0 +1,2 @@ +Before meal or after meal? +Before or after meal? diff --git a/dialog/en-us/person.mh.dialog b/dialog/en-us/person.mh.dialog new file mode 100644 index 0000000..e9f2d33 --- /dev/null +++ b/dialog/en-us/person.mh.dialog @@ -0,0 +1,2 @@ +For whom? +To whom is these readings addressed? diff --git a/dialog/en-us/top.mh.dialog b/dialog/en-us/top.mh.dialog new file mode 100644 index 0000000..4dead75 --- /dev/null +++ b/dialog/en-us/top.mh.dialog @@ -0,0 +1,2 @@ +please provide the top value. +top value? diff --git a/regex/en-us/Month.rx b/regex/en-us/Month.rx new file mode 100644 index 0000000..83ec252 --- /dev/null +++ b/regex/en-us/Month.rx @@ -0,0 +1 @@ +for (?Pthis|current|last|previous) month \ No newline at end of file diff --git a/regex/en-us/Value.rx b/regex/en-us/Value.rx new file mode 100644 index 0000000..a3fe028 --- /dev/null +++ b/regex/en-us/Value.rx @@ -0,0 +1,3 @@ +(heartbeat|rate|sugar level|temperature|pain) (?P.*) +(heart rate|sugar|temperature|pain) (?P.*) +(heart beat|sugar|temperature|pain) (?P.*) diff --git a/requirements.txt b/requirements.txt new file mode 100755 index 0000000..960409b --- /dev/null +++ b/requirements.txt @@ -0,0 +1,2 @@ +json2html==1.3.0 +sendgrid diff --git a/test/intent/define.json b/test/intent/define.json new file mode 100644 index 0000000..a1c0d2a --- /dev/null +++ b/test/intent/define.json @@ -0,0 +1,9 @@ +{ + "utterance": "define cat", + "intent_type": "DefineIntent", + "intent": { + "Define": "define", + "Word": "cat" + }, + "expected_dialog": "definition" +} diff --git a/vocab/en-us/Generate.voc b/vocab/en-us/Generate.voc new file mode 100644 index 0000000..05c95c2 --- /dev/null +++ b/vocab/en-us/Generate.voc @@ -0,0 +1,2 @@ +generate +send diff --git a/vocab/en-us/Health.voc b/vocab/en-us/Health.voc new file mode 100644 index 0000000..dc52f14 --- /dev/null +++ b/vocab/en-us/Health.voc @@ -0,0 +1,2 @@ +health +body diff --git a/vocab/en-us/Heartbeat.voc b/vocab/en-us/Heartbeat.voc new file mode 100644 index 0000000..736f3e7 --- /dev/null +++ b/vocab/en-us/Heartbeat.voc @@ -0,0 +1,2 @@ +heartbeat +heart rate diff --git a/vocab/en-us/Pain.voc b/vocab/en-us/Pain.voc new file mode 100644 index 0000000..68e5517 --- /dev/null +++ b/vocab/en-us/Pain.voc @@ -0,0 +1 @@ +pain diff --git a/vocab/en-us/Pressure.voc b/vocab/en-us/Pressure.voc new file mode 100644 index 0000000..bea46ca --- /dev/null +++ b/vocab/en-us/Pressure.voc @@ -0,0 +1 @@ +pressure diff --git a/vocab/en-us/Report.voc b/vocab/en-us/Report.voc new file mode 100644 index 0000000..05c3192 --- /dev/null +++ b/vocab/en-us/Report.voc @@ -0,0 +1 @@ +report diff --git a/vocab/en-us/Sugar.voc b/vocab/en-us/Sugar.voc new file mode 100644 index 0000000..ad031aa --- /dev/null +++ b/vocab/en-us/Sugar.voc @@ -0,0 +1,2 @@ +sugar +diabetes diff --git a/vocab/en-us/Temperature.voc b/vocab/en-us/Temperature.voc new file mode 100644 index 0000000..be03bca --- /dev/null +++ b/vocab/en-us/Temperature.voc @@ -0,0 +1,2 @@ +temperature +fever diff --git a/vocab/en-us/Track.voc b/vocab/en-us/Track.voc new file mode 100644 index 0000000..fedc6d3 --- /dev/null +++ b/vocab/en-us/Track.voc @@ -0,0 +1,3 @@ +track +set +record