Browse Source

Merge pull request #43 from branan/python_packages

Python packages
Branan Purvine-Riley 14 years ago
parent
commit
13147affed
  1. 107
      Sources/Plasma/Apps/plPythonPack/main.cpp
  2. 76
      Sources/Plasma/FeatureLib/pfPython/cyPythonInterface.cpp

107
Sources/Plasma/Apps/plPythonPack/main.cpp

@ -30,6 +30,7 @@ You can contact Cyan Worlds, Inc. by email legal@cyan.com
#include <vector> #include <vector>
#include <string> #include <string>
#include <algorithm>
#include <direct.h> #include <direct.h>
@ -37,22 +38,25 @@ static const char* kPackFileName = "python.pak";
static const char* kGlueFile = ".\\plasma\\glue.py"; static const char* kGlueFile = ".\\plasma\\glue.py";
static char* glueFile = (char*)kGlueFile; static char* glueFile = (char*)kGlueFile;
void WritePythonFile(const char *fileName, const char* path, hsStream *s) void WritePythonFile(std::string fileName, std::string path, hsStream *s)
{ {
hsUNIXStream pyStream, glueStream; hsUNIXStream pyStream, glueStream;
char* pathAndFile = new char[strlen(fileName)+strlen(path)+2]; std::string filePath;
strcpy(pathAndFile,path); size_t filestart = fileName.find_last_of('.');
char lastchar = pathAndFile[strlen(pathAndFile)-1]; if(filestart != std::string::npos)
if (lastchar != '\\' && lastchar != '/') filePath = fileName.substr(filestart+1, std::string::npos);
strcat(pathAndFile, "\\"); else
strcat(pathAndFile,fileName); filePath = fileName;
if (!pyStream.Open(pathAndFile) || !glueStream.Open(glueFile)) filePath += ".py";
filePath = path + filePath;
if (!pyStream.Open(filePath.c_str()) || !glueStream.Open(glueFile))
{ {
printf("Unable to open path %s, ",pathAndFile); printf("Unable to open path %s, ",filePath.c_str());
return; return;
} }
printf("==Packing %s, ",fileName); printf("==Packing %s, ",fileName.c_str());
pyStream.FastFwd(); pyStream.FastFwd();
UInt32 pyFileSize = pyStream.GetPosition(); UInt32 pyFileSize = pyStream.GetPosition();
@ -86,37 +90,21 @@ void WritePythonFile(const char *fileName, const char* path, hsStream *s)
// skip the CRs // skip the CRs
} }
PyObject* pythonCode = PythonInterface::CompileString(code, (char*)fileName); // import the module first, to make packages work correctly
PyImport_ImportModule((char*)fileName.c_str());
PyObject* pythonCode = PythonInterface::CompileString(code, (char*)fileName.c_str());
if (pythonCode) if (pythonCode)
{ {
// we need to find out if this is PythonFile module // we need to find out if this is PythonFile module
// create a module name... with the '.' as an X // create a module name... with the '.' as an X
// and create a python file name that is without the ".py" // and create a python file name that is without the ".py"
char* modulename = new char[strlen(fileName)+1]; PyObject* fModule = PythonInterface::CreateModule((char*)fileName.c_str());
char* pythonfilename = new char[strlen(fileName)+1];
int j;
for (j=0; j<strlen(fileName); j++)
{
if (fileName[j] == '.')
{
modulename[j] = 'X';
pythonfilename[j] = '\0';
}
else
{
modulename[j] = fileName[j];
pythonfilename[j] = fileName[j];
}
}
modulename[j] = '\0';
PyObject* fModule = PythonInterface::CreateModule(modulename);
delete [] modulename;
// run the code // run the code
if (PythonInterface::RunPYC(pythonCode, fModule) ) if (PythonInterface::RunPYC(pythonCode, fModule) )
{ {
// set the name of the file (in the global dictionary of the module) // set the name of the file (in the global dictionary of the module)
PyObject* dict = PyModule_GetDict(fModule); PyObject* dict = PyModule_GetDict(fModule);
PyObject* pfilename = PyString_FromString(pythonfilename); PyObject* pfilename = PyString_FromString(fileName.c_str());
PyDict_SetItemString(dict, "glue_name", pfilename); PyDict_SetItemString(dict, "glue_name", pfilename);
// next we need to: // next we need to:
// - create instance of class // - create instance of class
@ -132,20 +120,21 @@ void WritePythonFile(const char *fileName, const char* path, hsStream *s)
{ {
// oops, this is not a PythonFile modifier // oops, this is not a PythonFile modifier
// re-read the source and compile it without the glue code this time // re-read the source and compile it without the glue code this time
memset(code,0,totalSize);
pyStream.Rewind(); pyStream.Rewind();
amountRead = pyStream.Read(pyFileSize, code); amountRead = pyStream.Read(pyFileSize, code);
hsAssert(amountRead == pyFileSize, "Bad read"); hsAssert(amountRead == pyFileSize, "Bad read");
strcat(code,"\n"); code[amountRead] = '\n';
code[amountRead+1] = '\0';
k = 0; k = 0;
for (int i = 0; i < strlen(code)+1; i++) int len = strlen(code)+1;
for (int i = 0; i < len; i++)
{ {
if (code[i] != '\r') // is it not a CR? if (code[i] != '\r') // is it not a CR?
code[k++] = code[i]; code[k++] = code[i];
// else // else
// skip the CRs // skip the CRs
} }
pythonCode = PythonInterface::CompileString(code, (char*)fileName); pythonCode = PythonInterface::CompileString(code, (char*)fileName.c_str());
hsAssert(pythonCode,"Not sure why this didn't compile the second time???"); hsAssert(pythonCode,"Not sure why this didn't compile the second time???");
printf("an import file "); printf("an import file ");
} }
@ -155,7 +144,6 @@ void WritePythonFile(const char *fileName, const char* path, hsStream *s)
else else
{ {
printf("......blast! Error during run-code!\n"); printf("......blast! Error during run-code!\n");
s->WriteSwap32(0);
char* errmsg; char* errmsg;
int chars_read = PythonInterface::getOutputAndReset(&errmsg); int chars_read = PythonInterface::getOutputAndReset(&errmsg);
@ -165,7 +153,6 @@ void WritePythonFile(const char *fileName, const char* path, hsStream *s)
printf("\n"); printf("\n");
} }
} }
delete [] pythonfilename;
} }
// make sure that we have code to save // make sure that we have code to save
@ -204,7 +191,6 @@ void WritePythonFile(const char *fileName, const char* path, hsStream *s)
} }
delete [] code; delete [] code;
delete [] pathAndFile; // all done with the path and filename as one
pyStream.Close(); pyStream.Close();
glueStream.Close(); glueStream.Close();
@ -243,7 +229,7 @@ std::string ToLowerCase(std::string str)
return retVal; return retVal;
} }
void FindSubDirs(std::vector<std::string> &dirnames, std::vector<std::string> &pakNames, char *path) void FindSubDirs(std::vector<std::string> &dirnames, const char *path)
{ {
hsFolderIterator folder; hsFolderIterator folder;
if (path) if (path)
@ -257,7 +243,6 @@ void FindSubDirs(std::vector<std::string> &dirnames, std::vector<std::string> &p
if ((dirName != ".")&&(dirName != "..")&&(ToLowerCase(dirName) != "system") && (ToLowerCase(dirName) != "plasma")) if ((dirName != ".")&&(dirName != "..")&&(ToLowerCase(dirName) != "system") && (ToLowerCase(dirName) != "plasma"))
{ {
dirnames.push_back(dirName); dirnames.push_back(dirName);
pakNames.push_back(dirName+".pak");
} }
} }
} }
@ -315,6 +300,29 @@ std::string ConcatDirs(std::string fullPath, std::string partialPath)
return retVal; return retVal;
} }
void FindPackages(std::vector<std::string>& fileNames, std::vector<std::string>& pathNames, const char* path, std::string parent_package="")
{
std::vector<std::string> packages;
FindSubDirs(packages, path);
for (int i = 0; i < packages.size(); i++)
{
std::string packageName;
if(!parent_package.empty())
packageName = parent_package + ".";
packageName += packages[i];
std::vector<std::string> packageFileNames;
std::vector<std::string> packagePathNames;
std::string packagePath = path;
packagePath += "/" + packages[i];
FindFiles(packageFileNames, packagePathNames, packagePath.c_str());
for (int j = 0; j < packageFileNames.size(); j++) {
fileNames.push_back(packageName+"."+packageFileNames[j]);
pathNames.push_back(packagePathNames[j]+"/");
}
FindPackages(fileNames, pathNames, packagePath.c_str(), packageName);
}
}
void PackDirectory(std::string dir, std::string rootPath, std::string pakName, std::vector<std::string>& extraDirs, bool packSysAndPlasma = false) void PackDirectory(std::string dir, std::string rootPath, std::string pakName, std::vector<std::string>& extraDirs, bool packSysAndPlasma = false)
{ {
// make sure the dir ends in a slash // make sure the dir ends in a slash
@ -335,16 +343,20 @@ void PackDirectory(std::string dir, std::string rootPath, std::string pakName, s
std::vector<std::string> pathNames; std::vector<std::string> pathNames;
FindFiles(fileNames,pathNames,dir.c_str()); FindFiles(fileNames,pathNames,dir.c_str());
FindPackages(fileNames,pathNames,dir.c_str());
if (packSysAndPlasma) if (packSysAndPlasma)
{ {
printf("Adding the system and plasma directories to this pack file\n"); printf("Adding the system and plasma directories to this pack file\n");
std::string tempPath; std::string tempPath;
tempPath = dir + "system"; tempPath = dir + "system/";
FindFiles(fileNames,pathNames,tempPath.c_str()); FindFiles(fileNames,pathNames,tempPath.c_str());
tempPath = dir + "plasma"; FindPackages(fileNames,pathNames,tempPath.c_str());
tempPath = dir + "plasma/";
FindFiles(fileNames,pathNames,tempPath.c_str()); FindFiles(fileNames,pathNames,tempPath.c_str());
FindPackages(fileNames,pathNames,tempPath.c_str());
} }
// ok, we know how many files we're gonna pack, so make a fake index (we'll fill in later) // ok, we know how many files we're gonna pack, so make a fake index (we'll fill in later)
hsUNIXStream s; hsUNIXStream s;
if (!s.Open(pakName.c_str(), "wb")) if (!s.Open(pakName.c_str(), "wb"))
@ -371,8 +383,10 @@ void PackDirectory(std::string dir, std::string rootPath, std::string pakName, s
for (i = 0; i < fileNames.size(); i++) for (i = 0; i < fileNames.size(); i++)
{ {
// strip '.py' from the file name
std::string properFileName = fileNames[i].substr(0, fileNames[i].size()-3);
UInt32 initialPos = s.GetPosition(); UInt32 initialPos = s.GetPosition();
WritePythonFile(fileNames[i].c_str(), pathNames[i].c_str(), &s); WritePythonFile(properFileName, pathNames[i], &s);
UInt32 endPos = s.GetPosition(); UInt32 endPos = s.GetPosition();
filePositions[i] = initialPos; filePositions[i] = initialPos;
@ -426,18 +440,17 @@ void main(int argc, char *argv[])
} }
std::vector<std::string> dirNames; std::vector<std::string> dirNames;
std::vector<std::string> pakNames;
std::string rootPath; std::string rootPath;
if (argc == 1) if (argc == 1)
{ {
FindSubDirs(dirNames,pakNames,nil); FindSubDirs(dirNames,nil);
rootPath = AdjustEndingSlash(baseWorkingDir,true); rootPath = AdjustEndingSlash(baseWorkingDir,true);
} }
else else
{ {
std::string path = argv[1]; std::string path = argv[1];
FindSubDirs(dirNames,pakNames,argv[1]); FindSubDirs(dirNames,argv[1]);
rootPath = ConcatDirs(baseWorkingDir,path); rootPath = ConcatDirs(baseWorkingDir,path);
rootPath = AdjustEndingSlash(rootPath,true); rootPath = AdjustEndingSlash(rootPath,true);
} }
@ -445,6 +458,6 @@ void main(int argc, char *argv[])
PackDirectory(rootPath,rootPath,rootPath+kPackFileName,dirNames,true); PackDirectory(rootPath,rootPath,rootPath+kPackFileName,dirNames,true);
for (int i=0; i<dirNames.size(); i++) for (int i=0; i<dirNames.size(); i++)
{ {
PackDirectory(dirNames[i],rootPath,rootPath+pakNames[i],dirNames); PackDirectory(dirNames[i],rootPath,rootPath+dirNames[i]+".pak",dirNames);
} }
} }

76
Sources/Plasma/FeatureLib/pfPython/cyPythonInterface.cpp

@ -864,7 +864,13 @@ PYTHON_METHOD_DEFINITION(ptImportHook, find_module, args)
if (PyString_Check(module_path)) if (PyString_Check(module_path))
PYTHON_RETURN_NONE; PYTHON_RETURN_NONE;
std::string package_module_name = module_name;
package_module_name += ".__init__";
if (PythonPack::IsItPythonPacked(module_name)) if (PythonPack::IsItPythonPacked(module_name))
{
Py_INCREF(self);
return (PyObject*)self;
} else if (PythonPack::IsItPythonPacked(package_module_name.c_str()))
{ {
Py_INCREF(self); Py_INCREF(self);
return (PyObject*)self; return (PyObject*)self;
@ -873,25 +879,14 @@ PYTHON_METHOD_DEFINITION(ptImportHook, find_module, args)
PYTHON_RETURN_NONE; PYTHON_RETURN_NONE;
} }
PYTHON_METHOD_DEFINITION(ptImportHook, load_module, args) PyObject *ptImportHook_load_module_detail(ptImportHook *self, char* module_name, char* packed_name, bool isPackage, bool& found)
{ {
char* module_name;
if (!PyArg_ParseTuple(args, "s", &module_name))
{
PyErr_SetString(PyExc_TypeError, "load_module expects string");
PYTHON_RETURN_ERROR;
}
// Grab sys.__dict__ so we can get started
PyObject* sys_mod = PyImport_ImportModule("sys");
PyObject* sys_dict = PyModule_GetDict(sys_mod);
// We want to check sys.modules for the module first // We want to check sys.modules for the module first
// If it's not in there, we have to load the module // If it's not in there, we have to load the module
// and add it to the sys.modules dict for future reference, // and add it to the sys.modules dict for future reference,
// otherwise reload() will not work properly. // otherwise reload() will not work properly.
PyObject* result = nil; PyObject* result = nil;
PyObject* modules = PyDict_GetItemString(sys_dict, "modules"); PyObject* modules = PyImport_GetModuleDict();
hsAssert(PyDict_Check(modules), "sys.modules is not a dict"); hsAssert(PyDict_Check(modules), "sys.modules is not a dict");
if (result = PyDict_GetItemString(modules, module_name)) if (result = PyDict_GetItemString(modules, module_name))
@ -905,27 +900,58 @@ PYTHON_METHOD_DEFINITION(ptImportHook, load_module, args)
} }
else else
{ {
if (PyObject* pyc = PythonPack::OpenPythonPacked(module_name)) if (PyObject* pyc = PythonPack::OpenPythonPacked(packed_name))
{ {
result = PyImport_ExecCodeModule(module_name, pyc); result = PyImport_AddModule(module_name);
if (result == nil) if(!result)
{ return nil;
PyErr_Print(); PyObject* d = PyModule_GetDict(result);
PyDict_SetItemString(d, "__builtins__", PyEval_GetBuiltins());
PyObject *file = PyString_FromString(packed_name);
PyModule_AddObject(result, "__file__", file);
PyDict_SetItemString(d, "__loader__", (PyObject*)self);
if(isPackage) {
PyObject *path = PyString_FromString(module_name);
PyObject *l = PyList_New(1);
PyList_SetItem(l, 0, path);
PyDict_SetItemString(d, "__path__", l);
Py_DECREF(l);
} }
else PyObject* v = PyEval_EvalCode((PyCodeObject *)pyc, d, d);
if(!v)
{ {
PyModule_AddObject(result, "__loader__", (PyObject*)self); PyDict_DelItemString(modules, module_name);
PyDict_SetItemString(modules, module_name, result); return nil;
} }
Py_INCREF(result);
} }
else else {
found = false;
PyErr_SetString(PyExc_ImportError, "module not found in python.pak"); PyErr_SetString(PyExc_ImportError, "module not found in python.pak");
}
} }
if (result) return result;
return result; }
else
PYTHON_METHOD_DEFINITION(ptImportHook, load_module, args)
{
char* module_name;
if (!PyArg_ParseTuple(args, "s", &module_name))
{
PyErr_SetString(PyExc_TypeError, "load_module expects string");
PYTHON_RETURN_ERROR; PYTHON_RETURN_ERROR;
}
bool found = true;
PyObject *result = ptImportHook_load_module_detail(self, module_name, module_name, false, found);
if (!found)
{
PyErr_Clear();
std::string package_module_name = module_name;
package_module_name += ".__init__";
result = ptImportHook_load_module_detail(self, module_name, (char*)package_module_name.c_str(), true, found);
}
return result;
} }
PYTHON_START_METHODS_TABLE(ptImportHook) PYTHON_START_METHODS_TABLE(ptImportHook)

Loading…
Cancel
Save