| """Create an applet from a Python script. |
| |
| This puts up a dialog asking for a Python source file ('TEXT'). |
| The output is a file with the same name but its ".py" suffix dropped. |
| It is created by copying an applet template and then adding a 'PYC ' |
| resource named __main__ containing the compiled, marshalled script. |
| """ |
| |
| import sys |
| sys.stdout = sys.stderr |
| |
| import string |
| import os |
| import marshal |
| import imp |
| import macfs |
| import MacOS |
| from Res import * |
| |
| # .pyc file (and 'PYC ' resource magic number) |
| MAGIC = imp.get_magic() |
| |
| # Template file (searched on sys.path) |
| TEMPLATE = "PythonApplet" |
| |
| # Specification of our resource |
| RESTYPE = 'PYC ' |
| RESNAME = '__main__' |
| |
| # A resource with this name sets the "owner" (creator) of the destination |
| OWNERNAME = "owner resource" |
| |
| # OpenResFile mode parameters |
| READ = 1 |
| WRITE = 2 |
| |
| def main(): |
| |
| # Find the template |
| # (there's no point in proceeding if we can't find it) |
| |
| for p in sys.path: |
| template = os.path.join(p, TEMPLATE) |
| try: |
| tmpl = open(template, "rb") |
| tmpl.close() |
| break |
| except IOError: |
| continue |
| else: |
| die("Template %s not found" % `template`) |
| return |
| |
| # Ask for source text if not specified in sys.argv[1:] |
| |
| if not sys.argv[1:]: |
| srcfss, ok = macfs.StandardGetFile('TEXT') |
| if not ok: |
| return |
| filename = srcfss.as_pathname() |
| tp, tf = os.path.split(filename) |
| if tf[-3:] == '.py': |
| tf = tf[:-3] |
| else: |
| tf = tf + '.applet' |
| dstfss, ok = macfs.StandardPutFile('Save application as:', tf) |
| if not ok: return |
| process(template, filename, dstfss.as_pathname()) |
| else: |
| |
| # Loop over all files to be processed |
| for filename in sys.argv[1:]: |
| process(template, filename, '') |
| |
| undefs = ('????', ' ', '\0\0\0\0', 'BINA') |
| |
| def process(template, filename, output): |
| |
| print "Processing", `filename`, "..." |
| |
| # Read the source and compile it |
| # (there's no point overwriting the destination if it has a syntax error) |
| |
| fp = open(filename) |
| text = fp.read() |
| fp.close() |
| try: |
| code = compile(text, filename, "exec") |
| except (SyntaxError, EOFError): |
| die("Syntax error in script %s" % `filename`) |
| return |
| |
| # Set the destination file name |
| |
| if string.lower(filename[-3:]) == ".py": |
| destname = filename[:-3] |
| rsrcname = destname + '.rsrc' |
| else: |
| destname = filename + ".applet" |
| rsrcname = filename + '.rsrc' |
| |
| if output: |
| destname = output |
| # Copy the data from the template (creating the file as well) |
| |
| tmpl = open(template, "rb") |
| dest = open(destname, "wb") |
| data = tmpl.read() |
| if data: |
| dest.write(data) |
| dest.close() |
| tmpl.close() |
| |
| # Copy the creator of the template to the destination |
| # unless it already got one. Set type to APPL |
| |
| tctor, ttype = MacOS.GetCreatorAndType(template) |
| ctor, type = MacOS.GetCreatorAndType(destname) |
| if type in undefs: type = 'APPL' |
| if ctor in undefs: ctor = tctor |
| |
| # Open the output resource fork |
| |
| try: |
| output = FSpOpenResFile(destname, WRITE) |
| except MacOS.Error: |
| print "Creating resource fork..." |
| CreateResFile(destname) |
| output = FSpOpenResFile(destname, WRITE) |
| |
| # Copy the resources from the template |
| |
| input = FSpOpenResFile(template, READ) |
| newctor = copyres(input, output) |
| CloseResFile(input) |
| if newctor: ctor = newctor |
| |
| # Copy the resources from the target specific resource template, if any |
| |
| try: |
| input = FSpOpenResFile(rsrcname, READ) |
| except MacOS.Error: |
| pass |
| else: |
| newctor = copyres(input, output) |
| CloseResFile(input) |
| if newctor: ctor = newctor |
| |
| # Now set the creator and type of the destination |
| |
| MacOS.SetCreatorAndType(destname, ctor, type) |
| |
| # Make sure we're manipulating the output resource file now |
| |
| UseResFile(output) |
| |
| # Delete any existing 'PYC 'resource named __main__ |
| |
| try: |
| res = Get1NamedResource(RESTYPE, RESNAME) |
| res.RmveResource() |
| except Error: |
| pass |
| |
| # Create the raw data for the resource from the code object |
| |
| data = marshal.dumps(code) |
| del code |
| data = (MAGIC + '\0\0\0\0') + data |
| |
| # Create the resource and write it |
| |
| id = 0 |
| while id < 128: |
| id = Unique1ID(RESTYPE) |
| res = Resource(data) |
| res.AddResource(RESTYPE, id, RESNAME) |
| res.WriteResource() |
| res.ReleaseResource() |
| |
| # Close the output file |
| |
| CloseResFile(output) |
| |
| # Give positive feedback |
| |
| message("Applet %s created." % `destname`) |
| |
| |
| # Copy resources between two resource file descriptors. |
| # Exception: don't copy a __main__ resource. |
| # If a resource's name is "owner resource", its type is returned |
| # (so the caller can use it to set the destination's creator) |
| |
| def copyres(input, output): |
| ctor = None |
| UseResFile(input) |
| ntypes = Count1Types() |
| for itype in range(1, 1+ntypes): |
| type = Get1IndType(itype) |
| nresources = Count1Resources(type) |
| for ires in range(1, 1+nresources): |
| res = Get1IndResource(type, ires) |
| id, type, name = res.GetResInfo() |
| lcname = string.lower(name) |
| if (type, lcname) == (RESTYPE, RESNAME): |
| continue # Don't copy __main__ from template |
| if lcname == OWNERNAME: ctor = type |
| size = res.SizeResource() |
| attrs = res.GetResAttrs() |
| print id, type, name, size, hex(attrs) |
| res.LoadResource() |
| res.DetachResource() |
| UseResFile(output) |
| try: |
| res2 = Get1Resource(type, id) |
| except MacOS.Error: |
| res2 = None |
| if res2: |
| print "Overwriting..." |
| res2.RmveResource() |
| res.AddResource(type, id, name) |
| res.WriteResource() |
| attrs = attrs | res.GetResAttrs() |
| print "New attrs =", hex(attrs) |
| res.SetResAttrs(attrs) |
| UseResFile(input) |
| return ctor |
| |
| |
| # Show a message and exit |
| |
| def die(str): |
| message(str) |
| sys.exit(1) |
| |
| |
| # Show a message |
| |
| def message(str, id = 256): |
| from Dlg import * |
| d = GetNewDialog(id, -1) |
| if not d: |
| print "Error:", `str` |
| print "DLOG id =", id, "not found." |
| return |
| tp, h, rect = d.GetDItem(2) |
| SetIText(h, str) |
| while 1: |
| n = ModalDialog(None) |
| if n == 1: break |
| del d |
| |
| |
| if __name__ == '__main__': |
| main() |
| |