2017-02-07 08:43:41 +03:00
|
|
|
# This file is part of the MicroPython project, http://micropython.org/
|
|
|
|
#
|
|
|
|
# The MIT License (MIT)
|
|
|
|
#
|
|
|
|
# Copyright (c) 2016 Rami Ali
|
|
|
|
#
|
|
|
|
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
# of this software and associated documentation files (the "Software"), to deal
|
|
|
|
# in the Software without restriction, including without limitation the rights
|
|
|
|
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
# copies of the Software, and to permit persons to whom the Software is
|
|
|
|
# furnished to do so, subject to the following conditions:
|
|
|
|
#
|
|
|
|
# The above copyright notice and this permission notice shall be included in
|
|
|
|
# all copies or substantial portions of the Software.
|
|
|
|
#
|
|
|
|
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
# THE SOFTWARE.
|
|
|
|
|
|
|
|
""" gen-cpydiff generates documentation which outlines operations that differ between MicroPython
|
|
|
|
and CPython. This script is called by the docs Makefile for html and Latex and may be run
|
|
|
|
manually using the command make gen-cpydiff. """
|
|
|
|
|
|
|
|
import os
|
|
|
|
import subprocess
|
|
|
|
import time
|
|
|
|
import re
|
|
|
|
from collections import namedtuple
|
|
|
|
|
2017-06-30 10:22:17 +03:00
|
|
|
# MicroPython supports syntax of CPython 3.4 with some features from 3.5, and
|
2017-02-25 15:53:56 +03:00
|
|
|
# such version should be used to test for differences. If your default python3
|
|
|
|
# executable is of lower version, you can point MICROPY_CPYTHON3 environment var
|
|
|
|
# to the correct executable.
|
|
|
|
if os.name == "nt":
|
|
|
|
CPYTHON3 = os.getenv("MICROPY_CPYTHON3", "python3.exe")
|
2022-11-29 18:58:21 +03:00
|
|
|
MICROPYTHON = os.getenv(
|
|
|
|
"MICROPY_MICROPYTHON", "../ports/windows/build-standard/micropython.exe"
|
|
|
|
)
|
2017-02-25 15:53:56 +03:00
|
|
|
else:
|
|
|
|
CPYTHON3 = os.getenv("MICROPY_CPYTHON3", "python3")
|
2022-08-18 04:46:53 +03:00
|
|
|
MICROPYTHON = os.getenv("MICROPY_MICROPYTHON", "../ports/unix/build-standard/micropython")
|
2020-02-27 07:36:53 +03:00
|
|
|
|
2020-08-22 18:49:10 +03:00
|
|
|
TESTPATH = "../tests/cpydiff"
|
|
|
|
DOCPATH = "../docs/genrst"
|
2017-02-07 08:43:41 +03:00
|
|
|
INDEXTEMPLATE = "../docs/differences/index_template.txt"
|
|
|
|
INDEX = "index.rst"
|
2020-02-27 07:36:53 +03:00
|
|
|
|
2017-02-07 08:43:41 +03:00
|
|
|
HEADER = ".. This document was generated by tools/gen-cpydiff.py\n\n"
|
2020-01-01 14:30:18 +03:00
|
|
|
CLASSMAP = {"Core": "Core language", "Types": "Builtin types"}
|
2017-02-07 08:43:41 +03:00
|
|
|
INDEXPRIORITY = ["syntax", "core_language", "builtin_types", "modules"]
|
|
|
|
RSTCHARS = ["=", "-", "~", "`", ":"]
|
|
|
|
SPLIT = '"""\n|categories: |description: |cause: |workaround: '
|
|
|
|
TAB = " "
|
2020-02-27 07:36:53 +03:00
|
|
|
|
2017-02-07 08:43:41 +03:00
|
|
|
Output = namedtuple(
|
2020-02-27 07:36:53 +03:00
|
|
|
"output",
|
|
|
|
[
|
|
|
|
"name",
|
|
|
|
"class_",
|
|
|
|
"desc",
|
|
|
|
"cause",
|
2017-02-07 08:43:41 +03:00
|
|
|
"workaround",
|
2020-02-27 07:36:53 +03:00
|
|
|
"code",
|
2017-02-07 08:43:41 +03:00
|
|
|
"output_cpy",
|
|
|
|
"output_upy",
|
2020-02-27 07:36:53 +03:00
|
|
|
"status",
|
|
|
|
],
|
|
|
|
)
|
2017-02-07 08:43:41 +03:00
|
|
|
|
|
|
|
|
|
|
|
def readfiles():
|
|
|
|
"""Reads test files"""
|
|
|
|
tests = list(filter(lambda x: x.endswith(".py"), os.listdir(TESTPATH)))
|
|
|
|
tests.sort()
|
|
|
|
files = []
|
|
|
|
|
|
|
|
for test in tests:
|
2020-08-22 18:49:10 +03:00
|
|
|
test_fullpath = os.path.join(TESTPATH, test)
|
|
|
|
text = open(test_fullpath, "r").read()
|
2017-02-07 08:43:41 +03:00
|
|
|
|
|
|
|
try:
|
|
|
|
class_, desc, cause, workaround, code = [
|
|
|
|
x.rstrip() for x in list(filter(None, re.split(SPLIT, text)))
|
|
|
|
]
|
2022-03-23 22:18:23 +03:00
|
|
|
|
|
|
|
# remove black `fmt: on/off/skip` comments
|
|
|
|
code = "".join(
|
|
|
|
# skip comments are inline, so we replace just the comment
|
|
|
|
re.sub(r"\s*# fmt: skip", "", x)
|
|
|
|
for x in code.splitlines(keepends=True)
|
|
|
|
# on/off comments are on their own line, so we omit the entire line
|
|
|
|
if not re.match(r"\s*# fmt: (on|off)\s*", x)
|
|
|
|
)
|
|
|
|
|
2017-02-07 08:43:41 +03:00
|
|
|
output = Output(test, class_, desc, cause, workaround, code, "", "", "")
|
|
|
|
files.append(output)
|
|
|
|
except IndexError:
|
2020-08-22 18:49:10 +03:00
|
|
|
print("Incorrect format in file " + test_fullpath)
|
2017-02-07 08:43:41 +03:00
|
|
|
|
|
|
|
return files
|
|
|
|
|
2020-02-27 07:36:53 +03:00
|
|
|
|
2017-02-07 08:43:41 +03:00
|
|
|
def run_tests(tests):
|
|
|
|
"""executes all tests"""
|
|
|
|
results = []
|
|
|
|
for test in tests:
|
2020-08-22 18:49:10 +03:00
|
|
|
test_fullpath = os.path.join(TESTPATH, test.name)
|
|
|
|
with open(test_fullpath, "rb") as f:
|
2021-08-12 08:28:12 +03:00
|
|
|
input_py = f.read()
|
2017-02-07 08:43:41 +03:00
|
|
|
|
2017-02-25 15:53:56 +03:00
|
|
|
process = subprocess.Popen(
|
|
|
|
CPYTHON3,
|
|
|
|
shell=True,
|
|
|
|
stdout=subprocess.PIPE,
|
|
|
|
stdin=subprocess.PIPE,
|
|
|
|
stderr=subprocess.PIPE,
|
|
|
|
)
|
2021-08-12 08:28:12 +03:00
|
|
|
output_cpy = [com.decode("utf8") for com in process.communicate(input_py)]
|
2017-02-07 08:43:41 +03:00
|
|
|
|
2017-02-25 15:53:56 +03:00
|
|
|
process = subprocess.Popen(
|
|
|
|
MICROPYTHON,
|
|
|
|
shell=True,
|
|
|
|
stdout=subprocess.PIPE,
|
|
|
|
stdin=subprocess.PIPE,
|
|
|
|
stderr=subprocess.PIPE,
|
|
|
|
)
|
2021-08-12 08:28:12 +03:00
|
|
|
output_upy = [com.decode("utf8") for com in process.communicate(input_py)]
|
2017-02-07 08:43:41 +03:00
|
|
|
|
|
|
|
if output_cpy[0] == output_upy[0] and output_cpy[1] == output_upy[1]:
|
|
|
|
status = "Supported"
|
2020-08-22 18:49:10 +03:00
|
|
|
print("Supported operation!\nFile: " + test_fullpath)
|
2017-02-07 08:43:41 +03:00
|
|
|
else:
|
|
|
|
status = "Unsupported"
|
|
|
|
|
|
|
|
output = Output(
|
|
|
|
test.name,
|
|
|
|
test.class_,
|
|
|
|
test.desc,
|
|
|
|
test.cause,
|
|
|
|
test.workaround,
|
|
|
|
test.code,
|
|
|
|
output_cpy,
|
|
|
|
output_upy,
|
|
|
|
status,
|
|
|
|
)
|
|
|
|
results.append(output)
|
|
|
|
|
|
|
|
results.sort(key=lambda x: x.class_)
|
|
|
|
return results
|
|
|
|
|
2020-02-27 07:36:53 +03:00
|
|
|
|
2017-02-07 08:43:41 +03:00
|
|
|
def indent(block, spaces):
|
|
|
|
"""indents paragraphs of text for rst formatting"""
|
|
|
|
new_block = ""
|
|
|
|
for line in block.split("\n"):
|
|
|
|
new_block += spaces + line + "\n"
|
|
|
|
return new_block
|
|
|
|
|
2020-02-27 07:36:53 +03:00
|
|
|
|
2017-02-07 08:43:41 +03:00
|
|
|
def gen_table(contents):
|
|
|
|
"""creates a table given any set of columns"""
|
|
|
|
xlengths = []
|
|
|
|
ylengths = []
|
|
|
|
for column in contents:
|
|
|
|
col_len = 0
|
|
|
|
for entry in column:
|
|
|
|
lines = entry.split("\n")
|
|
|
|
for line in lines:
|
|
|
|
col_len = max(len(line) + 2, col_len)
|
|
|
|
xlengths.append(col_len)
|
|
|
|
for i in range(len(contents[0])):
|
|
|
|
ymax = 0
|
|
|
|
for j in range(len(contents)):
|
|
|
|
ymax = max(ymax, len(contents[j][i].split("\n")))
|
|
|
|
ylengths.append(ymax)
|
|
|
|
|
|
|
|
table_divider = "+" + "".join(["-" * i + "+" for i in xlengths]) + "\n"
|
|
|
|
table = table_divider
|
|
|
|
for i in range(len(ylengths)):
|
|
|
|
row = [column[i] for column in contents]
|
|
|
|
row = [entry + "\n" * (ylengths[i] - len(entry.split("\n"))) for entry in row]
|
|
|
|
row = [entry.split("\n") for entry in row]
|
|
|
|
for j in range(ylengths[i]):
|
|
|
|
k = 0
|
|
|
|
for entry in row:
|
|
|
|
width = xlengths[k]
|
|
|
|
table += "".join(["| {:{}}".format(entry[j], width - 1)])
|
|
|
|
k += 1
|
|
|
|
table += "|\n"
|
|
|
|
table += table_divider
|
|
|
|
return table + "\n"
|
2020-02-27 07:36:53 +03:00
|
|
|
|
2017-02-07 08:43:41 +03:00
|
|
|
|
|
|
|
def gen_rst(results):
|
|
|
|
"""creates restructured text documents to display tests"""
|
|
|
|
|
|
|
|
# make sure the destination directory exists
|
2020-08-22 18:49:10 +03:00
|
|
|
if not os.path.isdir(DOCPATH):
|
2017-02-07 08:43:41 +03:00
|
|
|
os.mkdir(DOCPATH)
|
|
|
|
|
|
|
|
toctree = []
|
|
|
|
class_ = []
|
|
|
|
for output in results:
|
|
|
|
section = output.class_.split(",")
|
|
|
|
for i in range(len(section)):
|
|
|
|
section[i] = section[i].rstrip()
|
|
|
|
if section[i] in CLASSMAP:
|
|
|
|
section[i] = CLASSMAP[section[i]]
|
|
|
|
if i >= len(class_) or section[i] != class_[i]:
|
|
|
|
if i == 0:
|
|
|
|
filename = section[i].replace(" ", "_").lower()
|
2020-08-22 18:49:10 +03:00
|
|
|
rst = open(os.path.join(DOCPATH, filename + ".rst"), "w")
|
2017-02-07 08:43:41 +03:00
|
|
|
rst.write(HEADER)
|
|
|
|
rst.write(section[i] + "\n")
|
|
|
|
rst.write(RSTCHARS[0] * len(section[i]))
|
|
|
|
rst.write(time.strftime("\nGenerated %a %d %b %Y %X UTC\n\n", time.gmtime()))
|
|
|
|
toctree.append(filename)
|
|
|
|
else:
|
|
|
|
rst.write(section[i] + "\n")
|
|
|
|
rst.write(RSTCHARS[min(i, len(RSTCHARS) - 1)] * len(section[i]))
|
|
|
|
rst.write("\n\n")
|
|
|
|
class_ = section
|
2020-08-22 18:49:10 +03:00
|
|
|
rst.write(".. _cpydiff_%s:\n\n" % os.path.splitext(output.name)[0])
|
2017-07-09 13:51:40 +03:00
|
|
|
rst.write(output.desc + "\n")
|
|
|
|
rst.write("~" * len(output.desc) + "\n\n")
|
2017-02-07 08:43:41 +03:00
|
|
|
if output.cause != "Unknown":
|
|
|
|
rst.write("**Cause:** " + output.cause + "\n\n")
|
|
|
|
if output.workaround != "Unknown":
|
|
|
|
rst.write("**Workaround:** " + output.workaround + "\n\n")
|
2020-02-27 07:36:53 +03:00
|
|
|
|
2017-02-07 08:43:41 +03:00
|
|
|
rst.write("Sample code::\n\n" + indent(output.code, TAB) + "\n")
|
|
|
|
output_cpy = indent("".join(output.output_cpy[0:2]), TAB).rstrip()
|
|
|
|
output_cpy = ("::\n\n" if output_cpy != "" else "") + output_cpy
|
|
|
|
output_upy = indent("".join(output.output_upy[0:2]), TAB).rstrip()
|
|
|
|
output_upy = ("::\n\n" if output_upy != "" else "") + output_upy
|
|
|
|
table = gen_table([["CPy output:", output_cpy], ["uPy output:", output_upy]])
|
|
|
|
rst.write(table)
|
|
|
|
|
|
|
|
template = open(INDEXTEMPLATE, "r")
|
2020-08-22 18:49:10 +03:00
|
|
|
index = open(os.path.join(DOCPATH, INDEX), "w")
|
2017-02-07 08:43:41 +03:00
|
|
|
index.write(HEADER)
|
|
|
|
index.write(template.read())
|
|
|
|
for section in INDEXPRIORITY:
|
|
|
|
if section in toctree:
|
|
|
|
index.write(indent(section + ".rst", TAB))
|
|
|
|
toctree.remove(section)
|
|
|
|
for section in toctree:
|
|
|
|
index.write(indent(section + ".rst", TAB))
|
2020-02-27 07:36:53 +03:00
|
|
|
|
2017-02-07 08:43:41 +03:00
|
|
|
|
|
|
|
def main():
|
|
|
|
"""Main function"""
|
|
|
|
|
2017-02-22 07:50:58 +03:00
|
|
|
# set search path so that test scripts find the test modules (and no other ones)
|
|
|
|
os.environ["PYTHONPATH"] = TESTPATH
|
|
|
|
os.environ["MICROPYPATH"] = TESTPATH
|
2017-02-07 08:43:41 +03:00
|
|
|
|
|
|
|
files = readfiles()
|
|
|
|
results = run_tests(files)
|
|
|
|
gen_rst(results)
|
|
|
|
|
2020-02-27 07:36:53 +03:00
|
|
|
|
2017-02-07 08:43:41 +03:00
|
|
|
main()
|