mirror of
https://github.com/python/cpython.git
synced 2024-12-21 15:53:43 +08:00
481 lines
13 KiB
C
481 lines
13 KiB
C
/***********************************************************
|
|
Copyright 1991-1997 by Stichting Mathematisch Centrum, Amsterdam,
|
|
The Netherlands.
|
|
|
|
All Rights Reserved
|
|
|
|
Permission to use, copy, modify, and distribute this software and its
|
|
documentation for any purpose and without fee is hereby granted,
|
|
provided that the above copyright notice appear in all copies and that
|
|
both that copyright notice and this permission notice appear in
|
|
supporting documentation, and that the names of Stichting Mathematisch
|
|
Centrum or CWI or Corporation for National Research Initiatives or
|
|
CNRI not be used in advertising or publicity pertaining to
|
|
distribution of the software without specific, written prior
|
|
permission.
|
|
|
|
While CWI is the initial source for this software, a modified version
|
|
is made available by the Corporation for National Research Initiatives
|
|
(CNRI) at the Internet address ftp://ftp.python.org.
|
|
|
|
STICHTING MATHEMATISCH CENTRUM AND CNRI DISCLAIM ALL WARRANTIES WITH
|
|
REGARD TO THIS SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF
|
|
MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL STICHTING MATHEMATISCH
|
|
CENTRUM OR CNRI BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL
|
|
DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR
|
|
PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
|
|
TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
|
|
PERFORMANCE OF THIS SOFTWARE.
|
|
|
|
******************************************************************/
|
|
|
|
#include "Python.h"
|
|
#include "osdefs.h"
|
|
#include "macglue.h"
|
|
#include "macdefs.h"
|
|
#include "pythonresources.h"
|
|
#ifdef HAVE_UNISTD_H
|
|
#include <unistd.h>
|
|
#endif
|
|
|
|
#ifdef TARGET_API_MAC_OSX
|
|
#define PATHNAMELEN 1024
|
|
#else
|
|
#define PATHNAMELEN 256
|
|
#endif
|
|
|
|
/* Return the initial python search path. This is called once from
|
|
** initsys() to initialize sys.path.
|
|
**
|
|
** If USE_BUILTIN_PATH is defined the path defined here is used
|
|
** (after prepending the python home dir to each item).
|
|
** If it is not defined the path is gotten from a resource in the
|
|
** Preferences file.
|
|
**
|
|
** XXXX This code needs cleaning up. The routines here have moved
|
|
** around quite a bit, and they're pretty messy for that reason.
|
|
*/
|
|
|
|
#include <Files.h>
|
|
#include <Aliases.h>
|
|
#include <Folders.h>
|
|
#include <Resources.h>
|
|
#include <TextUtils.h>
|
|
#include <Dialogs.h>
|
|
|
|
#ifdef USE_GUSI1
|
|
#include <GUSI.h>
|
|
#endif
|
|
|
|
#ifndef USE_BUILTIN_PATH
|
|
staticforward char *PyMac_GetPythonPath();
|
|
#endif
|
|
|
|
#define PYTHONPATH "\
|
|
:\n\
|
|
:Lib\n\
|
|
:Lib:stdwin\n\
|
|
:Lib:test\n\
|
|
:Lib:mac"
|
|
|
|
static int
|
|
getpreffilefss(FSSpec *fssp)
|
|
{
|
|
static int diditbefore=0;
|
|
static int rv = 1;
|
|
static FSSpec fss;
|
|
short prefdirRefNum;
|
|
long prefdirDirID;
|
|
long pyprefdirDirID;
|
|
Handle namehandle;
|
|
OSErr err;
|
|
|
|
if ( !diditbefore ) {
|
|
if ( (namehandle=GetNamedResource('STR ', PREFFILENAME_NAME)) == NULL ) {
|
|
(void)StopAlert(NOPREFNAME_ID, NULL);
|
|
exit(1);
|
|
}
|
|
|
|
if ( **namehandle == '\0' ) {
|
|
/* Empty string means don't use preferences file */
|
|
rv = 0;
|
|
} else {
|
|
/* There is a filename, construct the fsspec */
|
|
if ( FindFolder(kOnSystemDisk, 'pref', kDontCreateFolder, &prefdirRefNum,
|
|
&prefdirDirID) != noErr ) {
|
|
/* Something wrong with preferences folder */
|
|
(void)StopAlert(NOPREFDIR_ID, NULL);
|
|
exit(1);
|
|
}
|
|
/* make fsspec for the "Python" folder inside the prefs folder */
|
|
err = FSMakeFSSpec(prefdirRefNum, prefdirDirID, "\pPython", &fss);
|
|
if (err == fnfErr) {
|
|
/* it doesn't exist: create it */
|
|
err = FSpDirCreate(&fss, smSystemScript, &pyprefdirDirID);
|
|
} else {
|
|
/* it does exist, now find out the dirID of the Python prefs folder, brrr. */
|
|
CInfoPBRec info;
|
|
info.dirInfo.ioVRefNum = fss.vRefNum;
|
|
info.dirInfo.ioDrDirID = fss.parID;
|
|
info.dirInfo.ioNamePtr = fss.name;
|
|
info.dirInfo.ioFDirIndex = 0;
|
|
info.dirInfo.ioACUser = 0;
|
|
err = PBGetCatInfo(&info, 0);
|
|
if (err == noErr) {
|
|
pyprefdirDirID = info.dirInfo.ioDrDirID;
|
|
}
|
|
}
|
|
if (err != noErr) {
|
|
(void)StopAlert(NOPREFDIR_ID, NULL);
|
|
exit(1);
|
|
}
|
|
HLock(namehandle);
|
|
err = FSMakeFSSpec(fss.vRefNum, pyprefdirDirID, (unsigned char *)*namehandle, &fss);
|
|
HUnlock(namehandle);
|
|
if (err != noErr && err != fnfErr) {
|
|
(void)StopAlert(NOPREFDIR_ID, NULL);
|
|
exit(1);
|
|
}
|
|
}
|
|
ReleaseResource(namehandle);
|
|
diditbefore = 1;
|
|
}
|
|
*fssp = fss;
|
|
return rv;
|
|
}
|
|
|
|
char *
|
|
Py_GetPath()
|
|
{
|
|
/* Modified by Jack to do something a bit more sensible:
|
|
** - Prepend the python home-directory (which is obtained from a Preferences
|
|
** resource)
|
|
** - Add :
|
|
*/
|
|
static char *pythonpath;
|
|
char *p, *endp;
|
|
int newlen;
|
|
char *curwd;
|
|
|
|
if ( pythonpath ) return pythonpath;
|
|
#ifndef USE_BUILTIN_PATH
|
|
if ( pythonpath = PyMac_GetPythonPath() )
|
|
return pythonpath;
|
|
printf("Warning: No pythonpath resource found, using builtin default\n");
|
|
#endif
|
|
curwd = PyMac_GetPythonDir();
|
|
p = PYTHONPATH;
|
|
endp = p;
|
|
pythonpath = malloc(2);
|
|
if ( pythonpath == NULL ) return PYTHONPATH;
|
|
strcpy(pythonpath, ":");
|
|
while (*endp) {
|
|
endp = strchr(p, '\n');
|
|
if ( endp == NULL )
|
|
endp = p + strlen(p);
|
|
newlen = strlen(pythonpath) + 1 + strlen(curwd) + (endp-p);
|
|
pythonpath = realloc(pythonpath, newlen+1);
|
|
if ( pythonpath == NULL ) return PYTHONPATH;
|
|
strcat(pythonpath, "\n");
|
|
if ( *p == ':' ) {
|
|
p++;
|
|
strcat(pythonpath, curwd);
|
|
strncat(pythonpath, p, (endp-p));
|
|
newlen--; /* Ok, ok, we've allocated one byte too much */
|
|
} else {
|
|
/* We've allocated too much in this case */
|
|
newlen -= strlen(curwd);
|
|
pythonpath = realloc(pythonpath, newlen+1);
|
|
if ( pythonpath == NULL ) return PYTHONPATH;
|
|
strncat(pythonpath, p, (endp-p));
|
|
}
|
|
pythonpath[newlen] = '\0';
|
|
p = endp + 1;
|
|
}
|
|
return pythonpath;
|
|
}
|
|
|
|
|
|
/*
|
|
** Open/create the Python Preferences file, return the handle
|
|
*/
|
|
short
|
|
PyMac_OpenPrefFile()
|
|
{
|
|
AliasHandle handle;
|
|
FSSpec dirspec;
|
|
short prefrh;
|
|
OSErr err;
|
|
|
|
if ( !getpreffilefss(&dirspec))
|
|
return -1;
|
|
prefrh = FSpOpenResFile(&dirspec, fsRdWrShPerm);
|
|
if ( prefrh < 0 ) {
|
|
#if 0
|
|
action = CautionAlert(NOPREFFILE_ID, NULL);
|
|
if ( action == NOPREFFILE_NO )
|
|
exit(1);
|
|
#endif
|
|
FSpCreateResFile(&dirspec, 'Pyth', 'pref', 0);
|
|
prefrh = FSpOpenResFile(&dirspec, fsRdWrShPerm);
|
|
if ( prefrh == -1 ) {
|
|
/* This "cannot happen":-) */
|
|
printf("Cannot create preferences file, error %d\n", ResError());
|
|
exit(1);
|
|
}
|
|
if ( (err=PyMac_init_process_location()) != 0 ) {
|
|
printf("Cannot get application location, error %d\n", err);
|
|
exit(1);
|
|
}
|
|
dirspec = PyMac_ApplicationFSSpec;
|
|
dirspec.name[0] = 0;
|
|
if ((err=NewAlias(NULL, &dirspec, &handle)) != 0 ) {
|
|
printf("Cannot make alias to application directory, error %d\n", err);
|
|
exit(1);
|
|
}
|
|
AddResource((Handle)handle, 'alis', PYTHONHOME_ID, "\p");
|
|
UpdateResFile(prefrh);
|
|
|
|
} else {
|
|
UseResFile(prefrh);
|
|
}
|
|
return prefrh;
|
|
}
|
|
|
|
/*
|
|
** Return the name of the Python directory
|
|
*/
|
|
char *
|
|
PyMac_GetPythonDir()
|
|
{
|
|
static int diditbefore = 0;
|
|
static char name[PATHNAMELEN] = {':', '\0'};
|
|
AliasHandle handle;
|
|
FSSpec dirspec;
|
|
Boolean modified = 0;
|
|
short oldrh, prefrh = -1, homerh;
|
|
|
|
if ( diditbefore )
|
|
return name;
|
|
|
|
oldrh = CurResFile();
|
|
|
|
/* First look for an override in the application file */
|
|
UseResFile(PyMac_AppRefNum);
|
|
handle = (AliasHandle)Get1Resource('alis', PYTHONHOMEOVERRIDE_ID);
|
|
UseResFile(oldrh);
|
|
if ( handle != NULL ) {
|
|
homerh = PyMac_AppRefNum;
|
|
} else {
|
|
/* Try to open preferences file in the preferences folder. */
|
|
prefrh = PyMac_OpenPrefFile();
|
|
handle = (AliasHandle)Get1Resource('alis', PYTHONHOME_ID);
|
|
if ( handle == NULL ) {
|
|
/* (void)StopAlert(BADPREFFILE_ID, NULL); */
|
|
diditbefore=1;
|
|
return ":";
|
|
}
|
|
homerh = prefrh;
|
|
}
|
|
/* It exists. Resolve it (possibly updating it) */
|
|
if ( ResolveAlias(NULL, handle, &dirspec, &modified) != noErr ) {
|
|
(void)StopAlert(BADPREFFILE_ID, NULL);
|
|
diditbefore=1;
|
|
return ":";
|
|
}
|
|
if ( modified ) {
|
|
ChangedResource((Handle)handle);
|
|
UpdateResFile(homerh);
|
|
}
|
|
if ( prefrh != -1 ) CloseResFile(prefrh);
|
|
UseResFile(oldrh);
|
|
|
|
if ( PyMac_GetFullPathname(&dirspec, name, PATHNAMELEN) == 0 ) {
|
|
strcat(name, ":");
|
|
} else {
|
|
/* If all fails, we return the current directory */
|
|
printf("Python home dir exists but I cannot find the pathname!!\n");
|
|
name[0] = 0;
|
|
(void)getcwd(name, sizeof(name));
|
|
}
|
|
diditbefore = 1;
|
|
return name;
|
|
}
|
|
|
|
#ifndef USE_BUILTIN_PATH
|
|
char *
|
|
PyMac_GetPythonPath(void)
|
|
{
|
|
short oldrh, prefrh = -1;
|
|
char *rv;
|
|
int i, newlen;
|
|
Str255 pathitem;
|
|
int resource_id;
|
|
OSErr err;
|
|
Handle h;
|
|
|
|
oldrh = CurResFile();
|
|
/*
|
|
** This is a bit tricky. We check here whether the application file
|
|
** contains an override. This is to forestall us finding another STR# resource
|
|
** with "our" id and using that for path initialization
|
|
*/
|
|
UseResFile(PyMac_AppRefNum);
|
|
SetResLoad(0);
|
|
if ( (h=Get1Resource('STR#', PYTHONPATHOVERRIDE_ID)) ) {
|
|
ReleaseResource(h);
|
|
resource_id = PYTHONPATHOVERRIDE_ID;
|
|
} else {
|
|
resource_id = PYTHONPATH_ID;
|
|
}
|
|
SetResLoad(1);
|
|
UseResFile(oldrh);
|
|
|
|
/* Open the preferences file only if there is no override */
|
|
if ( resource_id != PYTHONPATHOVERRIDE_ID )
|
|
prefrh = PyMac_OpenPrefFile();
|
|
/* At this point, we may or may not have the preferences file open, and it
|
|
** may or may not contain a sys.path STR# resource. We don't care, if it doesn't
|
|
** exist we use the one from the application (the default).
|
|
** We put an initial '\n' in front of the path that we don't return to the caller
|
|
*/
|
|
if( (rv = malloc(2)) == NULL )
|
|
goto out;
|
|
strcpy(rv, "\n");
|
|
|
|
for(i=1; ; i++) {
|
|
GetIndString(pathitem, resource_id, i);
|
|
if( pathitem[0] == 0 )
|
|
break;
|
|
if ( pathitem[0] >= 9 && strncmp((char *)pathitem+1, "$(PYTHON)", 9) == 0 ) {
|
|
/* We have to put the directory in place */
|
|
char *dir = PyMac_GetPythonDir();
|
|
|
|
newlen = strlen(rv) + strlen(dir) + (pathitem[0]-9) + 2;
|
|
if( (rv=realloc(rv, newlen)) == NULL)
|
|
goto out;
|
|
strcat(rv, dir);
|
|
/* Skip a colon at the beginning of the item */
|
|
if ( pathitem[0] > 9 && pathitem[1+9] == ':' ) {
|
|
memcpy(rv+strlen(rv), pathitem+1+10, pathitem[0]-10);
|
|
newlen--;
|
|
} else {
|
|
memcpy(rv+strlen(rv), pathitem+1+9, pathitem[0]-9);
|
|
}
|
|
rv[newlen-2] = '\n';
|
|
rv[newlen-1] = 0;
|
|
} else if ( pathitem[0] >= 14 && strncmp((char *)pathitem+1, "$(APPLICATION)", 14) == 0 ) {
|
|
/* This is the application itself */
|
|
|
|
if ( (err=PyMac_init_process_location()) != 0 ) {
|
|
printf("Cannot get application location, error %d\n", err);
|
|
exit(1);
|
|
}
|
|
|
|
newlen = strlen(rv) + strlen(PyMac_ApplicationPath) + 2;
|
|
if( (rv=realloc(rv, newlen)) == NULL)
|
|
goto out;
|
|
strcpy(rv+strlen(rv), PyMac_ApplicationPath);
|
|
rv[newlen-2] = '\n';
|
|
rv[newlen-1] = 0;
|
|
|
|
} else {
|
|
/* Use as-is */
|
|
newlen = strlen(rv) + (pathitem[0]) + 2;
|
|
if( (rv=realloc(rv, newlen)) == NULL)
|
|
goto out;
|
|
memcpy(rv+strlen(rv), pathitem+1, pathitem[0]);
|
|
rv[newlen-2] = '\n';
|
|
rv[newlen-1] = 0;
|
|
}
|
|
}
|
|
if( strlen(rv) == 1) {
|
|
free(rv);
|
|
rv = NULL;
|
|
}
|
|
if ( rv ) {
|
|
rv[strlen(rv)-1] = 0;
|
|
rv++;
|
|
}
|
|
out:
|
|
if ( prefrh != -1) CloseResFile(prefrh);
|
|
UseResFile(oldrh);
|
|
return rv;
|
|
}
|
|
#endif /* !USE_BUILTIN_PATH */
|
|
|
|
void
|
|
PyMac_PreferenceOptions(PyMac_PrefRecord *pr)
|
|
{
|
|
short oldrh, prefrh = -1;
|
|
Handle handle;
|
|
int size;
|
|
PyMac_PrefRecord *p;
|
|
int action;
|
|
|
|
|
|
oldrh = CurResFile();
|
|
|
|
/* Attempt to load overrides from application */
|
|
UseResFile(PyMac_AppRefNum);
|
|
handle = Get1Resource('Popt', PYTHONOPTIONSOVERRIDE_ID);
|
|
UseResFile(oldrh);
|
|
|
|
/* Otherwise get options from prefs file or any other open resource file */
|
|
if ( handle == NULL ) {
|
|
prefrh = PyMac_OpenPrefFile();
|
|
handle = GetResource('Popt', PYTHONOPTIONS_ID);
|
|
}
|
|
if ( handle == NULL ) {
|
|
return;
|
|
}
|
|
HLock(handle);
|
|
size = GetHandleSize(handle);
|
|
p = (PyMac_PrefRecord *)*handle;
|
|
if ( p->version == POPT_VERSION_CURRENT && size == sizeof(PyMac_PrefRecord) ) {
|
|
*pr = *p;
|
|
} else {
|
|
action = CautionAlert(BADPREFERENCES_ID, NULL);
|
|
if ( action == BADPREF_DELETE ) {
|
|
OSErr err;
|
|
|
|
RemoveResource(handle);
|
|
if ( (err=ResError()) ) printf("RemoveResource: %d\n", err);
|
|
if ( prefrh != -1 ) {
|
|
UpdateResFile(prefrh);
|
|
if ( (err=ResError()) ) printf("UpdateResFile: %d\n", err);
|
|
}
|
|
} else if ( action == BADPREF_QUIT )
|
|
exit(1);
|
|
}
|
|
HUnlock(handle);
|
|
|
|
if ( prefrh != -1) CloseResFile(prefrh);
|
|
UseResFile(oldrh);
|
|
}
|
|
|
|
#ifdef USE_GUSI1
|
|
void
|
|
PyMac_SetGUSIOptions()
|
|
{
|
|
Handle h;
|
|
short oldrh, prefrh = -1;
|
|
|
|
oldrh = CurResFile();
|
|
|
|
/* Try override from the application resource fork */
|
|
UseResFile(PyMac_AppRefNum);
|
|
h = Get1Resource('GU\267I', GUSIOPTIONSOVERRIDE_ID);
|
|
UseResFile(oldrh);
|
|
|
|
/* If that didn't work try nonoverride from anywhere */
|
|
if ( h == NULL ) {
|
|
prefrh = PyMac_OpenPrefFile();
|
|
h = GetResource('GU\267I', GUSIOPTIONS_ID);
|
|
}
|
|
if ( h ) GUSILoadConfiguration(h);
|
|
if ( prefrh != -1) CloseResFile(prefrh);
|
|
UseResFile(oldrh);
|
|
}
|
|
#endif /* USE_GUSI1 */
|