Skip to content

Commit

Permalink
Add pyxform version to XForm output, closes #393
Browse files Browse the repository at this point in the history
  • Loading branch information
MartijnR authored and yanokwa committed Dec 20, 2019
1 parent 75721b0 commit f41227c
Show file tree
Hide file tree
Showing 11 changed files with 55 additions and 7 deletions.
1 change: 1 addition & 0 deletions pyxform/constants.py
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,7 @@
COMPACT_TAG = "compact_tag"

VERSION = "version"
PYXFORM_VERSION = "pyxform_version"
PUBLIC_KEY = "public_key"
SUBMISSION_URL = "submission_url"
AUTO_SEND = "auto_send"
Expand Down
4 changes: 4 additions & 0 deletions pyxform/survey.py
Original file line number Diff line number Diff line change
Expand Up @@ -130,6 +130,7 @@ class Survey(Section):
"public_key": unicode,
"instance_xmlns": unicode,
"version": unicode,
"pyxform_version": unicode,
"choices": dict,
"style": unicode,
"attribute": dict,
Expand Down Expand Up @@ -475,6 +476,9 @@ def xml_instance(self, **kwargs):
if self.version:
result.setAttribute("version", self.version)

if self.pyxform_version:
result.setAttribute("odk:pyxform-version", self.pyxform_version)

if self.prefix:
result.setAttribute("odk:prefix", self.prefix)

Expand Down
20 changes: 15 additions & 5 deletions pyxform/tests/builder_tests.py
Original file line number Diff line number Diff line change
Expand Up @@ -159,8 +159,10 @@ def test_specify_other(self):
},
],
}
actual_dict = survey.to_json_dict()
actual_dict.pop("pyxform_version", None)
self.maxDiff = None
self.assertEqual(survey.to_json_dict(), expected_dict)
self.assertEqual(actual_dict, expected_dict)

