|  | #!/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) |