| #!/usr/bin/env python3 |
| # coding=utf-8 |
| # Copyright (c) 2020, Google Inc. |
| # |
| # Permission to use, copy, modify, and/or distribute this software for any |
| # purpose with or without fee is hereby granted, provided that the above |
| # copyright notice and this permission notice appear in all copies. |
| # |
| # THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES |
| # WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF |
| # MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY |
| # SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES |
| # WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION |
| # OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN |
| # CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. |
| |
| """This script translates JSON test vectors to BoringSSL's "FileTest" format. |
| |
| Usage: translate_test_vectors.py TEST_VECTORS_JSON_FILE |
| |
| The TEST_VECTORS_JSON_FILE is expected to come from the JSON copy of |
| RFC 9180's test vectors, linked from its [TestVectors] citation. |
| The output is written to "hpke_test_vectors.txt". |
| """ |
| |
| import collections |
| import json |
| import sys |
| |
| HPKE_MODE_BASE = 0 |
| HPKE_MODE_PSK = 1 |
| HPKE_MODE_AUTH = 2 |
| HPKE_DHKEM_P256_SHA256 = 0x0010 |
| HPKE_DHKEM_X25519_SHA256 = 0x0020 |
| HPKE_HKDF_SHA256 = 0x0001 |
| HPKE_AEAD_EXPORT_ONLY = 0xffff |
| |
| |
| def read_test_vectors_and_generate_code(json_file_in_path, test_file_out_path): |
| """Translates JSON test vectors into BoringSSL's FileTest language. |
| |
| Args: |
| json_file_in_path: Path to the JSON test vectors file. |
| test_file_out_path: Path to output file. |
| """ |
| |
| # Load the JSON file into |test_vecs|. |
| with open(json_file_in_path) as file_in: |
| test_vecs = json.load(file_in) |
| |
| lines = [] |
| for test in test_vecs: |
| # Filter out test cases that we don't use. |
| if (test["mode"] not in (HPKE_MODE_BASE, HPKE_MODE_AUTH) or |
| test["kem_id"] not in (HPKE_DHKEM_X25519_SHA256, |
| HPKE_DHKEM_P256_SHA256) or |
| test["aead_id"] == HPKE_AEAD_EXPORT_ONLY or |
| test["kdf_id"] != HPKE_HKDF_SHA256): |
| continue |
| |
| keys = ["mode", "kem_id", "kdf_id", "aead_id", "info", "skRm", "skEm", "pkRm", "pkEm", "ikmE", "ikmR"] |
| |
| if test["mode"] == HPKE_MODE_AUTH: |
| keys.append("pkSm") |
| keys.append("skSm") |
| |
| for key in keys: |
| lines.append("{} = {}".format(key, str(test[key]))) |
| |
| for i, enc in enumerate(test["encryptions"]): |
| lines.append("# encryptions[{}]".format(i)) |
| for key in ("aad", "ct", "pt"): |
| lines.append("{} = {}".format(key, str(enc[key]))) |
| |
| for i, exp in enumerate(test["exports"]): |
| lines.append("# exports[{}]".format(i)) |
| for key in ("exporter_context", "L", "exported_value"): |
| lines.append("{} = {}".format(key, str(exp[key]))) |
| |
| lines.append("") |
| |
| with open(test_file_out_path, "w") as file_out: |
| file_out.write("\n".join(lines)) |
| |
| |
| def main(argv): |
| if len(argv) != 2: |
| print(__doc__) |
| sys.exit(1) |
| |
| read_test_vectors_and_generate_code(argv[1], "hpke_test_vectors.txt") |
| |
| |
| if __name__ == "__main__": |
| main(sys.argv) |