def test_select_one_question_with_identical_choice_name(self):
"""
Expand Down Expand Up @@ -199,8 +201,10 @@ def test_select_one_question_with_identical_choice_name(self):
},
],
}
actual_dict = survey.to_json_dict()
actual_dict.pop("pyxform_version", None)
self.maxDiff = None
self.assertEqual(survey.to_json_dict(), expected_dict)
self.assertEqual(actual_dict, expected_dict)

def test_loop(self):
survey = utils.create_survey_from_fixture("loop", filetype=FIXTURE_FILETYPE)
Expand Down Expand Up @@ -312,8 +316,10 @@ def test_loop(self):
},
],
}
actual_dict = survey.to_json_dict()
actual_dict.pop("pyxform_version", None)
self.maxDiff = None
self.assertEqual(survey.to_json_dict(), expected_dict)
self.assertEqual(actual_dict, expected_dict)

def test_sms_columns(self):
survey = utils.create_survey_from_fixture("sms_info", filetype=FIXTURE_FILETYPE)
Expand Down Expand Up @@ -449,7 +455,9 @@ def test_sms_columns(self):
"title": "SMS Example",
"type": "survey",
}
self.assertEqual(survey.to_json_dict(), expected_dict)
actual_dict = survey.to_json_dict()
actual_dict.pop("pyxform_version", None)
self.assertEqual(actual_dict, expected_dict)

def test_style_column(self):
survey = utils.create_survey_from_fixture(
Expand Down Expand Up @@ -488,7 +496,9 @@ def test_style_column(self):
"title": "My Survey",
"type": "survey",
}
self.assertEqual(survey.to_json_dict(), expected_dict)
actual_dict = survey.to_json_dict()
actual_dict.pop("pyxform_version", None)
self.assertEqual(actual_dict, expected_dict)

STRIP_NS_FROM_TAG_RE = re.compile(r"\{.+\}")

Expand Down
1 change: 1 addition & 0 deletions pyxform/tests/group_test.py
Original file line number Diff line number Diff line change
Expand Up @@ -57,6 +57,7 @@ def test_json(self):
},
],
}
x_results.pop("pyxform_version", None)
self.maxDiff = None
self.assertEqual(x_results, expected_dict)

Expand Down
4 changes: 3 additions & 1 deletion pyxform/tests/loop_tests.py
Original file line number Diff line number Diff line change
Expand Up @@ -89,4 +89,6 @@ def test_loop(self):
},
],
}
self.assertEquals(survey.to_json_dict(), expected_dict)
actual_dict = survey.to_json_dict()
actual_dict.pop("pyxform_version", None)
self.assertEquals(actual_dict, expected_dict)
4 changes: 3 additions & 1 deletion pyxform/tests/settings_test.py
Original file line number Diff line number Diff line change
Expand Up @@ -55,7 +55,9 @@ def test_survey_reader(self):
},
],
}
self.assertEqual(survey_reader.to_json_dict(), expected_dict)
actual_dict = survey_reader.to_json_dict()
actual_dict.pop("pyxform_version", None)
self.assertEqual(actual_dict, expected_dict)

def test_settings(self):
survey = create_survey_from_path(self.path)
Expand Down
1 change: 1 addition & 0 deletions pyxform/tests/tests_by_file.py
Original file line number Diff line number Diff line change
Expand Up @@ -37,6 +37,7 @@ def runTest(self):
path_to_excel_file, default_name=root_filename
)
survey = pyxform.create_survey_element_from_dict(json_survey)
survey.pyxform_version = ""
survey.print_xform_to_file(path_to_output_xform)

# Compare with the expected output:
Expand Down
13 changes: 13 additions & 0 deletions pyxform/tests/utils.py
Original file line number Diff line number Diff line change
Expand Up @@ -64,6 +64,10 @@ def assertXFormEqual(self, xform1, xform2):
xform1 = ETree.fromstring(xform1.encode("utf-8"))
xform2 = ETree.fromstring(xform2.encode("utf-8"))

# Remove the odk:pyxform-version attribute from the primary instance child
self.remove_pyxform_version_attribute(xform1)
self.remove_pyxform_version_attribute(xform2)

# Sort tags under <model> section in each form
self.sort_model(xform1)
self.sort_model(xform2)
Expand Down Expand Up @@ -93,6 +97,15 @@ def elem_get_tag(elem):
key = elem_get_tag
elems[:] = sorted(elems, key=key)

def remove_pyxform_version_attribute(self, xform):
xforms_ns = "{http://www.w3.org/2002/xforms}"
odk_ns = "{http://www.opendatakit.org/xforms}"
primary_instance = xform.find(".//" + xforms_ns + "instance")

# Remove the pyxform-version attribute
if primary_instance is not None:
primary_instance[0].attrib.pop(odk_ns + "pyxform-version", None)

def sort_model(self, xform):
ns = "{http://www.w3.org/2002/xforms}"
model = xform.find(".//" + ns + "model")
Expand Down
2 changes: 2 additions & 0 deletions pyxform/tests/xls2json_tests.py
Original file line number Diff line number Diff line change
Expand Up @@ -43,6 +43,7 @@ def test_simple_yes_or_no_question(self):
with codecs.open(output_path, "rb", encoding="utf-8") as actual_file:
expected_json = json.load(expected_file)
actual_json = json.load(actual_file)
actual_json.pop("pyxform_version", None)
self.assertEqual(expected_json, actual_json)

def test_hidden(self):
Expand Down Expand Up @@ -140,6 +141,7 @@ def test_table(self):
with codecs.open(output_path, "rb", encoding="utf-8") as actual_file:
expected_json = json.load(expected_file)
actual_json = json.load(actual_file)
actual_json.pop("pyxform_version", None)
self.assertEqual(expected_json, actual_json)

def test_choice_filter_choice_fields(self):
Expand Down
4 changes: 4 additions & 0 deletions pyxform/tests_v1/pyxform_test_case.py
Original file line number Diff line number Diff line change
Expand Up @@ -174,6 +174,10 @@ def assertPyxformXform(self, **kwargs):
else:
survey = kwargs.get("survey")

# Remove the pyxform-version attribute
if survey:
survey.pyxform_version = ""

xml = survey._to_pretty_xml()
root = ETree.fromstring(xml.encode("utf-8"))

Expand Down
8 changes: 8 additions & 0 deletions pyxform/xls2json.py
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@
import os
import re
import sys
import subprocess
from collections import Counter

from pyxform import aliases, constants
Expand Down Expand Up @@ -406,6 +407,7 @@ def workbook_to_json(
# problems for formhub.
# constants.VERSION : datetime.datetime.now().strftime("%Y%m%d%H"),
constants.CHILDREN: [],
constants.PYXFORM_VERSION: get_git_describe_tags(),
}
# Here the default settings are overridden by those in the settings sheet
json_dict.update(settings)
Expand Down Expand Up @@ -1436,6 +1438,12 @@ def get_parameters(raw_parameters, allowed_parameters):
return params


def get_git_describe_tags():
return subprocess.check_output(
["git", "describe", "--tags", "HEAD"], universal_newlines=True
).strip()[1:]


class SpreadsheetReader(object):
def __init__(self, path_or_file):
path = path_or_file
Expand Down

0 comments on commit f41227c

Please sign in to comment.