Programing

현재 스크립트 디렉토리를 올바르게 결정하는 방법은 무엇입니까?

lottogame 2020. 4. 19. 13:52
반응형

현재 스크립트 디렉토리를 올바르게 결정하는 방법은 무엇입니까?


파이썬에서 현재 스크립트 디렉토리를 결정하는 가장 좋은 방법은 무엇입니까?

파이썬 코드를 호출하는 많은 방법으로 인해 좋은 해결책을 찾기가 어렵다는 것을 알았습니다.

몇 가지 문제가 있습니다 :

  • __file__로 스크립트를 실행하면 정의되지 않습니다 exec.execfile
  • __module__ 모듈에서만 정의됩니다

사용 사례:

  • ./myfile.py
  • python myfile.py
  • ./somedir/myfile.py
  • python somedir/myfile.py
  • execfile('myfile.py') (다른 스크립트에서 다른 디렉토리에 있으며 다른 현재 디렉토리를 가질 수 있습니다.

완벽한 솔루션이 없다는 것을 알고 있지만 대부분의 경우를 해결하는 최선의 방법을 찾고 있습니다.

가장 많이 사용되는 방법 os.path.dirname(os.path.abspath(__file__))이지만 다른 스크립트를 사용하여 스크립트를 실행하면 실제로 작동하지 않습니다 exec().

경고

현재 디렉토리를 사용하는 모든 솔루션이 실패합니다. 이는 스크립트가 호출되는 방식에 따라 다를 수 있거나 실행중인 스크립트 내에서 변경 될 수 있습니다.


os.path.dirname(os.path.abspath(__file__))

실제로 당신이 얻는 최고의 것입니다.

exec/ execfile; 로 스크립트를 실행하는 것은 일반적이지 않습니다 . 일반적으로 모듈 인프라를 사용하여 스크립트를로드해야합니다. 이러한 방법을 사용해야하는 경우, 내가 설정하는 것이 좋습니다 __file__에서 globals하면 해당 파일 이름을 읽을 수 있도록 스크립트에 전달합니다.

실행 코드에서 파일 이름을 얻는 다른 방법은 없습니다. CWD가 완전히 다른 위치에있을 수 있습니다.


를 통해 스크립트가 호출되는 경우를 실제로 다루려면 모듈을 execfile(...)사용 inspect하여 파일 이름 (경로 포함)을 추론 할 수 있습니다 . 내가 아는 한, 이것은 나열된 모든 경우에 적용됩니다.

filename = inspect.getframeinfo(inspect.currentframe()).filename
path = os.path.dirname(os.path.abspath(filename))

#!/usr/bin/env python
import inspect
import os
import sys

def get_script_dir(follow_symlinks=True):
    if getattr(sys, 'frozen', False): # py2exe, PyInstaller, cx_Freeze
        path = os.path.abspath(sys.executable)
    else:
        path = inspect.getabsfile(get_script_dir)
    if follow_symlinks:
        path = os.path.realpath(path)
    return os.path.dirname(path)

print(get_script_dir())

CPython, Jython, Pypy에서 작동합니다. 스크립트를 사용하여 스크립트를 실행하면 작동합니다 execfile()( sys.argv[0]그리고 __file__기반 솔루션은 여기서 실패합니다). 스크립트가 실행 가능한 zip 파일 (/ eg) 안에 있으면 작동합니다 . 스크립트가 PYTHONPATH=/path/to/library.zip python -mscript_to_runzip 파일에서 "가져온"( ) 경우 작동 합니다. 이 경우 아카이브 경로를 반환합니다. 스크립트가 독립 실행 형 실행 파일 ( sys.frozen) 로 컴파일 된 경우 작동합니다 . 심볼릭 링크에서 작동합니다 ( realpath심볼릭 링크 제거). 대화식 인터프리터에서 작동합니다. 이 경우 현재 작업 디렉토리를 반환합니다.


Python 3.4 이상에서는 더 간단한 pathlib모듈을 사용할 수 있습니다 .

from inspect import currentframe, getframeinfo
from pathlib import Path

filename = getframeinfo(currentframe()).filename
parent = Path(filename).resolve().parent

os.path...방법은 파이썬 2의 '할 일'입니다.

Python 3에서는 다음과 같이 스크립트 디렉토리를 찾을 수 있습니다.

from pathlib import Path
cwd = Path(__file__).parents[0]

사용 os.path.dirname(os.path.abspath(__file__))되는 경우에 대한 실제 필요성이 있는지 여부를 매우 신중하게 사용 하고 조사하십시오 exec. 스크립트를 모듈로 사용할 수없는 경우 문제가 발생할 수 있습니다.

Zen of Python # 8 을 명심하십시오 . 유용 사례에 대한 좋은 주장이 있다고 생각 exec되면 문제의 배경에 대해 더 자세히 알려주십시오.


할 것이다

import os
cwd = os.getcwd()

당신이 원하는 것을합니까? "현재 스크립트 디렉토리"가 정확히 무엇을 의미하는지 잘 모르겠습니다. 제공 한 사용 사례에 대해 예상되는 결과는 무엇입니까?


import os
import sys

def get_script_path():
    return os.path.dirname(os.path.realpath(sys.argv[0]))


my_script_dir = get_script_path()
print my_script_dir

이것은 스택의 최상위에 스크립트의 디렉토리를 제공합니다 (즉, 파이썬이 아닌 실행중인 디렉토리 C:/).


먼저 익명의 코드를 삽입하는 방법에 대해 이야기하고 있다면 유스 케이스가 누락 된 것입니다.

code.compile_command()
code.interact()
imp.load_compiled()
imp.load_dynamic()
imp.load_module()
__builtin__.compile()
loading C compiled shared objects? example: _socket?)

