moto/scripts/implementation_coverage.py

282 lines
9.9 KiB
Python
Raw Normal View History

2017-08-14 21:51:50 +00:00
#!/usr/bin/env python
import moto
2018-05-09 07:21:15 +00:00
import os
from botocore import xform_name
from botocore.session import Session
import boto3
script_dir = os.path.dirname(os.path.abspath(__file__))
2022-03-09 17:57:25 +00:00
alternative_service_names = {"lambda": "awslambda"}
def get_moto_implementation(service_name):
2020-10-06 06:46:05 +00:00
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
)
2021-11-08 23:04:44 +00:00
mock = None
mock_name = None
if hasattr(moto, "mock_{}".format(alt_service_name)):
2021-11-08 23:04:44 +00:00
mock_name = "mock_{}".format(alt_service_name)
mock = getattr(moto, mock_name)
elif hasattr(moto, "mock_{}".format(service_name)):
2021-11-08 23:04:44 +00:00
mock_name = "mock_{}".format(service_name)
mock = getattr(moto, mock_name)
if mock is None:
2021-11-08 23:04:44 +00:00
return None, None
backends = list(mock().backends.values())
if backends:
2021-11-08 23:04:44 +00:00
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:
2021-11-08 23:04:44 +00:00
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),
2021-11-08 23:04:44 +00:00
"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:
2021-11-08 23:04:44 +00:00
moto_client, _ = get_moto_implementation(service_name)
2020-10-06 06:46:05 +00:00
real_client = boto3.client(service_name, region_name="us-east-1")
implemented = []
not_implemented = []
2020-10-06 06:46:05 +00:00
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] = {
2020-10-06 06:46:05 +00:00
"implemented": implemented,
"not_implemented": not_implemented,
}
return coverage
2018-05-09 07:21:15 +00:00
def print_implementation_coverage(coverage):
2017-11-20 18:34:40 +00:00
for service_name in sorted(coverage):
2020-10-06 06:46:05 +00:00
implemented = coverage.get(service_name)["implemented"]
not_implemented = coverage.get(service_name)["not_implemented"]
operations = sorted(implemented + not_implemented)
if implemented and not_implemented:
2020-10-06 06:46:05 +00:00
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))
2018-05-09 07:21:15 +00:00
def write_implementation_coverage_to_file(coverage):
implementation_coverage_file = "{}/../IMPLEMENTATION_COVERAGE.md".format(script_dir)
# rewrite the implementation coverage file with updated values
2018-05-09 07:21:15 +00:00
# try deleting the implementation coverage file
try:
os.remove(implementation_coverage_file)
2018-05-09 07:21:15 +00:00
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):
2020-10-06 06:46:05 +00:00
implemented = coverage.get(service_name)["implemented"]
if len(implemented) == 0:
completely_unimplemented.append(service_name)
continue
2020-10-06 06:46:05 +00:00
not_implemented = coverage.get(service_name)["not_implemented"]
operations = sorted(implemented + not_implemented)
if implemented and not_implemented:
2020-10-06 06:46:05 +00:00
percentage_implemented = int(
100.0 * len(implemented) / (len(implemented) + len(not_implemented))
)
elif implemented:
percentage_implemented = 100
else:
percentage_implemented = 0
2018-05-09 07:21:15 +00:00
file.write("\n")
file.write("## {}\n".format(service_name))
file.write("<details>\n")
2020-10-06 06:46:05 +00:00
file.write(
"<summary>{}% implemented</summary>\n\n".format(percentage_implemented)
)
2018-05-09 07:21:15 +00:00
for op in operations:
if op in implemented:
file.write("- [X] {}\n".format(op))
else:
file.write("- [ ] {}\n".format(op))
file.write("</details>\n")
2018-05-09 07:21:15 +00:00
file.write("\n")
file.write("## Unimplemented:\n")
file.write("<details>\n\n")
for service in completely_unimplemented:
file.write("- {}\n".format(service))
file.write("</details>")
2018-05-09 07:21:15 +00:00
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")
2021-11-08 23:04:44 +00:00
file.write(".. |start-h3| raw:: html\n\n")
file.write(" <h3>")
file.write("\n\n")
file.write(".. |end-h3| raw:: html\n\n")
file.write(" </h3>")
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")
2021-11-08 23:04:44 +00:00
file.write("|start-h3| Example usage |end-h3|\n\n")
file.write(f""".. sourcecode:: python
@{coverage[service_name]['name']}
2022-02-18 11:54:58 +00:00
def test_{coverage[service_name]['name'][5:]}_behaviour:
2021-11-08 23:04:44 +00:00
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")
2021-11-08 23:04:44 +00:00
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")
2021-11-08 23:04:44 +00:00
file.write("Please see a list of all currently supported services. Each service will have a list of the endpoints that are implemented.\n")
2021-11-09 22:29:28 +00:00
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")
2021-11-08 23:04:44 +00:00
file.write(" :titlesonly:\n")
file.write(" :maxdepth: 1\n")
file.write(" :glob:\n")
file.write("\n")
2021-11-08 23:04:44 +00:00
file.write(" *\n")
2020-10-06 06:46:05 +00:00
if __name__ == "__main__":
2018-05-09 07:21:15 +00:00
cov = calculate_implementation_coverage()
write_implementation_coverage_to_file(cov)
xcov = calculate_extended_implementation_coverage()
write_implementation_coverage_to_docs(xcov)