#!/usr/bin/env python import moto import os from botocore import xform_name from botocore.session import Session import boto3 script_dir = os.path.dirname(os.path.abspath(__file__)) alternative_service_names = {"lambda": "awslambda", "dynamodb": "dynamodb2", "rds": "rds2"} def get_moto_implementation(service_name): service_name = ( service_name.replace("-", "") if "-" in service_name else service_name ) alt_service_name = ( alternative_service_names[service_name] if service_name in alternative_service_names else service_name ) mock = None mock_name = None if hasattr(moto, "mock_{}".format(alt_service_name)): mock_name = "mock_{}".format(alt_service_name) mock = getattr(moto, mock_name) elif hasattr(moto, "mock_{}".format(service_name)): mock_name = "mock_{}".format(service_name) mock = getattr(moto, mock_name) if mock is None: return None, None backends = list(mock().backends.values()) if backends: return backends[0], mock_name def get_module_name(o): klass = o.__class__ module = klass.__module__ if module == 'builtins': return klass.__qualname__ # avoid outputs like 'builtins.str' return module + '.' + klass.__qualname__ def calculate_extended_implementation_coverage(): service_names = Session().get_available_services() coverage = {} for service_name in service_names: moto_client, mock_name = get_moto_implementation(service_name) if not moto_client: continue real_client = boto3.client(service_name, region_name="us-east-1") implemented = dict() not_implemented = [] operation_names = [ xform_name(op) for op in real_client.meta.service_model.operation_names ] for op in operation_names: if moto_client and op in dir(moto_client): implemented[op] = getattr(moto_client, op) else: not_implemented.append(op) coverage[service_name] = { "docs": moto_client.__doc__, "module_name": get_module_name(moto_client), "name": mock_name, "implemented": implemented, "not_implemented": not_implemented, } return coverage def calculate_implementation_coverage(): service_names = Session().get_available_services() coverage = {} for service_name in service_names: moto_client, _ = get_moto_implementation(service_name) real_client = boto3.client(service_name, region_name="us-east-1") implemented = [] not_implemented = [] operation_names = [ xform_name(op) for op in real_client.meta.service_model.operation_names ] for op in operation_names: if moto_client and op in dir(moto_client): implemented.append(op) else: not_implemented.append(op) coverage[service_name] = { "implemented": implemented, "not_implemented": not_implemented, } return coverage def print_implementation_coverage(coverage): for service_name in sorted(coverage): implemented = coverage.get(service_name)["implemented"] not_implemented = coverage.get(service_name)["not_implemented"] operations = sorted(implemented + not_implemented) if implemented and not_implemented: percentage_implemented = int( 100.0 * len(implemented) / (len(implemented) + len(not_implemented)) ) elif implemented: percentage_implemented = 100 else: percentage_implemented = 0 print("") print("## {}\n".format(service_name)) print("{}% implemented\n".format(percentage_implemented)) for op in operations: if op in implemented: print("- [X] {}".format(op)) else: print("- [ ] {}".format(op)) def write_implementation_coverage_to_file(coverage): implementation_coverage_file = "{}/../IMPLEMENTATION_COVERAGE.md".format(script_dir) # rewrite the implementation coverage file with updated values # try deleting the implementation coverage file try: os.remove(implementation_coverage_file) except OSError: pass print("Writing to {}".format(implementation_coverage_file)) with open(implementation_coverage_file, "w+") as file: completely_unimplemented = [] for service_name in sorted(coverage): implemented = coverage.get(service_name)["implemented"] if len(implemented) == 0: completely_unimplemented.append(service_name) continue not_implemented = coverage.get(service_name)["not_implemented"] operations = sorted(implemented + not_implemented) if implemented and not_implemented: percentage_implemented = int( 100.0 * len(implemented) / (len(implemented) + len(not_implemented)) ) elif implemented: percentage_implemented = 100 else: percentage_implemented = 0 file.write("\n") file.write("## {}\n".format(service_name)) file.write("
\n") file.write( "{}% implemented\n\n".format(percentage_implemented) ) for op in operations: if op in implemented: file.write("- [X] {}\n".format(op)) else: file.write("- [ ] {}\n".format(op)) file.write("
\n") file.write("\n") file.write("## Unimplemented:\n") file.write("
\n\n") for service in completely_unimplemented: file.write("- {}\n".format(service)) file.write("
") def write_implementation_coverage_to_docs(coverage): implementation_coverage_file = "{}/../docs/docs/services/index.rst".format(script_dir) # rewrite the implementation coverage file with updated values # try deleting the implementation coverage file try: os.remove(implementation_coverage_file) except OSError: pass print("Writing to {}".format(implementation_coverage_file)) completely_unimplemented = [] for service_name in sorted(coverage): implemented = coverage.get(service_name)["implemented"] if len(implemented) == 0: completely_unimplemented.append(service_name) continue not_implemented = coverage.get(service_name)["not_implemented"] operations = sorted(list(implemented.keys()) + not_implemented) service_coverage_file = "{}/../docs/docs/services/{}.rst".format(script_dir, service_name) shorthand = service_name.replace(" ", "_") with open(service_coverage_file, "w+") as file: file.write(f".. _implementedservice_{shorthand}:\n") file.write("\n") file.write(".. |start-h3| raw:: html\n\n") file.write("

") file.write("\n\n") file.write(".. |end-h3| raw:: html\n\n") file.write("

") file.write("\n\n") title = f"{service_name}" file.write("=" * len(title) + "\n") file.write(title + "\n") file.write(("=" * len(title)) + "\n") file.write("\n") if coverage[service_name]["docs"]: # Only show auto-generated documentation if it exists file.write(".. autoclass:: " + coverage[service_name].get("module_name")) file.write("\n\n") file.write("|start-h3| Example usage |end-h3|\n\n") file.write(f""".. sourcecode:: python @{coverage[service_name]['name']} def test_{coverage[service_name]['name'][5:]}_behaviour: boto3.client("{service_name}") ... """) file.write("\n\n") file.write("|start-h3| Implemented features for this service |end-h3|\n\n") for op in operations: if op in implemented: file.write("- [X] {}\n".format(op)) docs = getattr(implemented[op], "__doc__") if docs: file.write(f" {docs}\n\n") else: file.write("- [ ] {}\n".format(op)) file.write("\n") with open(implementation_coverage_file, "w+") as file: file.write(".. _implemented_services:\n") file.write("\n") file.write("\n") file.write("====================\n") file.write("Implemented Services\n") file.write("====================\n") file.write("\n") file.write("Please see a list of all currently supported services. Each service will have a list of the endpoints that are implemented.\n") file.write("Each service will also have an example on how to mock an individual service.\n\n") file.write("Note that you can mock multiple services at the same time:\n\n") file.write(".. sourcecode:: python\n\n") file.write(" @mock_s3\n") file.write(" @mock_sqs\n") file.write(" def test_both_s3_and_sqs():\n") file.write(" ...\n") file.write("\n\n") file.write(".. sourcecode:: python\n\n") file.write(" @mock_all\n") file.write(" def test_all_supported_services_at_the_same_time():\n") file.write(" ...\n") file.write("\n") file.write("\n") file.write(".. toctree::\n") file.write(" :titlesonly:\n") file.write(" :maxdepth: 1\n") file.write(" :glob:\n") file.write("\n") file.write(" *\n") if __name__ == "__main__": cov = calculate_implementation_coverage() write_implementation_coverage_to_file(cov) xcov = calculate_extended_implementation_coverage() write_implementation_coverage_to_docs(xcov)