installpanda.py 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332
  1. #!/usr/bin/env python
  2. ########################################################################
  3. #
  4. # To install panda using this script, type 'installpanda.py'.
  5. # To specify an alternate location than the filesystem root /,
  6. # either pass it as only argument or set the DESTDIR environment
  7. # variable. This script only functions on Linux, for now.
  8. #
  9. ########################################################################
  10. import os, sys, platform
  11. from distutils.sysconfig import get_python_lib
  12. from optparse import OptionParser
  13. from makepandacore import *
  14. MIME_INFO = (
  15. ("egg", "model/x-egg", "EGG model file", "pview"),
  16. ("bam", "model/x-bam", "Panda3D binary model file", "pview"),
  17. ("egg.pz", "model/x-compressed-egg", "Compressed EGG model file", "pview"),
  18. ("bam.pz", "model/x-compressed-bam", "Compressed Panda3D binary model file", "pview"),
  19. )
  20. MIME_INFO_PLUGIN = (
  21. ("p3d", "application/x-panda3d", "Panda3D game/applet", "panda3d"),
  22. )
  23. APP_INFO = (
  24. ("pview", "Panda3D Model Viewer", ("egg", "bam", "egg.pz", "bam.pz")),
  25. )
  26. APP_INFO_PLUGIN = (
  27. ("panda3d", "Panda3D", ("p3d")),
  28. )
  29. def WriteApplicationsFile(fname, appinfo, mimeinfo):
  30. fhandle = open(fname, "w")
  31. for app, desc, exts in appinfo:
  32. fhandle.write("%s\n" % (app))
  33. fhandle.write("\tcommand=%s\n" % (app))
  34. fhandle.write("\tname=%s\n" % (desc))
  35. fhandle.write("\tcan_open_multiple_files=true\n")
  36. fhandle.write("\texpects_uris=false\n")
  37. fhandle.write("\trequires_terminal=false\n")
  38. fhandle.write("\tmime_types=")
  39. first = True
  40. for ext, mime, desc2, app2 in mimeinfo:
  41. if ext in exts:
  42. if first:
  43. fhandle.write(mime)
  44. first = False
  45. else:
  46. fhandle.write("," + mime)
  47. fhandle.write("\n\n")
  48. fhandle.close()
  49. def WriteMimeXMLFile(fname, info):
  50. fhandle = open(fname, "w")
  51. fhandle.write("<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n")
  52. fhandle.write("<mime-info xmlns=\"http://www.freedesktop.org/standards/shared-mime-info\">\n")
  53. for ext, mime, desc, app in info:
  54. fhandle.write("\t<mime-type type=\"%s\">\n" % (mime))
  55. fhandle.write("\t\t<comment xml:lang=\"en\">%s</comment>\n" % (desc))
  56. fhandle.write("\t\t<glob pattern=\"*.%s\"/>\n" % (ext))
  57. fhandle.write("\t</mime-type>\n")
  58. fhandle.write("</mime-info>\n")
  59. fhandle.close()
  60. def WriteMimeFile(fname, info):
  61. fhandle = open(fname, "w")
  62. for ext, mime, desc, app in info:
  63. fhandle.write("%s:\n" % (mime))
  64. if "." in ext:
  65. fhandle.write("\tregex,2: %s$\n" % (ext.replace(".", "\\.")))
  66. fhandle.write("\text: %s\n" % (ext))
  67. fhandle.write("\n")
  68. fhandle.close()
  69. def WriteKeysFile(fname, info):
  70. fhandle = open(fname, "w")
  71. for ext, mime, desc, app in info:
  72. fhandle.write("%s:\n" % (mime))
  73. fhandle.write("\tdescription=%s\n" % (desc))
  74. fhandle.write("\tdefault_action_type=application\n")
  75. fhandle.write("\tshort_list_application_ids_for_novice_user_level=%s\n" % (app))
  76. fhandle.write("\topen=%s %%f\n" % (app))
  77. fhandle.write("\tview=%s %%f\n" % (app))
  78. fhandle.write("\n")
  79. fhandle.close()
  80. def GetDebLibDir():
  81. """ Returns the lib dir according to the debian system. """
  82. # We're on Debian or Ubuntu, which use multiarch directories.
  83. # Call dpkg-architecture to get the multiarch libdir.
  84. handle = os.popen("dpkg-architecture -qDEB_HOST_MULTIARCH")
  85. multiarch = handle.read().strip()
  86. if handle.close():
  87. # It failed. Old Debian/Ubuntu version?
  88. pass
  89. elif len(multiarch) > 0:
  90. return "lib/" + multiarch
  91. return "lib"
  92. def GetRPMLibDir():
  93. """ Returns the lib dir according to the rpm system. """
  94. handle = os.popen("rpm -E '%_lib'")
  95. result = handle.read().strip()
  96. handle.close()
  97. if len(result) > 0:
  98. assert result == "lib64" or result == "lib"
  99. return result
  100. else:
  101. return "lib"
  102. def GetLibDir():
  103. """ Returns the directory to install architecture-dependent
  104. libraries in, relative to the prefix directory. This may be
  105. something like "lib" or "lib64" or in some cases, something
  106. similar to "lib/x86_64-linux-gnu". """
  107. if sys.platform in ("darwin", "win32", "cygwin"):
  108. return "lib"
  109. # This one's a bit tricky. Some systems require us to install
  110. # 64-bits libraries into /usr/lib64, some into /usr/lib.
  111. # Debian forbids installing to lib64 nowadays, and the only distros
  112. # I know of that use lib64 are all RPM-based. So, the 'solution'
  113. # seems to be to use the rpm command to give us the libdir for now,
  114. # unless we know we're on debian, since rpm may be installed on
  115. # Debian and will give the wrong result. Ugh.
  116. if os.environ.get("DEB_HOST_MULTIARCH"):
  117. # We're building inside the Debian build environment.
  118. return "lib/" + os.environ["DEB_HOST_MULTIARCH"]
  119. if os.path.isfile('/etc/debian_version'):
  120. return GetDebLibDir()
  121. elif os.path.isfile('/etc/arch-release'):
  122. # ArchLinux has lib64, but it is a symlink to lib.
  123. return "lib"
  124. else:
  125. # Okay, maybe we're on an RPM-based system?
  126. return GetRPMLibDir()
  127. # If Python is installed into /usr/lib64, it's probably safe
  128. # to assume that we should install there as well.
  129. python_lib = get_python_lib(1)
  130. if python_lib.startswith('/usr/lib64/') or \
  131. python_lib.startswith('/usr/local/lib64/'):
  132. return "lib64"
  133. return "lib"
  134. def InstallPanda(destdir="", prefix="/usr", outputdir="built", libdir=GetLibDir(), python_versions=[]):
  135. if (not prefix.startswith("/")):
  136. prefix = "/" + prefix
  137. libdir = prefix + "/" + libdir
  138. # Create the directory structure that we will be putting our files in.
  139. # Don't use os.makedirs or mkdir -p; neither properly set permissions for
  140. # created intermediate directories.
  141. MakeDirectory(destdir+prefix+"/bin", mode=0o755, recursive=True)
  142. MakeDirectory(destdir+prefix+"/include", mode=0o755)
  143. MakeDirectory(destdir+prefix+"/include/panda3d", mode=0o755)
  144. MakeDirectory(destdir+prefix+"/share", mode=0o755)
  145. MakeDirectory(destdir+prefix+"/share/panda3d", mode=0o755)
  146. MakeDirectory(destdir+prefix+"/share/mime-info", mode=0o755)
  147. MakeDirectory(destdir+prefix+"/share/mime", mode=0o755)
  148. MakeDirectory(destdir+prefix+"/share/mime/packages", mode=0o755)
  149. MakeDirectory(destdir+prefix+"/share/application-registry", mode=0o755)
  150. MakeDirectory(destdir+prefix+"/share/applications", mode=0o755)
  151. MakeDirectory(destdir+libdir+"/panda3d", mode=0o755, recursive=True)
  152. for python_version in python_versions:
  153. MakeDirectory(destdir+python_version["purelib"], mode=0o755, recursive=True)
  154. MakeDirectory(destdir+python_version["platlib"]+"/panda3d", mode=0o755, recursive=True)
  155. if (sys.platform.startswith("freebsd")):
  156. MakeDirectory(destdir+prefix+"/etc", mode=0o755)
  157. MakeDirectory(destdir+"/usr/local/libdata/ldconfig", mode=0o755, recursive=True)
  158. else:
  159. MakeDirectory(destdir+"/etc/ld.so.conf.d", mode=0o755, recursive=True)
  160. # Write the Config.prc file.
  161. Configrc = ReadFile(outputdir+"/etc/Config.prc")
  162. Configrc = Configrc.replace("model-path $THIS_PRC_DIR/..", "model-path "+prefix+"/share/panda3d")
  163. if (sys.platform.startswith("freebsd")):
  164. WriteFile(destdir+prefix+"/etc/Config.prc", Configrc)
  165. oscmd("cp "+outputdir+"/etc/Confauto.prc "+destdir+prefix+"/etc/Confauto.prc")
  166. else:
  167. WriteFile(destdir+"/etc/Config.prc", Configrc)
  168. oscmd("cp "+outputdir+"/etc/Confauto.prc "+destdir+"/etc/Confauto.prc")
  169. oscmd("cp -R "+outputdir+"/include/* "+destdir+prefix+"/include/panda3d/")
  170. oscmd("cp -R "+outputdir+"/pandac "+destdir+prefix+"/share/panda3d/")
  171. oscmd("cp -R "+outputdir+"/models "+destdir+prefix+"/share/panda3d/")
  172. if os.path.isdir("samples"): oscmd("cp -R samples "+destdir+prefix+"/share/panda3d/")
  173. if os.path.isdir(outputdir+"/direct"): oscmd("cp -R "+outputdir+"/direct "+destdir+prefix+"/share/panda3d/")
  174. if os.path.isdir(outputdir+"/Pmw"): oscmd("cp -R "+outputdir+"/Pmw "+destdir+prefix+"/share/panda3d/")
  175. if os.path.isdir(outputdir+"/plugins"): oscmd("cp -R "+outputdir+"/plugins "+destdir+prefix+"/share/panda3d/")
  176. for python_version in python_versions:
  177. for base in os.listdir(outputdir + "/panda3d"):
  178. suffix = python_version["ext_suffix"]
  179. platlib = python_version["platlib"]
  180. if base.endswith(".py") or (base.endswith(suffix) and '.' not in base[:-len(suffix)]):
  181. oscmd("cp "+outputdir+"/panda3d/"+base+" "+destdir+platlib+"/panda3d/"+base)
  182. WriteMimeFile(destdir+prefix+"/share/mime-info/panda3d.mime", MIME_INFO)
  183. WriteKeysFile(destdir+prefix+"/share/mime-info/panda3d.keys", MIME_INFO)
  184. WriteMimeXMLFile(destdir+prefix+"/share/mime/packages/panda3d.xml", MIME_INFO)
  185. WriteApplicationsFile(destdir+prefix+"/share/application-registry/panda3d.applications", APP_INFO, MIME_INFO)
  186. if os.path.isfile(outputdir+"/bin/pview"):
  187. oscmd("cp makepanda/pview.desktop "+destdir+prefix+"/share/applications/pview.desktop")
  188. oscmd("cp doc/ReleaseNotes "+destdir+prefix+"/share/panda3d/ReleaseNotes")
  189. for python_version in python_versions:
  190. pth_file = python_version["purelib"] + "/panda3d.pth"
  191. oscmd("echo '"+prefix+"/share/panda3d' > "+destdir+pth_file)
  192. if os.path.isdir(outputdir+"/panda3d.dist-info"):
  193. oscmd("cp -R "+outputdir+"/panda3d.dist-info "+destdir+python_version["platlib"])
  194. if (sys.platform.startswith("freebsd")):
  195. oscmd("echo '"+libdir+"/panda3d'> "+destdir+"/usr/local/libdata/ldconfig/panda3d")
  196. else:
  197. oscmd("echo '"+libdir+"/panda3d'> "+destdir+"/etc/ld.so.conf.d/panda3d.conf")
  198. for base in os.listdir(outputdir+"/lib"):
  199. if (not base.endswith(".a")) or base == "libp3pystub.a":
  200. # We really need to specify -R in order not to follow symlinks on non-GNU
  201. oscmd("cp -R -P "+outputdir+"/lib/"+base+" "+destdir+libdir+"/panda3d/"+base)
  202. for base in os.listdir(outputdir+"/bin"):
  203. if not base.startswith("deploy-stub"):
  204. oscmd("cp -R -P "+outputdir+"/bin/"+base+" "+destdir+prefix+"/bin/"+base)
  205. DeleteVCS(destdir+prefix+"/share/panda3d")
  206. DeleteBuildFiles(destdir+prefix+"/share/panda3d")
  207. DeleteEmptyDirs(destdir+prefix+"/share/panda3d")
  208. DeleteVCS(destdir+prefix+"/include/panda3d")
  209. DeleteBuildFiles(destdir+prefix+"/include/panda3d")
  210. DeleteEmptyDirs(destdir+prefix+"/include/panda3d")
  211. # Change permissions on include directory.
  212. os.chmod(destdir + prefix + "/include/panda3d", 0o755)
  213. for root, dirs, files in os.walk(destdir + prefix + "/include/panda3d"):
  214. for basename in dirs:
  215. os.chmod(os.path.join(root, basename), 0o755)
  216. for basename in files:
  217. os.chmod(os.path.join(root, basename), 0o644)
  218. # rpmlint doesn't like this file, for some reason.
  219. if (os.path.isfile(destdir+prefix+"/share/panda3d/direct/leveleditor/copyfiles.pl")):
  220. os.remove(destdir+prefix+"/share/panda3d/direct/leveleditor/copyfiles.pl")
  221. def InstallRuntime(destdir="", prefix="/usr", outputdir="built", libdir=GetLibDir()):
  222. if (not prefix.startswith("/")):
  223. prefix = "/" + prefix
  224. libdir = prefix + "/" + libdir
  225. oscmd("mkdir -m 0755 -p "+destdir+prefix+"/bin")
  226. oscmd("mkdir -m 0755 -p "+destdir+prefix+"/share/mime-info")
  227. oscmd("mkdir -m 0755 -p "+destdir+prefix+"/share/mime/packages")
  228. oscmd("mkdir -m 0755 -p "+destdir+prefix+"/share/application-registry")
  229. oscmd("mkdir -m 0755 -p "+destdir+prefix+"/share/applications")
  230. if (os.path.exists(outputdir+"/plugins/nppanda3d.so")):
  231. oscmd("mkdir -m 0755 -p "+destdir+libdir)
  232. oscmd("cp "+outputdir+"/plugins/nppanda3d.so "+destdir+libdir+"/nppanda3d.so")
  233. if sys.platform.startswith("freebsd"):
  234. oscmd("mkdir -m 0755 -p "+destdir+libdir+"/browser_plugins/symlinks/gecko19")
  235. oscmd("mkdir -m 0755 -p "+destdir+libdir+"/libxul/plugins")
  236. oscmd("ln -f -s "+libdir+"/nppanda3d.so "+destdir+libdir+"/browser_plugins/symlinks/gecko19/nppanda3d.so")
  237. oscmd("ln -f -s "+libdir+"/nppanda3d.so "+destdir+libdir+"/libxul/plugins/nppanda3d.so")
  238. else:
  239. oscmd("mkdir -m 0755 -p "+destdir+libdir+"/mozilla/plugins")
  240. oscmd("mkdir -m 0755 -p "+destdir+libdir+"/mozilla-firefox/plugins")
  241. oscmd("mkdir -m 0755 -p "+destdir+libdir+"/xulrunner-addons/plugins")
  242. oscmd("ln -f -s "+libdir+"/nppanda3d.so "+destdir+libdir+"/mozilla/plugins/nppanda3d.so")
  243. oscmd("ln -f -s "+libdir+"/nppanda3d.so "+destdir+libdir+"/mozilla-firefox/plugins/nppanda3d.so")
  244. oscmd("ln -f -s "+libdir+"/nppanda3d.so "+destdir+libdir+"/xulrunner-addons/plugins/nppanda3d.so")
  245. WriteMimeFile(destdir+prefix+"/share/mime-info/panda3d-runtime.mime", MIME_INFO_PLUGIN)
  246. WriteKeysFile(destdir+prefix+"/share/mime-info/panda3d-runtime.keys", MIME_INFO_PLUGIN)
  247. WriteMimeXMLFile(destdir+prefix+"/share/mime/packages/panda3d-runtime.xml", MIME_INFO_PLUGIN)
  248. WriteApplicationsFile(destdir+prefix+"/share/application-registry/panda3d-runtime.applications", APP_INFO_PLUGIN, MIME_INFO_PLUGIN)
  249. oscmd("cp makepanda/panda3d.desktop "+destdir+prefix+"/share/applications/panda3d.desktop")
  250. oscmd("cp "+outputdir+"/bin/panda3d "+destdir+prefix+"/bin/")
  251. if (__name__ == "__main__"):
  252. if (sys.platform.startswith("win") or sys.platform == "darwin"):
  253. exit("This script is not supported on Windows or Mac OS X at the moment!")
  254. destdir = os.environ.get("DESTDIR", "/")
  255. parser = OptionParser()
  256. parser.add_option('', '--outputdir', dest = 'outputdir', help = 'Makepanda\'s output directory (default: built)', default = 'built')
  257. parser.add_option('', '--destdir', dest = 'destdir', help = 'Destination directory [default=%s]' % destdir, default = destdir)
  258. parser.add_option('', '--prefix', dest = 'prefix', help = 'Prefix [default=/usr/local]', default = '/usr/local')
  259. parser.add_option('', '--runtime', dest = 'runtime', help = 'Specify if runtime build [default=no]', action = 'store_true', default = False)
  260. parser.add_option('', '--verbose', dest = 'verbose', help = 'Print commands that are executed [default=no]', action = 'store_true', default = False)
  261. (options, args) = parser.parse_args()
  262. destdir = options.destdir
  263. if (destdir.endswith("/")):
  264. destdir = destdir[:-1]
  265. if (destdir == "/"):
  266. destdir = ""
  267. if (destdir != "" and not os.path.isdir(destdir)):
  268. exit("Directory '%s' does not exist!" % destdir)
  269. SetOutputDir(options.outputdir)
  270. if options.verbose:
  271. SetVerbose(True)
  272. if (options.runtime):
  273. print("Installing Panda3D Runtime into " + destdir + options.prefix)
  274. InstallRuntime(destdir = destdir, prefix = options.prefix, outputdir = options.outputdir)
  275. else:
  276. print("Installing Panda3D SDK into " + destdir + options.prefix)
  277. InstallPanda(destdir=destdir,
  278. prefix=options.prefix,
  279. outputdir=options.outputdir,
  280. python_versions=ReadPythonVersionInfoFile())
  281. print("Installation finished!")
  282. if not destdir:
  283. warn_prefix = "%sNote:%s " % (GetColor("red"), GetColor())
  284. print(warn_prefix + "You may need to call this command to update the library cache:")
  285. print(" sudo ldconfig")