PageRenderTime 54ms CodeModel.GetById 18ms RepoModel.GetById 0ms app.codeStats 1ms

/lib-python/2.7/plat-mac/gensuitemodule.py

https://bitbucket.org/dac_io/pypy
Python | 1216 lines | 1093 code | 54 blank | 69 comment | 114 complexity | 49e6e29330d15a5818c933808413ce47 MD5 | raw file
  1. """
  2. gensuitemodule - Generate an AE suite module from an aete/aeut resource
  3. Based on aete.py.
  4. Reading and understanding this code is left as an exercise to the reader.
  5. """
  6. from warnings import warnpy3k
  7. warnpy3k("In 3.x, the gensuitemodule module is removed.", stacklevel=2)
  8. import MacOS
  9. import EasyDialogs
  10. import os
  11. import string
  12. import sys
  13. import types
  14. import StringIO
  15. import keyword
  16. import macresource
  17. import aetools
  18. import distutils.sysconfig
  19. import OSATerminology
  20. from Carbon.Res import *
  21. import Carbon.Folder
  22. import MacOS
  23. import getopt
  24. import plistlib
  25. _MAC_LIB_FOLDER=os.path.dirname(aetools.__file__)
  26. DEFAULT_STANDARD_PACKAGEFOLDER=os.path.join(_MAC_LIB_FOLDER, 'lib-scriptpackages')
  27. DEFAULT_USER_PACKAGEFOLDER=distutils.sysconfig.get_python_lib()
  28. def usage():
  29. sys.stderr.write("Usage: %s [opts] application-or-resource-file\n" % sys.argv[0])
  30. sys.stderr.write("""Options:
  31. --output pkgdir Pathname of the output package (short: -o)
  32. --resource Parse resource file in stead of launching application (-r)
  33. --base package Use another base package in stead of default StdSuites (-b)
  34. --edit old=new Edit suite names, use empty new to skip a suite (-e)
  35. --creator code Set creator code for package (-c)
  36. --dump Dump aete resource to stdout in stead of creating module (-d)
  37. --verbose Tell us what happens (-v)
  38. """)
  39. sys.exit(1)
  40. def main():
  41. if len(sys.argv) > 1:
  42. SHORTOPTS = "rb:o:e:c:dv"
  43. LONGOPTS = ("resource", "base=", "output=", "edit=", "creator=", "dump", "verbose")
  44. try:
  45. opts, args = getopt.getopt(sys.argv[1:], SHORTOPTS, LONGOPTS)
  46. except getopt.GetoptError:
  47. usage()
  48. process_func = processfile
  49. basepkgname = 'StdSuites'
  50. output = None
  51. edit_modnames = []
  52. creatorsignature = None
  53. dump = None
  54. verbose = None
  55. for o, a in opts:
  56. if o in ('-r', '--resource'):
  57. process_func = processfile_fromresource
  58. if o in ('-b', '--base'):
  59. basepkgname = a
  60. if o in ('-o', '--output'):
  61. output = a
  62. if o in ('-e', '--edit'):
  63. split = a.split('=')
  64. if len(split) != 2:
  65. usage()
  66. edit_modnames.append(split)
  67. if o in ('-c', '--creator'):
  68. if len(a) != 4:
  69. sys.stderr.write("creator must be 4-char string\n")
  70. sys.exit(1)
  71. creatorsignature = a
  72. if o in ('-d', '--dump'):
  73. dump = sys.stdout
  74. if o in ('-v', '--verbose'):
  75. verbose = sys.stderr
  76. if output and len(args) > 1:
  77. sys.stderr.write("%s: cannot specify --output with multiple inputs\n" % sys.argv[0])
  78. sys.exit(1)
  79. for filename in args:
  80. process_func(filename, output=output, basepkgname=basepkgname,
  81. edit_modnames=edit_modnames, creatorsignature=creatorsignature,
  82. dump=dump, verbose=verbose)
  83. else:
  84. main_interactive()
  85. def main_interactive(interact=0, basepkgname='StdSuites'):
  86. if interact:
  87. # Ask for save-filename for each module
  88. edit_modnames = None
  89. else:
  90. # Use default filenames for each module
  91. edit_modnames = []
  92. appsfolder = Carbon.Folder.FSFindFolder(-32765, 'apps', 0)
  93. filename = EasyDialogs.AskFileForOpen(
  94. message='Select scriptable application',
  95. dialogOptionFlags=0x1056, # allow selection of .app bundles
  96. defaultLocation=appsfolder)
  97. if not filename:
  98. return
  99. if not is_scriptable(filename):
  100. if EasyDialogs.AskYesNoCancel(
  101. "Warning: application does not seem scriptable",
  102. yes="Continue", default=2, no="") <= 0:
  103. return
  104. try:
  105. processfile(filename, edit_modnames=edit_modnames, basepkgname=basepkgname,
  106. verbose=sys.stderr)
  107. except MacOS.Error, arg:
  108. print "Error getting terminology:", arg
  109. print "Retry, manually parsing resources"
  110. processfile_fromresource(filename, edit_modnames=edit_modnames,
  111. basepkgname=basepkgname, verbose=sys.stderr)
  112. def is_scriptable(application):
  113. """Return true if the application is scriptable"""
  114. if os.path.isdir(application):
  115. plistfile = os.path.join(application, 'Contents', 'Info.plist')
  116. if not os.path.exists(plistfile):
  117. return False
  118. plist = plistlib.Plist.fromFile(plistfile)
  119. return plist.get('NSAppleScriptEnabled', False)
  120. # If it is a file test for an aete/aeut resource.
  121. currf = CurResFile()
  122. try:
  123. refno = macresource.open_pathname(application)
  124. except MacOS.Error:
  125. return False
  126. UseResFile(refno)
  127. n_terminology = Count1Resources('aete') + Count1Resources('aeut') + \
  128. Count1Resources('scsz') + Count1Resources('osiz')
  129. CloseResFile(refno)
  130. UseResFile(currf)
  131. return n_terminology > 0
  132. def processfile_fromresource(fullname, output=None, basepkgname=None,
  133. edit_modnames=None, creatorsignature=None, dump=None, verbose=None):
  134. """Process all resources in a single file"""
  135. if not is_scriptable(fullname) and verbose:
  136. print >>verbose, "Warning: app does not seem scriptable: %s" % fullname
  137. cur = CurResFile()
  138. if verbose:
  139. print >>verbose, "Processing", fullname
  140. rf = macresource.open_pathname(fullname)
  141. try:
  142. UseResFile(rf)
  143. resources = []
  144. for i in range(Count1Resources('aete')):
  145. res = Get1IndResource('aete', 1+i)
  146. resources.append(res)
  147. for i in range(Count1Resources('aeut')):
  148. res = Get1IndResource('aeut', 1+i)
  149. resources.append(res)
  150. if verbose:
  151. print >>verbose, "\nLISTING aete+aeut RESOURCES IN", repr(fullname)
  152. aetelist = []
  153. for res in resources:
  154. if verbose:
  155. print >>verbose, "decoding", res.GetResInfo(), "..."
  156. data = res.data
  157. aete = decode(data, verbose)
  158. aetelist.append((aete, res.GetResInfo()))
  159. finally:
  160. if rf != cur:
  161. CloseResFile(rf)
  162. UseResFile(cur)
  163. # switch back (needed for dialogs in Python)
  164. UseResFile(cur)
  165. if dump:
  166. dumpaetelist(aetelist, dump)
  167. compileaetelist(aetelist, fullname, output=output,
  168. basepkgname=basepkgname, edit_modnames=edit_modnames,
  169. creatorsignature=creatorsignature, verbose=verbose)
  170. def processfile(fullname, output=None, basepkgname=None,
  171. edit_modnames=None, creatorsignature=None, dump=None,
  172. verbose=None):
  173. """Ask an application for its terminology and process that"""
  174. if not is_scriptable(fullname) and verbose:
  175. print >>verbose, "Warning: app does not seem scriptable: %s" % fullname
  176. if verbose:
  177. print >>verbose, "\nASKING FOR aete DICTIONARY IN", repr(fullname)
  178. try:
  179. aedescobj, launched = OSATerminology.GetAppTerminology(fullname)
  180. except MacOS.Error, arg:
  181. if arg[0] in (-1701, -192): # errAEDescNotFound, resNotFound
  182. if verbose:
  183. print >>verbose, "GetAppTerminology failed with errAEDescNotFound/resNotFound, trying manually"
  184. aedata, sig = getappterminology(fullname, verbose=verbose)
  185. if not creatorsignature:
  186. creatorsignature = sig
  187. else:
  188. raise
  189. else:
  190. if launched:
  191. if verbose:
  192. print >>verbose, "Launched", fullname
  193. raw = aetools.unpack(aedescobj)
  194. if not raw:
  195. if verbose:
  196. print >>verbose, 'Unpack returned empty value:', raw
  197. return
  198. if not raw[0].data:
  199. if verbose:
  200. print >>verbose, 'Unpack returned value without data:', raw
  201. return
  202. aedata = raw[0]
  203. aete = decode(aedata.data, verbose)
  204. if dump:
  205. dumpaetelist([aete], dump)
  206. return
  207. compileaete(aete, None, fullname, output=output, basepkgname=basepkgname,
  208. creatorsignature=creatorsignature, edit_modnames=edit_modnames,
  209. verbose=verbose)
  210. def getappterminology(fullname, verbose=None):
  211. """Get application terminology by sending an AppleEvent"""
  212. # First check that we actually can send AppleEvents
  213. if not MacOS.WMAvailable():
  214. raise RuntimeError, "Cannot send AppleEvents, no access to window manager"
  215. # Next, a workaround for a bug in MacOS 10.2: sending events will hang unless
  216. # you have created an event loop first.
  217. import Carbon.Evt
  218. Carbon.Evt.WaitNextEvent(0,0)
  219. if os.path.isdir(fullname):
  220. # Now get the signature of the application, hoping it is a bundle
  221. pkginfo = os.path.join(fullname, 'Contents', 'PkgInfo')
  222. if not os.path.exists(pkginfo):
  223. raise RuntimeError, "No PkgInfo file found"
  224. tp_cr = open(pkginfo, 'rb').read()
  225. cr = tp_cr[4:8]
  226. else:
  227. # Assume it is a file
  228. cr, tp = MacOS.GetCreatorAndType(fullname)
  229. # Let's talk to it and ask for its AETE
  230. talker = aetools.TalkTo(cr)
  231. try:
  232. talker._start()
  233. except (MacOS.Error, aetools.Error), arg:
  234. if verbose:
  235. print >>verbose, 'Warning: start() failed, continuing anyway:', arg
  236. reply = talker.send("ascr", "gdte")
  237. #reply2 = talker.send("ascr", "gdut")
  238. # Now pick the bits out of the return that we need.
  239. return reply[1]['----'], cr
  240. def compileaetelist(aetelist, fullname, output=None, basepkgname=None,
  241. edit_modnames=None, creatorsignature=None, verbose=None):
  242. for aete, resinfo in aetelist:
  243. compileaete(aete, resinfo, fullname, output=output,
  244. basepkgname=basepkgname, edit_modnames=edit_modnames,
  245. creatorsignature=creatorsignature, verbose=verbose)
  246. def dumpaetelist(aetelist, output):
  247. import pprint
  248. pprint.pprint(aetelist, output)
  249. def decode(data, verbose=None):
  250. """Decode a resource into a python data structure"""
  251. f = StringIO.StringIO(data)
  252. aete = generic(getaete, f)
  253. aete = simplify(aete)
  254. processed = f.tell()
  255. unprocessed = len(f.read())
  256. total = f.tell()
  257. if unprocessed and verbose:
  258. verbose.write("%d processed + %d unprocessed = %d total\n" %
  259. (processed, unprocessed, total))
  260. return aete
  261. def simplify(item):
  262. """Recursively replace singleton tuples by their constituent item"""
  263. if type(item) is types.ListType:
  264. return map(simplify, item)
  265. elif type(item) == types.TupleType and len(item) == 2:
  266. return simplify(item[1])
  267. else:
  268. return item
  269. # Here follows the aete resource decoder.
  270. # It is presented bottom-up instead of top-down because there are direct
  271. # references to the lower-level part-decoders from the high-level part-decoders.
  272. def getbyte(f, *args):
  273. c = f.read(1)
  274. if not c:
  275. raise EOFError, 'in getbyte' + str(args)
  276. return ord(c)
  277. def getword(f, *args):
  278. getalign(f)
  279. s = f.read(2)
  280. if len(s) < 2:
  281. raise EOFError, 'in getword' + str(args)
  282. return (ord(s[0])<<8) | ord(s[1])
  283. def getlong(f, *args):
  284. getalign(f)
  285. s = f.read(4)
  286. if len(s) < 4:
  287. raise EOFError, 'in getlong' + str(args)
  288. return (ord(s[0])<<24) | (ord(s[1])<<16) | (ord(s[2])<<8) | ord(s[3])
  289. def getostype(f, *args):
  290. getalign(f)
  291. s = f.read(4)
  292. if len(s) < 4:
  293. raise EOFError, 'in getostype' + str(args)
  294. return s
  295. def getpstr(f, *args):
  296. c = f.read(1)
  297. if len(c) < 1:
  298. raise EOFError, 'in getpstr[1]' + str(args)
  299. nbytes = ord(c)
  300. if nbytes == 0: return ''
  301. s = f.read(nbytes)
  302. if len(s) < nbytes:
  303. raise EOFError, 'in getpstr[2]' + str(args)
  304. return s
  305. def getalign(f):
  306. if f.tell() & 1:
  307. c = f.read(1)
  308. ##if c != '\0':
  309. ## print align:', repr(c)
  310. def getlist(f, description, getitem):
  311. count = getword(f)
  312. list = []
  313. for i in range(count):
  314. list.append(generic(getitem, f))
  315. getalign(f)
  316. return list
  317. def alt_generic(what, f, *args):
  318. print "generic", repr(what), args
  319. res = vageneric(what, f, args)
  320. print '->', repr(res)
  321. return res
  322. def generic(what, f, *args):
  323. if type(what) == types.FunctionType:
  324. return apply(what, (f,) + args)
  325. if type(what) == types.ListType:
  326. record = []
  327. for thing in what:
  328. item = apply(generic, thing[:1] + (f,) + thing[1:])
  329. record.append((thing[1], item))
  330. return record
  331. return "BAD GENERIC ARGS: %r" % (what,)
  332. getdata = [
  333. (getostype, "type"),
  334. (getpstr, "description"),
  335. (getword, "flags")
  336. ]
  337. getargument = [
  338. (getpstr, "name"),
  339. (getostype, "keyword"),
  340. (getdata, "what")
  341. ]
  342. getevent = [
  343. (getpstr, "name"),
  344. (getpstr, "description"),
  345. (getostype, "suite code"),
  346. (getostype, "event code"),
  347. (getdata, "returns"),
  348. (getdata, "accepts"),
  349. (getlist, "optional arguments", getargument)
  350. ]
  351. getproperty = [
  352. (getpstr, "name"),
  353. (getostype, "code"),
  354. (getdata, "what")
  355. ]
  356. getelement = [
  357. (getostype, "type"),
  358. (getlist, "keyform", getostype)
  359. ]
  360. getclass = [
  361. (getpstr, "name"),
  362. (getostype, "class code"),
  363. (getpstr, "description"),
  364. (getlist, "properties", getproperty),
  365. (getlist, "elements", getelement)
  366. ]
  367. getcomparison = [
  368. (getpstr, "operator name"),
  369. (getostype, "operator ID"),
  370. (getpstr, "operator comment"),
  371. ]
  372. getenumerator = [
  373. (getpstr, "enumerator name"),
  374. (getostype, "enumerator ID"),
  375. (getpstr, "enumerator comment")
  376. ]
  377. getenumeration = [
  378. (getostype, "enumeration ID"),
  379. (getlist, "enumerator", getenumerator)
  380. ]
  381. getsuite = [
  382. (getpstr, "suite name"),
  383. (getpstr, "suite description"),
  384. (getostype, "suite ID"),
  385. (getword, "suite level"),
  386. (getword, "suite version"),
  387. (getlist, "events", getevent),
  388. (getlist, "classes", getclass),
  389. (getlist, "comparisons", getcomparison),
  390. (getlist, "enumerations", getenumeration)
  391. ]
  392. getaete = [
  393. (getword, "major/minor version in BCD"),
  394. (getword, "language code"),
  395. (getword, "script code"),
  396. (getlist, "suites", getsuite)
  397. ]
  398. def compileaete(aete, resinfo, fname, output=None, basepkgname=None,
  399. edit_modnames=None, creatorsignature=None, verbose=None):
  400. """Generate code for a full aete resource. fname passed for doc purposes"""
  401. [version, language, script, suites] = aete
  402. major, minor = divmod(version, 256)
  403. if not creatorsignature:
  404. creatorsignature, dummy = MacOS.GetCreatorAndType(fname)
  405. packagename = identify(os.path.splitext(os.path.basename(fname))[0])
  406. if language:
  407. packagename = packagename+'_lang%d'%language
  408. if script:
  409. packagename = packagename+'_script%d'%script
  410. if len(packagename) > 27:
  411. packagename = packagename[:27]
  412. if output:
  413. # XXXX Put this in site-packages if it isn't a full pathname?
  414. if not os.path.exists(output):
  415. os.mkdir(output)
  416. pathname = output
  417. else:
  418. pathname = EasyDialogs.AskFolder(message='Create and select package folder for %s'%packagename,
  419. defaultLocation=DEFAULT_USER_PACKAGEFOLDER)
  420. output = pathname
  421. if not pathname:
  422. return
  423. packagename = os.path.split(os.path.normpath(pathname))[1]
  424. if not basepkgname:
  425. basepkgname = EasyDialogs.AskFolder(message='Package folder for base suite (usually StdSuites)',
  426. defaultLocation=DEFAULT_STANDARD_PACKAGEFOLDER)
  427. if basepkgname:
  428. dirname, basepkgname = os.path.split(os.path.normpath(basepkgname))
  429. if dirname and not dirname in sys.path:
  430. sys.path.insert(0, dirname)
  431. basepackage = __import__(basepkgname)
  432. else:
  433. basepackage = None
  434. suitelist = []
  435. allprecompinfo = []
  436. allsuites = []
  437. for suite in suites:
  438. compiler = SuiteCompiler(suite, basepackage, output, edit_modnames, verbose)
  439. code, modname, precompinfo = compiler.precompilesuite()
  440. if not code:
  441. continue
  442. allprecompinfo = allprecompinfo + precompinfo
  443. suiteinfo = suite, pathname, modname
  444. suitelist.append((code, modname))
  445. allsuites.append(compiler)
  446. for compiler in allsuites:
  447. compiler.compilesuite(major, minor, language, script, fname, allprecompinfo)
  448. initfilename = os.path.join(output, '__init__.py')
  449. fp = open(initfilename, 'w')
  450. MacOS.SetCreatorAndType(initfilename, 'Pyth', 'TEXT')
  451. fp.write('"""\n')
  452. fp.write("Package generated from %s\n"%ascii(fname))
  453. if resinfo:
  454. fp.write("Resource %s resid %d %s\n"%(ascii(resinfo[1]), resinfo[0], ascii(resinfo[2])))
  455. fp.write('"""\n')
  456. fp.write('import aetools\n')
  457. fp.write('Error = aetools.Error\n')
  458. suitelist.sort()
  459. for code, modname in suitelist:
  460. fp.write("import %s\n" % modname)
  461. fp.write("\n\n_code_to_module = {\n")
  462. for code, modname in suitelist:
  463. fp.write(" '%s' : %s,\n"%(ascii(code), modname))
  464. fp.write("}\n\n")
  465. fp.write("\n\n_code_to_fullname = {\n")
  466. for code, modname in suitelist:
  467. fp.write(" '%s' : ('%s.%s', '%s'),\n"%(ascii(code), packagename, modname, modname))
  468. fp.write("}\n\n")
  469. for code, modname in suitelist:
  470. fp.write("from %s import *\n"%modname)
  471. # Generate property dicts and element dicts for all types declared in this module
  472. fp.write("\ndef getbaseclasses(v):\n")
  473. fp.write(" if not getattr(v, '_propdict', None):\n")
  474. fp.write(" v._propdict = {}\n")
  475. fp.write(" v._elemdict = {}\n")
  476. fp.write(" for superclassname in getattr(v, '_superclassnames', []):\n")
  477. fp.write(" superclass = eval(superclassname)\n")
  478. fp.write(" getbaseclasses(superclass)\n")
  479. fp.write(" v._propdict.update(getattr(superclass, '_propdict', {}))\n")
  480. fp.write(" v._elemdict.update(getattr(superclass, '_elemdict', {}))\n")
  481. fp.write(" v._propdict.update(getattr(v, '_privpropdict', {}))\n")
  482. fp.write(" v._elemdict.update(getattr(v, '_privelemdict', {}))\n")
  483. fp.write("\n")
  484. fp.write("import StdSuites\n")
  485. allprecompinfo.sort()
  486. if allprecompinfo:
  487. fp.write("\n#\n# Set property and element dictionaries now that all classes have been defined\n#\n")
  488. for codenamemapper in allprecompinfo:
  489. for k, v in codenamemapper.getall('class'):
  490. fp.write("getbaseclasses(%s)\n" % v)
  491. # Generate a code-to-name mapper for all of the types (classes) declared in this module
  492. application_class = None
  493. if allprecompinfo:
  494. fp.write("\n#\n# Indices of types declared in this module\n#\n")
  495. fp.write("_classdeclarations = {\n")
  496. for codenamemapper in allprecompinfo:
  497. for k, v in codenamemapper.getall('class'):
  498. fp.write(" %r : %s,\n" % (k, v))
  499. if k == 'capp':
  500. application_class = v
  501. fp.write("}\n")
  502. if suitelist:
  503. fp.write("\n\nclass %s(%s_Events"%(packagename, suitelist[0][1]))
  504. for code, modname in suitelist[1:]:
  505. fp.write(",\n %s_Events"%modname)
  506. fp.write(",\n aetools.TalkTo):\n")
  507. fp.write(" _signature = %r\n\n"%(creatorsignature,))
  508. fp.write(" _moduleName = '%s'\n\n"%packagename)
  509. if application_class:
  510. fp.write(" _elemdict = %s._elemdict\n" % application_class)
  511. fp.write(" _propdict = %s._propdict\n" % application_class)
  512. fp.close()
  513. class SuiteCompiler:
  514. def __init__(self, suite, basepackage, output, edit_modnames, verbose):
  515. self.suite = suite
  516. self.basepackage = basepackage
  517. self.edit_modnames = edit_modnames
  518. self.output = output
  519. self.verbose = verbose
  520. # Set by precompilesuite
  521. self.pathname = None
  522. self.modname = None
  523. # Set by compilesuite
  524. self.fp = None
  525. self.basemodule = None
  526. self.enumsneeded = {}
  527. def precompilesuite(self):
  528. """Parse a single suite without generating the output. This step is needed
  529. so we can resolve recursive references by suites to enums/comps/etc declared
  530. in other suites"""
  531. [name, desc, code, level, version, events, classes, comps, enums] = self.suite
  532. modname = identify(name)
  533. if len(modname) > 28:
  534. modname = modname[:27]
  535. if self.edit_modnames is None:
  536. self.pathname = EasyDialogs.AskFileForSave(message='Python output file',
  537. savedFileName=modname+'.py')
  538. else:
  539. for old, new in self.edit_modnames:
  540. if old == modname:
  541. modname = new
  542. if modname:
  543. self.pathname = os.path.join(self.output, modname + '.py')
  544. else:
  545. self.pathname = None
  546. if not self.pathname:
  547. return None, None, None
  548. self.modname = os.path.splitext(os.path.split(self.pathname)[1])[0]
  549. if self.basepackage and code in self.basepackage._code_to_module:
  550. # We are an extension of a baseclass (usually an application extending
  551. # Standard_Suite or so). Import everything from our base module
  552. basemodule = self.basepackage._code_to_module[code]
  553. else:
  554. # We are not an extension.
  555. basemodule = None
  556. self.enumsneeded = {}
  557. for event in events:
  558. self.findenumsinevent(event)
  559. objc = ObjectCompiler(None, self.modname, basemodule, interact=(self.edit_modnames is None),
  560. verbose=self.verbose)
  561. for cls in classes:
  562. objc.compileclass(cls)
  563. for cls in classes:
  564. objc.fillclasspropsandelems(cls)
  565. for comp in comps:
  566. objc.compilecomparison(comp)
  567. for enum in enums:
  568. objc.compileenumeration(enum)
  569. for enum in self.enumsneeded.keys():
  570. objc.checkforenum(enum)
  571. objc.dumpindex()
  572. precompinfo = objc.getprecompinfo(self.modname)
  573. return code, self.modname, precompinfo
  574. def compilesuite(self, major, minor, language, script, fname, precompinfo):
  575. """Generate code for a single suite"""
  576. [name, desc, code, level, version, events, classes, comps, enums] = self.suite
  577. # Sort various lists, so re-generated source is easier compared
  578. def class_sorter(k1, k2):
  579. """Sort classes by code, and make sure main class sorts before synonyms"""
  580. # [name, code, desc, properties, elements] = cls
  581. if k1[1] < k2[1]: return -1
  582. if k1[1] > k2[1]: return 1
  583. if not k2[3] or k2[3][0][1] == 'c@#!':
  584. # This is a synonym, the other one is better
  585. return -1
  586. if not k1[3] or k1[3][0][1] == 'c@#!':
  587. # This is a synonym, the other one is better
  588. return 1
  589. return 0
  590. events.sort()
  591. classes.sort(class_sorter)
  592. comps.sort()
  593. enums.sort()
  594. self.fp = fp = open(self.pathname, 'w')
  595. MacOS.SetCreatorAndType(self.pathname, 'Pyth', 'TEXT')
  596. fp.write('"""Suite %s: %s\n' % (ascii(name), ascii(desc)))
  597. fp.write("Level %d, version %d\n\n" % (level, version))
  598. fp.write("Generated from %s\n"%ascii(fname))
  599. fp.write("AETE/AEUT resource version %d/%d, language %d, script %d\n" % \
  600. (major, minor, language, script))
  601. fp.write('"""\n\n')
  602. fp.write('import aetools\n')
  603. fp.write('import MacOS\n\n')
  604. fp.write("_code = %r\n\n"% (code,))
  605. if self.basepackage and code in self.basepackage._code_to_module:
  606. # We are an extension of a baseclass (usually an application extending
  607. # Standard_Suite or so). Import everything from our base module
  608. fp.write('from %s import *\n'%self.basepackage._code_to_fullname[code][0])
  609. basemodule = self.basepackage._code_to_module[code]
  610. elif self.basepackage and code.lower() in self.basepackage._code_to_module:
  611. # This is needed by CodeWarrior and some others.
  612. fp.write('from %s import *\n'%self.basepackage._code_to_fullname[code.lower()][0])
  613. basemodule = self.basepackage._code_to_module[code.lower()]
  614. else:
  615. # We are not an extension.
  616. basemodule = None
  617. self.basemodule = basemodule
  618. self.compileclassheader()
  619. self.enumsneeded = {}
  620. if events:
  621. for event in events:
  622. self.compileevent(event)
  623. else:
  624. fp.write(" pass\n\n")
  625. objc = ObjectCompiler(fp, self.modname, basemodule, precompinfo, interact=(self.edit_modnames is None),
  626. verbose=self.verbose)
  627. for cls in classes:
  628. objc.compileclass(cls)
  629. for cls in classes:
  630. objc.fillclasspropsandelems(cls)
  631. for comp in comps:
  632. objc.compilecomparison(comp)
  633. for enum in enums:
  634. objc.compileenumeration(enum)
  635. for enum in self.enumsneeded.keys():
  636. objc.checkforenum(enum)
  637. objc.dumpindex()
  638. def compileclassheader(self):
  639. """Generate class boilerplate"""
  640. classname = '%s_Events'%self.modname
  641. if self.basemodule:
  642. modshortname = string.split(self.basemodule.__name__, '.')[-1]
  643. baseclassname = '%s_Events'%modshortname
  644. self.fp.write("class %s(%s):\n\n"%(classname, baseclassname))
  645. else:
  646. self.fp.write("class %s:\n\n"%classname)
  647. def compileevent(self, event):
  648. """Generate code for a single event"""
  649. [name, desc, code, subcode, returns, accepts, arguments] = event
  650. fp = self.fp
  651. funcname = identify(name)
  652. #
  653. # generate name->keyword map
  654. #
  655. if arguments:
  656. fp.write(" _argmap_%s = {\n"%funcname)
  657. for a in arguments:
  658. fp.write(" %r : %r,\n"%(identify(a[0]), a[1]))
  659. fp.write(" }\n\n")
  660. #
  661. # Generate function header
  662. #
  663. has_arg = (not is_null(accepts))
  664. opt_arg = (has_arg and is_optional(accepts))
  665. fp.write(" def %s(self, "%funcname)
  666. if has_arg:
  667. if not opt_arg:
  668. fp.write("_object, ") # Include direct object, if it has one
  669. else:
  670. fp.write("_object=None, ") # Also include if it is optional
  671. else:
  672. fp.write("_no_object=None, ") # For argument checking
  673. fp.write("_attributes={}, **_arguments):\n") # include attribute dict and args
  674. #
  675. # Generate doc string (important, since it may be the only
  676. # available documentation, due to our name-remaping)
  677. #
  678. fp.write(' """%s: %s\n'%(ascii(name), ascii(desc)))
  679. if has_arg:
  680. fp.write(" Required argument: %s\n"%getdatadoc(accepts))
  681. elif opt_arg:
  682. fp.write(" Optional argument: %s\n"%getdatadoc(accepts))
  683. for arg in arguments:
  684. fp.write(" Keyword argument %s: %s\n"%(identify(arg[0]),
  685. getdatadoc(arg[2])))
  686. fp.write(" Keyword argument _attributes: AppleEvent attribute dictionary\n")
  687. if not is_null(returns):
  688. fp.write(" Returns: %s\n"%getdatadoc(returns))
  689. fp.write(' """\n')
  690. #
  691. # Fiddle the args so everything ends up in 'arguments' dictionary
  692. #
  693. fp.write(" _code = %r\n"% (code,))
  694. fp.write(" _subcode = %r\n\n"% (subcode,))
  695. #
  696. # Do keyword name substitution
  697. #
  698. if arguments:
  699. fp.write(" aetools.keysubst(_arguments, self._argmap_%s)\n"%funcname)
  700. else:
  701. fp.write(" if _arguments: raise TypeError, 'No optional args expected'\n")
  702. #
  703. # Stuff required arg (if there is one) into arguments
  704. #
  705. if has_arg:
  706. fp.write(" _arguments['----'] = _object\n")
  707. elif opt_arg:
  708. fp.write(" if _object:\n")
  709. fp.write(" _arguments['----'] = _object\n")
  710. else:
  711. fp.write(" if _no_object is not None: raise TypeError, 'No direct arg expected'\n")
  712. fp.write("\n")
  713. #
  714. # Do enum-name substitution
  715. #
  716. for a in arguments:
  717. if is_enum(a[2]):
  718. kname = a[1]
  719. ename = a[2][0]
  720. if ename != '****':
  721. fp.write(" aetools.enumsubst(_arguments, %r, _Enum_%s)\n" %
  722. (kname, identify(ename)))
  723. self.enumsneeded[ename] = 1
  724. fp.write("\n")
  725. #
  726. # Do the transaction
  727. #
  728. fp.write(" _reply, _arguments, _attributes = self.send(_code, _subcode,\n")
  729. fp.write(" _arguments, _attributes)\n")
  730. #
  731. # Error handling
  732. #
  733. fp.write(" if _arguments.get('errn', 0):\n")
  734. fp.write(" raise aetools.Error, aetools.decodeerror(_arguments)\n")
  735. fp.write(" # XXXX Optionally decode result\n")
  736. #
  737. # Decode result
  738. #
  739. fp.write(" if '----' in _arguments:\n")
  740. if is_enum(returns):
  741. fp.write(" # XXXX Should do enum remapping here...\n")
  742. fp.write(" return _arguments['----']\n")
  743. fp.write("\n")
  744. def findenumsinevent(self, event):
  745. """Find all enums for a single event"""
  746. [name, desc, code, subcode, returns, accepts, arguments] = event
  747. for a in arguments:
  748. if is_enum(a[2]):
  749. ename = a[2][0]
  750. if ename != '****':
  751. self.enumsneeded[ename] = 1
  752. #
  753. # This class stores the code<->name translations for a single module. It is used
  754. # to keep the information while we're compiling the module, but we also keep these objects
  755. # around so if one suite refers to, say, an enum in another suite we know where to
  756. # find it. Finally, if we really can't find a code, the user can add modules by
  757. # hand.
  758. #
  759. class CodeNameMapper:
  760. def __init__(self, interact=1, verbose=None):
  761. self.code2name = {
  762. "property" : {},
  763. "class" : {},
  764. "enum" : {},
  765. "comparison" : {},
  766. }
  767. self.name2code = {
  768. "property" : {},
  769. "class" : {},
  770. "enum" : {},
  771. "comparison" : {},
  772. }
  773. self.modulename = None
  774. self.star_imported = 0
  775. self.can_interact = interact
  776. self.verbose = verbose
  777. def addnamecode(self, type, name, code):
  778. self.name2code[type][name] = code
  779. if code not in self.code2name[type]:
  780. self.code2name[type][code] = name
  781. def hasname(self, name):
  782. for dict in self.name2code.values():
  783. if name in dict:
  784. return True
  785. return False
  786. def hascode(self, type, code):
  787. return code in self.code2name[type]
  788. def findcodename(self, type, code):
  789. if not self.hascode(type, code):
  790. return None, None, None
  791. name = self.code2name[type][code]
  792. if self.modulename and not self.star_imported:
  793. qualname = '%s.%s'%(self.modulename, name)
  794. else:
  795. qualname = name
  796. return name, qualname, self.modulename
  797. def getall(self, type):
  798. return self.code2name[type].items()
  799. def addmodule(self, module, name, star_imported):
  800. self.modulename = name
  801. self.star_imported = star_imported
  802. for code, name in module._propdeclarations.items():
  803. self.addnamecode('property', name, code)
  804. for code, name in module._classdeclarations.items():
  805. self.addnamecode('class', name, code)
  806. for code in module._enumdeclarations.keys():
  807. self.addnamecode('enum', '_Enum_'+identify(code), code)
  808. for code, name in module._compdeclarations.items():
  809. self.addnamecode('comparison', name, code)
  810. def prepareforexport(self, name=None):
  811. if not self.modulename:
  812. self.modulename = name
  813. return self
  814. class ObjectCompiler:
  815. def __init__(self, fp, modname, basesuite, othernamemappers=None, interact=1,
  816. verbose=None):
  817. self.fp = fp
  818. self.verbose = verbose
  819. self.basesuite = basesuite
  820. self.can_interact = interact
  821. self.modulename = modname
  822. self.namemappers = [CodeNameMapper(self.can_interact, self.verbose)]
  823. if othernamemappers:
  824. self.othernamemappers = othernamemappers[:]
  825. else:
  826. self.othernamemappers = []
  827. if basesuite:
  828. basemapper = CodeNameMapper(self.can_interact, self.verbose)
  829. basemapper.addmodule(basesuite, '', 1)
  830. self.namemappers.append(basemapper)
  831. def getprecompinfo(self, modname):
  832. list = []
  833. for mapper in self.namemappers:
  834. emapper = mapper.prepareforexport(modname)
  835. if emapper:
  836. list.append(emapper)
  837. return list
  838. def findcodename(self, type, code):
  839. while 1:
  840. # First try: check whether we already know about this code.
  841. for mapper in self.namemappers:
  842. if mapper.hascode(type, code):
  843. return mapper.findcodename(type, code)
  844. # Second try: maybe one of the other modules knows about it.
  845. for mapper in self.othernamemappers:
  846. if mapper.hascode(type, code):
  847. self.othernamemappers.remove(mapper)
  848. self.namemappers.append(mapper)
  849. if self.fp:
  850. self.fp.write("import %s\n"%mapper.modulename)
  851. break
  852. else:
  853. # If all this has failed we ask the user for a guess on where it could
  854. # be and retry.
  855. if self.fp:
  856. m = self.askdefinitionmodule(type, code)
  857. else:
  858. m = None
  859. if not m: return None, None, None
  860. mapper = CodeNameMapper(self.can_interact, self.verbose)
  861. mapper.addmodule(m, m.__name__, 0)
  862. self.namemappers.append(mapper)
  863. def hasname(self, name):
  864. for mapper in self.othernamemappers:
  865. if mapper.hasname(name) and mapper.modulename != self.modulename:
  866. if self.verbose:
  867. print >>self.verbose, "Duplicate Python identifier:", name, self.modulename, mapper.modulename
  868. return True
  869. return False
  870. def askdefinitionmodule(self, type, code):
  871. if not self.can_interact:
  872. if self.verbose:
  873. print >>self.verbose, "** No definition for %s '%s' found" % (type, code)
  874. return None
  875. path = EasyDialogs.AskFileForSave(message='Where is %s %s declared?'%(type, code))
  876. if not path: return
  877. path, file = os.path.split(path)
  878. modname = os.path.splitext(file)[0]
  879. if not path in sys.path:
  880. sys.path.insert(0, path)
  881. m = __import__(modname)
  882. self.fp.write("import %s\n"%modname)
  883. return m
  884. def compileclass(self, cls):
  885. [name, code, desc, properties, elements] = cls
  886. pname = identify(name)
  887. if self.namemappers[0].hascode('class', code):
  888. # plural forms and such
  889. othername, dummy, dummy = self.namemappers[0].findcodename('class', code)
  890. if self.fp:
  891. self.fp.write("\n%s = %s\n"%(pname, othername))
  892. else:
  893. if self.fp:
  894. self.fp.write('\nclass %s(aetools.ComponentItem):\n' % pname)
  895. self.fp.write(' """%s - %s """\n' % (ascii(name), ascii(desc)))
  896. self.fp.write(' want = %r\n' % (code,))
  897. self.namemappers[0].addnamecode('class', pname, code)
  898. is_application_class = (code == 'capp')
  899. properties.sort()
  900. for prop in properties:
  901. self.compileproperty(prop, is_application_class)
  902. elements.sort()
  903. for elem in elements:
  904. self.compileelement(elem)
  905. def compileproperty(self, prop, is_application_class=False):
  906. [name, code, what] = prop
  907. if code == 'c@#!':
  908. # Something silly with plurals. Skip it.
  909. return
  910. pname = identify(name)
  911. if self.namemappers[0].hascode('property', code):
  912. # plural forms and such
  913. othername, dummy, dummy = self.namemappers[0].findcodename('property', code)
  914. if pname == othername:
  915. return
  916. if self.fp:
  917. self.fp.write("\n_Prop_%s = _Prop_%s\n"%(pname, othername))
  918. else:
  919. if self.fp:
  920. self.fp.write("class _Prop_%s(aetools.NProperty):\n" % pname)
  921. self.fp.write(' """%s - %s """\n' % (ascii(name), ascii(what[1])))
  922. self.fp.write(" which = %r\n" % (code,))
  923. self.fp.write(" want = %r\n" % (what[0],))
  924. self.namemappers[0].addnamecode('property', pname, code)
  925. if is_application_class and self.fp:
  926. self.fp.write("%s = _Prop_%s()\n" % (pname, pname))
  927. def compileelement(self, elem):
  928. [code, keyform] = elem
  929. if self.fp:
  930. self.fp.write("# element %r as %s\n" % (code, keyform))
  931. def fillclasspropsandelems(self, cls):
  932. [name, code, desc, properties, elements] = cls
  933. cname = identify(name)
  934. if self.namemappers[0].hascode('class', code) and \
  935. self.namemappers[0].findcodename('class', code)[0] != cname:
  936. # This is an other name (plural or so) for something else. Skip.
  937. if self.fp and (elements or len(properties) > 1 or (len(properties) == 1 and
  938. properties[0][1] != 'c@#!')):
  939. if self.verbose:
  940. print >>self.verbose, '** Skip multiple %s of %s (code %r)' % (cname, self.namemappers[0].findcodename('class', code)[0], code)
  941. raise RuntimeError, "About to skip non-empty class"
  942. return
  943. plist = []
  944. elist = []
  945. superclasses = []
  946. for prop in properties:
  947. [pname, pcode, what] = prop
  948. if pcode == "c@#^":
  949. superclasses.append(what)
  950. if pcode == 'c@#!':
  951. continue
  952. pname = identify(pname)
  953. plist.append(pname)
  954. superclassnames = []
  955. for superclass in superclasses:
  956. superId, superDesc, dummy = superclass
  957. superclassname, fullyqualifiedname, module = self.findcodename("class", superId)
  958. # I don't think this is correct:
  959. if superclassname == cname:
  960. pass # superclassnames.append(fullyqualifiedname)
  961. else:
  962. superclassnames.append(superclassname)
  963. if self.fp:
  964. self.fp.write("%s._superclassnames = %r\n"%(cname, superclassnames))
  965. for elem in elements:
  966. [ecode, keyform] = elem
  967. if ecode == 'c@#!':
  968. continue
  969. name, ename, module = self.findcodename('class', ecode)
  970. if not name:
  971. if self.fp:
  972. self.fp.write("# XXXX %s element %r not found!!\n"%(cname, ecode))
  973. else:
  974. elist.append((name, ename))
  975. plist.sort()
  976. elist.sort()
  977. if self.fp:
  978. self.fp.write("%s._privpropdict = {\n"%cname)
  979. for n in plist:
  980. self.fp.write(" '%s' : _Prop_%s,\n"%(n, n))
  981. self.fp.write("}\n")
  982. self.fp.write("%s._privelemdict = {\n"%cname)
  983. for n, fulln in elist:
  984. self.fp.write(" '%s' : %s,\n"%(n, fulln))
  985. self.fp.write("}\n")
  986. def compilecomparison(self, comp):
  987. [name, code, comment] = comp
  988. iname = identify(name)
  989. self.namemappers[0].addnamecode('comparison', iname, code)
  990. if self.fp:
  991. self.fp.write("class %s(aetools.NComparison):\n" % iname)
  992. self.fp.write(' """%s - %s """\n' % (ascii(name), ascii(comment)))
  993. def compileenumeration(self, enum):
  994. [code, items] = enum
  995. name = "_Enum_%s" % identify(code)
  996. if self.fp:
  997. self.fp.write("%s = {\n" % name)
  998. for item in items:
  999. self.compileenumerator(item)
  1000. self.fp.write("}\n\n")
  1001. self.namemappers[0].addnamecode('enum', name, code)
  1002. return code
  1003. def compileenumerator(self, item):
  1004. [name, code, desc] = item
  1005. self.fp.write(" %r : %r,\t# %s\n" % (identify(name), code, ascii(desc)))
  1006. def checkforenum(self, enum):
  1007. """This enum code is used by an event. Make sure it's available"""
  1008. name, fullname, module = self.findcodename('enum', enum)
  1009. if not name:
  1010. if self.fp:
  1011. self.fp.write("_Enum_%s = None # XXXX enum %s not found!!\n"%(identify(enum), ascii(enum)))
  1012. return
  1013. if module:
  1014. if self.fp:
  1015. self.fp.write("from %s import %s\n"%(module, name))
  1016. def dumpindex(self):
  1017. if not self.fp:
  1018. return
  1019. self.fp.write("\n#\n# Indices of types declared in this module\n#\n")
  1020. self.fp.write("_classdeclarations = {\n")
  1021. classlist = self.namemappers[0].getall('class')
  1022. classlist.sort()
  1023. for k, v in classlist:
  1024. self.fp.write(" %r : %s,\n" % (k, v))
  1025. self.fp.write("}\n")
  1026. self.fp.write("\n_propdeclarations = {\n")
  1027. proplist = self.namemappers[0].getall('property')
  1028. proplist.sort()
  1029. for k, v in proplist:
  1030. self.fp.write(" %r : _Prop_%s,\n" % (k, v))
  1031. self.fp.write("}\n")
  1032. self.fp.write("\n_compdeclarations = {\n")
  1033. complist = self.namemappers[0].getall('comparison')
  1034. complist.sort()
  1035. for k, v in complist:
  1036. self.fp.write(" %r : %s,\n" % (k, v))
  1037. self.fp.write("}\n")
  1038. self.fp.write("\n_enumdeclarations = {\n")
  1039. enumlist = self.namemappers[0].getall('enum')
  1040. enumlist.sort()
  1041. for k, v in enumlist:
  1042. self.fp.write(" %r : %s,\n" % (k, v))
  1043. self.fp.write("}\n")
  1044. def compiledata(data):
  1045. [type, description, flags] = data
  1046. return "%r -- %r %s" % (type, description, compiledataflags(flags))
  1047. def is_null(data):
  1048. return data[0] == 'null'
  1049. def is_optional(data):
  1050. return (data[2] & 0x8000)
  1051. def is_enum(data):
  1052. return (data[2] & 0x2000)
  1053. def getdatadoc(data):
  1054. [type, descr, flags] = data
  1055. if descr:
  1056. return ascii(descr)
  1057. if type == '****':
  1058. return 'anything'
  1059. if type == 'obj ':
  1060. return 'an AE object reference'
  1061. return "undocumented, typecode %r"%(type,)
  1062. dataflagdict = {15: "optional", 14: "list", 13: "enum", 12: "mutable"}
  1063. def compiledataflags(flags):
  1064. bits = []
  1065. for i in range(16):
  1066. if flags & (1<<i):
  1067. if i in dataflagdict.keys():
  1068. bits.append(dataflagdict[i])
  1069. else:
  1070. bits.append(repr(i))
  1071. return '[%s]' % string.join(bits)
  1072. def ascii(str):
  1073. """Return a string with all non-ascii characters hex-encoded"""
  1074. if type(str) != type(''):
  1075. return map(ascii, str)
  1076. rv = ''
  1077. for c in str:
  1078. if c in ('\t', '\n', '\r') or ' ' <= c < chr(0x7f):
  1079. rv = rv + c
  1080. else:
  1081. rv = rv + '\\' + 'x%02.2x' % ord(c)
  1082. return rv
  1083. def identify(str):
  1084. """Turn any string into an identifier:
  1085. - replace space by _
  1086. - replace other illegal chars by _xx_ (hex code)
  1087. - append _ if the result is a python keyword
  1088. """
  1089. if not str:
  1090. return "empty_ae_name_"
  1091. rv = ''
  1092. ok = string.ascii_letters + '_'
  1093. ok2 = ok + string.digits
  1094. for c in str:
  1095. if c in ok:
  1096. rv = rv + c
  1097. elif c == ' ':
  1098. rv = rv + '_'
  1099. else:
  1100. rv = rv + '_%02.2x_'%ord(c)
  1101. ok = ok2
  1102. if keyword.iskeyword(rv):
  1103. rv = rv + '_'
  1104. return rv
  1105. # Call the main program
  1106. if __name__ == '__main__':
  1107. main()
  1108. sys.exit(1)