그러나 진짜 질문은, 당신의 목표는 무엇입니까-당신은 일종의 보안을 시행하려고합니까? 또는 당신은 무엇을로드하는 것에 관심이 있습니까?

보안에 관심이있는 경우 exec / execfile을 통해 가져 오는 파일 이름은 중요하지 않습니다. rexec 를 사용해야 합니다. 다음을 제공합니다.

이 모듈에는 r_eval (), r_execfile (), r_exec () 및 r_import () 메소드를 지원하는 RExec 클래스가 포함되어 있습니다.이 메소드는 표준 Python 함수 eval (), execfile () 및 exec 및 import 문의 제한된 버전입니다. 이 제한된 환경에서 실행되는 코드는 안전하다고 간주되는 모듈과 기능에만 액세스 할 수 있습니다. RExec 추가 기능을 원하는대로 서브 클래 싱 할 수 있습니다.

그러나 이것이 학문적 추구에 더 가깝다면 여기에 좀 더 깊이 파고들 수있는 몇 가지 구피 접근법이 있습니다.

스크립트 예 :

./deep.py

print ' >> level 1'
execfile('deeper.py')
print ' << level 1'

./deeper.py

print '\t >> level 2'
exec("import sys; sys.path.append('/tmp'); import deepest")
print '\t << level 2'

/tmp/deepest.py

print '\t\t >> level 3'
print '\t\t\t I can see the earths core.'
print '\t\t << level 3'

./codespy.py

import sys, os

def overseer(frame, event, arg):
    print "loaded(%s)" % os.path.abspath(frame.f_code.co_filename)

sys.settrace(overseer)
execfile("deep.py")
sys.exit(0)

산출

loaded(/Users/synthesizerpatel/deep.py)
>> level 1
loaded(/Users/synthesizerpatel/deeper.py)
    >> level 2
loaded(/Users/synthesizerpatel/<string>)
loaded(/tmp/deepest.py)
        >> level 3
            I can see the earths core.
        << level 3
    << level 2
<< level 1

물론 이것은 리소스를 많이 사용하는 방법이므로 모든 코드를 추적해야합니다.별로 효율적이지 않습니다. 그러나 둥지에 깊숙이 들어가도 계속 작동하기 때문에 새로운 접근법이라고 생각합니다. 'eval'을 무시할 수 없습니다. 당신이 비록 execfile ()을 재정의합니다.

이 접근법은 '가져 오기'가 아닌 exec / exec 파일에만 적용됩니다. 더 높은 수준의 '모듈'로드 후크의 경우 sys.path_hooks (PyMOTW의 쓰기 제공) 를 사용할 수 있습니다 .

그게 내가 머리 꼭대기에서 벗어난 전부 야


여기에 지금까지 게시 된 모든 솔루션보다 여전히 나은 부분 솔루션이 있습니다.

import sys, os, os.path, inspect

#os.chdir("..")

if '__file__' not in locals():
    __file__ = inspect.getframeinfo(inspect.currentframe())[0]

print os.path.dirname(os.path.abspath(__file__))

이제는 모든 호출이 작동하지만 누군가 chdir()현재 디렉토리를 변경하는 데 사용 하면 실패합니다.

메모:

  • sys.argv[0]작동하지 않을 것입니다 -c. 스크립트를 실행하면python -c "execfile('path-tester.py')"
  • https://gist.github.com/1385555 에서 완전한 테스트를 게시했으며 이를 개선 할 수 있습니다.

대부분의 경우 작동합니다.

import os,sys
dirname=os.path.dirname(os.path.realpath(sys.argv[0]))

희망적으로 이것은 도움이됩니다 :-어디에서나 스크립트 / 모듈을 실행하면 스크립트 __file__의 위치를 ​​나타내는 모듈 변수 인 변수 에 액세스 할 수 있습니다 .

인터프리터를 사용하는 경우 반면에, 당신은 당신의 이름을 얻을 것이다 그 변수에 대한 액세스 권한이없는 NameError그리고 os.getcwd()당신이 다른 곳에서 파일을 실행하는 경우 당신에게 잘못된 디렉토리를 줄 것이다.

솔루션은 모든 경우에 원하는 것을 제공합니다.

from inspect import getsourcefile
from os.path import abspath
abspath(getsourcefile(lambda:0))

나는 그것을 철저히 테스트하지는 않았지만 내 문제를 해결했습니다.


단지 pwd
jupyter 노트북에서 : pwd+ Shift + 입력
스파이더에서 : pwd+ F9

참고 URL : https://stackoverflow.com/questions/3718657/how-to-properly-determine-current-script-directory

반응형