minor fixes and refusal to use -shk option on Windows

This commit is contained in:
Ivan X 2016-01-01 00:19:32 +09:00
parent 53404e7784
commit 8f6fae4bb4

View File

@ -1,4 +1,4 @@
#!/usr/bin/env python2.7 #!/usr/bin/env python
# vim: set tabstop=4 shiftwidth=4 expandtab filetype=python: # vim: set tabstop=4 shiftwidth=4 expandtab filetype=python:
"""cppo: Copy or catalog one or all files from a ProDOS raw disk image. """cppo: Copy or catalog one or all files from a ProDOS raw disk image.
@ -43,6 +43,7 @@ import datetime
import shutil import shutil
import errno import errno
import uuid import uuid
import subprocess
# Intentially fails on pre-2.6 so user can see what's wrong # Intentially fails on pre-2.6 so user can see what's wrong
b'ERROR: cppo requires Python 2.6 or later, including 3.x.' b'ERROR: cppo requires Python 2.6 or later, including 3.x.'
@ -700,7 +701,7 @@ def writecharHex(arg1, arg2, arg3):
# exit: 21=invalid arg1, 22=invalid arg2, 23=invalid arg3 # exit: 21=invalid arg1, 22=invalid arg2, 23=invalid arg3
if not isinstance(arg1, bytearray): sys.exit(21) if not isinstance(arg1, bytearray): sys.exit(21)
if (arg2<0): sys.exit(22) if (arg2<0): sys.exit(22)
if not isinstance(arg3, type("".encode().decode())): sys.exit(23) if not isinstance(arg3, type("".encode("L1").decode("L1"))): sys.exit(23)
arg1[arg2:arg2+1] = to_bytes(arg3) arg1[arg2:arg2+1] = to_bytes(arg3)
def writecharsHex(arg1, arg2, arg3): def writecharsHex(arg1, arg2, arg3):
@ -712,7 +713,7 @@ def writecharsHex(arg1, arg2, arg3):
# exit: 21=invalid arg1, 22=invalid arg2, 23=invalid arg3 # exit: 21=invalid arg1, 22=invalid arg2, 23=invalid arg3
if not isinstance(arg1, bytearray): sys.exit(21) if not isinstance(arg1, bytearray): sys.exit(21)
if (arg2<0): sys.exit(22) if (arg2<0): sys.exit(22)
if not isinstance(arg3, type("".encode().decode())): sys.exit(23) if not isinstance(arg3, type("".encode("L1").decode("L1"))): sys.exit(23)
arg1[arg2:arg2+len(to_bytes(arg3))] = to_bytes(arg3) arg1[arg2:arg2+len(to_bytes(arg3))] = to_bytes(arg3)
#---- IvanX general purpose functions ----# #---- IvanX general purpose functions ----#
@ -735,8 +736,9 @@ def to_hex(val):
if isinstance(val, bytes): # bytes if isinstance(val, bytes): # bytes
return b2a_hex(val).decode("L1") return b2a_hex(val).decode("L1")
elif isnumber(val): elif isnumber(val):
# .encode().decode() always returns unicode in both P2 and P3 # hex returns str/bytes in P2, but str/unicode in P3, so
return (hex(val)[2:].split("L")[0]).encode("L1").decode("L1") # .encode().decode() always returns unicode in either
return hex(val)[2:].encode("L1").decode("L1").split("L")[0]
else: else:
raise Exception("to_hex() requires bytes, int/long, or [bin-ustr]") raise Exception("to_hex() requires bytes, int/long, or [bin-ustr]")
@ -758,7 +760,7 @@ def to_dec(val):
return int(val[0], 2) return int(val[0], 2)
elif isinstance(val, bytes): # bytes elif isinstance(val, bytes): # bytes
return int(to_hex(val), 16) return int(to_hex(val), 16)
elif isinstance(val, type("".encode().decode())): # hex-ustr elif isinstance(val, type("".encode("L1").decode("L1"))): # hex-ustr
return int(val, 16) return int(val, 16)
else: else:
raise Exception("to_dec() requires bytes, hex-ustr or [bin-ustr]") raise Exception("to_dec() requires bytes, hex-ustr or [bin-ustr]")
@ -767,7 +769,7 @@ def to_bin(val):
"""convert bytes, hex-ustr, or int/long to bin-ustr""" """convert bytes, hex-ustr, or int/long to bin-ustr"""
if isinstance(val, bytes): # bytes if isinstance(val, bytes): # bytes
return (bin(to_dec(to_hex(val))))[2:].encode("L1").decode("L1") return (bin(to_dec(to_hex(val))))[2:].encode("L1").decode("L1")
elif isinstance(val, type("".encode().decode())): # hex-ustr elif isinstance(val, type("".encode("L1").decode("L1"))): # hex-ustr
return (bin(int(val, 16)))[2:].encode("L1").decode("L1") return (bin(int(val, 16)))[2:].encode("L1").decode("L1")
elif isnumber(val): # int/long elif isnumber(val): # int/long
return (bin(val))[2:].encode("L1").decode("L1") return (bin(val))[2:].encode("L1").decode("L1")
@ -781,7 +783,7 @@ def to_bytes(val):
val = to_hex(val[0]) val = to_hex(val[0])
if isnumber(val): # int/long if isnumber(val): # int/long
val = to_hex(val) val = to_hex(val)
if isinstance(val, type("".encode().decode())): # hex-ustr if isinstance(val, type("".encode("L1").decode("L1"))): # hex-ustr
return a2b_hex(bytes(val.encode("L1"))) # works on both P2 and P3 return a2b_hex(bytes(val.encode("L1"))) # works on both P2 and P3
else: else:
raise Exception( raise Exception(
@ -888,10 +890,14 @@ def isnumber(number):
# --- start # --- start
args = sys.argv args = sys.argv
if (len(args) == 1):
while True: # breaks when there are no more arguments starting with dash
if (len(args) == 1):
usage() usage()
while (slyce(args[1],0,1) == "-"): if (slyce(args[1],0,1) != "-"):
break
if (args[1] == "-s"): if (args[1] == "-s"):
g.silent = 1 g.silent = 1
args = args[1:] #shift args = args[1:] #shift
@ -939,7 +945,10 @@ if (g.SHK or
g.imageFile[-3:].lower() == "shk" or g.imageFile[-3:].lower() == "shk" or
g.imageFile[-3:].lower() == "sdk" or g.imageFile[-3:].lower() == "sdk" or
g.imageFile[-3:].lower() == "bxy"): g.imageFile[-3:].lower() == "bxy"):
if not g.AD: if (os.name == "nt"):
print("ShrinkIt archives cannot be extracted on Windows.")
sys.exit(2)
elif not g.AD and not g.DIR:
print("ShrinkIt archives must be used with -ad option.") print("ShrinkIt archives must be used with -ad option.")
sys.exit(2) sys.exit(2)
elif (g.DIR or g.EX): elif (g.DIR or g.EX):
@ -947,13 +956,14 @@ if (g.SHK or
elif (len(args) == 4): elif (len(args) == 4):
print("Only entire ShrinkIt archives can be extracted, not one file.") print("Only entire ShrinkIt archives can be extracted, not one file.")
usage(2) usage(2)
elif (not os.path.isfile("/usr/local/bin/nulib2") and
not os.path.isfile("/usr/bin/nulib2") and
not os.path.isfile("nulib2")):
print("Nulib2 not found. Can't expand ShrinkIt archive.")
sys.exit(2)
else: else:
g.SHK = 1 try:
with open(os.devnull, "w") as fnull:
subprocess.call("nulib2", stdout = fnull, stderr = fnull)
g.SHK=1
except Exception:
print("Nulib2 is not available; not expanding ShrinkIt archive.")
sys.exit(2)
if g.SHK: if g.SHK:
unshkdir = ("/tmp/cppo-" + str(uuid.uuid4())) unshkdir = ("/tmp/cppo-" + str(uuid.uuid4()))
makedirs(unshkdir) makedirs(unshkdir)
@ -963,7 +973,7 @@ if g.SHK:
if not name.startswith(".")] if not name.startswith(".")]
if (len(fileNames) == 1 and os.path.isdir(unshkdir + "/" + fileNames[0])): if (len(fileNames) == 1 and os.path.isdir(unshkdir + "/" + fileNames[0])):
oneDir = True oneDir = True
volumeName=toProDOSName(fileNames[0]) volumeName = toProDOSName(fileNames[0])
else: else:
oneDir = False oneDir = False
volumeName = toProDOSName(os.path.basename(g.imageFile)) volumeName = toProDOSName(os.path.basename(g.imageFile))
@ -975,17 +985,15 @@ if g.SHK:
# recursively process unshrunk archive hierarchy # recursively process unshrunk archive hierarchy
for dirName, subdirList, fileList in os.walk(unshkdir): for dirName, subdirList, fileList in os.walk(unshkdir):
subdirList.sort() subdirList.sort()
g.targetDir = (args[2] + (("/" + volumeName) if not oneDir else "") + g.targetDir = (args[2] + ("" if oneDir else ("/" + volumeName)) +
("/" if (dirName.count('/') > 2) else "") + ("/" if (dirName.count('/') > 2) else "") +
"/".join(dirName.split('/')[3:])) # chop off tempdir "/".join(dirName.split('/')[3:])) # chop off tempdir
g.ADdir = (g.targetDir + "/.AppleDouble") g.ADdir = (g.targetDir + "/.AppleDouble")
print("/".join(dirName.split('/')[3:])) print("/".join(dirName.split('/')[3:]) if oneDir else "\n"+volumeName)
makedirs(g.targetDir) makedirs(g.targetDir)
makedirs(g.ADdir) makedirs(g.ADdir)
for fname in sorted(fileList): for fname in sorted(fileList):
processEntry(dirName, fname) processEntry(dirName, fname)
if (unshkdir.count('/') > 2):
unshkdir = "/".join(dirName.split('/')[0:3])
shutil.rmtree(unshkdir, True) shutil.rmtree(unshkdir, True)
syncExit() syncExit()