summaryrefslogtreecommitdiffstatshomepage
path: root/tools/upip_utarfile.py
diff options
context:
space:
mode:
authorPaul Sokolovsky <pfalcon@users.sourceforge.net>2016-11-07 18:13:56 +0300
committerPaul Sokolovsky <pfalcon@users.sourceforge.net>2016-11-07 18:39:41 +0300
commit61d74fdef836753cbf1cf0e9ee7b8e50cb80f41d (patch)
treedaab5d53086add05df2565248296eb4a3b3de0c7 /tools/upip_utarfile.py
parentbc4441afa753176e1901017708d8a739c38806eb (diff)
downloadmicropython-61d74fdef836753cbf1cf0e9ee7b8e50cb80f41d.tar.gz
micropython-61d74fdef836753cbf1cf0e9ee7b8e50cb80f41d.zip
tools, unix: Replace upip tarball with just source files.
To make its inclusion as frozen modules in multiple ports less magic. Ports are just expected to symlink 2 files into their scripts/modules subdirs. Unix port updated to use this and in general follow frozen modules setup tested and tried on baremetal ports, where there's "scripts" predefined dir (overridable with FROZEN_DIR make var), and a user just drops Python files there.
Diffstat (limited to 'tools/upip_utarfile.py')
-rw-r--r--tools/upip_utarfile.py94
1 files changed, 94 insertions, 0 deletions
diff --git a/tools/upip_utarfile.py b/tools/upip_utarfile.py
new file mode 100644
index 0000000000..65ce0bdca8
--- /dev/null
+++ b/tools/upip_utarfile.py
@@ -0,0 +1,94 @@
+import uctypes
+
+# http://www.gnu.org/software/tar/manual/html_node/Standard.html
+TAR_HEADER = {
+ "name": (uctypes.ARRAY | 0, uctypes.UINT8 | 100),
+ "size": (uctypes.ARRAY | 124, uctypes.UINT8 | 12),
+}
+
+DIRTYPE = "dir"
+REGTYPE = "file"
+
+def roundup(val, align):
+ return (val + align - 1) & ~(align - 1)
+
+class FileSection:
+
+ def __init__(self, f, content_len, aligned_len):
+ self.f = f
+ self.content_len = content_len
+ self.align = aligned_len - content_len
+
+ def read(self, sz=65536):
+ if self.content_len == 0:
+ return b""
+ if sz > self.content_len:
+ sz = self.content_len
+ data = self.f.read(sz)
+ sz = len(data)
+ self.content_len -= sz
+ return data
+
+ def readinto(self, buf):
+ if self.content_len == 0:
+ return 0
+ if len(buf) > self.content_len:
+ buf = memoryview(buf)[:self.content_len]
+ sz = self.f.readinto(buf)
+ self.content_len -= sz
+ return sz
+
+ def skip(self):
+ sz = self.content_len + self.align
+ if sz:
+ buf = bytearray(16)
+ while sz:
+ s = min(sz, 16)
+ self.f.readinto(buf, s)
+ sz -= s
+
+class TarInfo:
+
+ def __str__(self):
+ return "TarInfo(%r, %s, %d)" % (self.name, self.type, self.size)
+
+class TarFile:
+
+ def __init__(self, name=None, fileobj=None):
+ if fileobj:
+ self.f = fileobj
+ else:
+ self.f = open(name, "rb")
+ self.subf = None
+
+ def next(self):
+ if self.subf:
+ self.subf.skip()
+ buf = self.f.read(512)
+ if not buf:
+ return None
+
+ h = uctypes.struct(uctypes.addressof(buf), TAR_HEADER, uctypes.LITTLE_ENDIAN)
+
+ # Empty block means end of archive
+ if h.name[0] == 0:
+ return None
+
+ d = TarInfo()
+ d.name = str(h.name, "utf-8").rstrip()
+ d.size = int(bytes(h.size).rstrip(), 8)
+ d.type = [REGTYPE, DIRTYPE][d.name[-1] == "/"]
+ self.subf = d.subf = FileSection(self.f, d.size, roundup(d.size, 512))
+ return d
+
+ def __iter__(self):
+ return self
+
+ def __next__(self):
+ v = self.next()
+ if v is None:
+ raise StopIteration
+ return v
+
+ def extractfile(self, tarinfo):
+ return tarinfo.subf