import subprocess
# 셸 명령어를 실행하여 'ls -l | grep ".txt"' 결과를 출력
= subprocess.run('ls -l | grep ".txt"', shell=True, capture_output=True, text=True)
result print(result.stdout)
-rw-r--r-- 1 runner docker 6533 Mar 31 12:08 output.txt
Python shell명령 수행하기 (Subprocess)
gabriel yang
August 30, 2024
Python의 subprocess
모듈은 파이썬 코드 내에서 외부 셸 명령을 실행할 수 있는 강력한 도구입니다. 이 모듈은 간단한 명령 실행부터 복잡한 상호작용이 필요한 명령어까지 다양한 작업을 수행할 수 있습니다.
subprocess
모듈 소개subprocess
모듈은 Python 3에서 외부 명령을 실행하기 위해 사용되는 표준 라이브러리입니다. 이 모듈을 사용하면 파이썬 코드 내에서 셸 명령어를 실행하고, 그 결과를 가져와 처리할 수 있습니다. os.system()
와 같은 이전 방식에 비해 더 강력하고 유연한 기능을 제공합니다.
가장 간단한 형태로, subprocess.run()
함수를 사용하여 명령어를 실행할 수 있습니다. 이 함수는 명령어를 실행한 후, subprocess.CompletedProcess
객체를 반환합니다.
import subprocess
# 'ls' 명령어를 실행하고 결과를 화면에 출력
result = subprocess.run(['ls', './images'], capture_output=True, text=True)
print(f"stdout: {result.stdout}")
stdout: paste-1.png
위 코드에서는 ls
명령어를 실행하여 현재 디렉토리의 파일 목록을 출력합니다. capture_output=True
는 표준 출력(stdout)과 표준 에러(stderr)를 캡처하도록 설정하며, text=True
는 출력을 텍스트로 반환하도록 합니다.
cd
명령을 사용할 수 없습니다.subprocess는 새로운 프로세스를 생성하여 시스템 명령어나 외부 프로그램을 실행하는 데 사용됩니다. 시스템 명령어는 일반적으로 실행 파일로 존재하는 명령어입니다.
반면, cd는 단순한 실행 파일이 아니고, 셸 자체의 상태를 변경하는 명령이기 때문에 시스템에서 독립적으로 실행할 수 없습니다.
따라서 cd 명령어를 subprocess를 통해 실행하더라도, 작업 디렉토리를 변경하려는 시도가 성공하지 않습니다.
작업 디렉토리를 변경하려면 os 모듈의 os라이브러리의 os.chdir()
함수를 사용해야 합니다.
subprocess
로 복잡한 명령어 실행때로는 명령어를 셸에서 실행해야 할 필요가 있습니다. subprocess.run()
에서 shell=True
옵션을 사용하면 명령어를 셸에서 실행할 수 있습니다.
import subprocess
# 셸 명령어를 실행하여 'ls -l | grep ".txt"' 결과를 출력
result = subprocess.run('ls -l | grep ".txt"', shell=True, capture_output=True, text=True)
print(result.stdout)
-rw-r--r-- 1 runner docker 6533 Mar 31 12:08 output.txt
이 예제에서는 ls -l
명령어의 결과를 grep
명령어로 필터링하여 .txt
파일만 출력합니다. shell=True
를 사용하여 복잡한 셸 명령어를 실행할 수 있습니다.
명령어의 결과를 변수에 저장하고, 이를 파이썬 코드에서 처리할 수 있습니다. subprocess.run()
의 capture_output=True
옵션을 사용하면 명령어의 표준 출력과 표준 에러를 캡처할 수 있습니다.
이 예제에서는 ls
명령어의 출력을 output.txt
파일에 저장합니다. stdout
매개변수를 파일 객체 f
로 지정하여 출력을 파일에 기록할 수 있습니다.
subprocess.run()
이 반환하는 subprocess.CompletedProcess
객체를 통해 명령어의 반환 코드, 표준 출력, 표준 에러 등을 확인할 수 있습니다.
명령어 실행이 성공했는지 여부를 확인하기 위해 반환 코드를 확인할 수 있습니다. 반환 코드가 0이면 성공, 0이 아니면 실패를 의미합니다.
import subprocess
result = subprocess.run(['ls', 'non_existing_file'], capture_output=True, text=True)
print(f"Return code: {result.returncode}")
if result.returncode != 0:
print("Command failed")
Return code: 2
Command failed
위 코드에서는 존재하지 않는 파일을 조회하려고 시도하여, 명령어가 실패했음을 확인할 수 있습니다.
# 'ls' 명령어를 실행하고 결과를 화면에 출력
result = subprocess.run(['ls', './없는폴더명'], capture_output=True, text=True)
print(f"stdout: {result.stdout}")
print(f"stderr: {result.stderr}")
stdout:
stderr: ls: cannot access './없는폴더명': No such file or directory
위의 코드에서는 존재하지 않는 폴더에 대해서 ls
명령을 수행해서 error가 발생했습니다. 에러는 result.stderr
로 정보가 전달됩니다.
파이프라인 명령어를 구현하려면 subprocess.PIPE
를 사용하여 여러 명령어의 출력을 다음 명령어의 입력으로 전달할 수 있습니다.
import subprocess
# 'ls -l' 명령어를 실행하고, 그 출력을 'grep ".txt"'에 전달
p1 = subprocess.Popen(['ls', '-l'], stdout=subprocess.PIPE)
p2 = subprocess.Popen(['grep', '.txt'], stdin=p1.stdout, stdout=subprocess.PIPE)
p1.stdout.close() # 첫 번째 프로세스의 출력을 닫음
output, error = p2.communicate()
print(output.decode())
-rw-r--r-- 1 runner docker 5527 Mar 31 12:13 output.txt
이 예제에서는 ls -l
명령어의 결과를 grep
명령어로 필터링하여 .py
파일만 출력합니다. subprocess.Popen
을 사용하여 프로세스를 생성하고, 파이프를 통해 데이터를 전달합니다.
p1 = subprocess.Popen(\['ls', '-l'\], stdout=subprocess.PIPE)
: p1
의 stdout=subprocess.PIPE
는 ls -l 명령어의 출력을 파이프로 보내도록 설정합니다
p2 = subprocess.Popen(\['grep', '.txt'\], stdin=p1.stdout, stdout=subprocess.PIPE)
: 이 파이프는 p2
프로세스로 전달되며 stdin=p1.stdout
은 첫 번째 프로세스(ls -l)의 출력을 grep 명령어의 입력으로 사용하도록 설정합니다.
p1.stdout.close()
: 첫 번째 프로세스(p1)의 출력 스트림을 닫습니다. 이는 grep 명령어가 첫 번째 프로세스의 출력을 다 받았음을 알려주기 위함입니다. 이렇게 해야 grep 명령어가 더 이상 입력을 기다리지 않고 종료할 수 있습니다.
-output, error = p2.communicate()
: communicate() 메서드는 프로세스가 끝날 때까지 기다린 후, 출력과 에러를 한꺼번에 수집하는 데 사용됩니다. 프로세스의 출력(stdout)과 에러(stderr)는 각각 output과 error에 저장됩니다.
비동기적으로 명령어를 실행해야 할 때는 subprocess.Popen
을 사용합니다. 이 방법을 사용하면 명령어 실행 중에도 다른 작업을 수행할 수 있습니다.
import subprocess
# 'sleep 5' 명령어를 비동기적으로 실행
process = subprocess.Popen(['sleep', '5'])
# 다른 작업을 수행
print("Waiting for the command to complete...")
# 명령어가 완료될 때까지 기다림
process.wait()
print("Command completed after 5 secs")
Waiting for the command to complete...
Command completed after 5 secs
이 코드는 sleep 5
명령어를 실행한 후, 명령어 실행이 완료될 때까지 기다리지 않고 다른 작업을 수행합니다.
<h3>카테고리 다른 글</h3>
Date | Title | Author |
---|---|---|
Jan 1, 3000 | 전체 카테고리 | gabriel yang |
Nov 26, 2024 | VSCode에서 Python 디버깅 (launch.json 설정) | gabriel yang |
Nov 23, 2024 | Python 설치된 패키지 확인, 설치와 복원하기 | gabriel yang |
Oct 14, 2024 | JSON 파일에서 특정 key의 값 변경 후 저장하는 방법 | gabriel yang |
Oct 10, 2024 | 날짜 문자열을 날짜 형식으로 변환하기 | gabriel yang |
Oct 8, 2024 | Request를 통한 JIRA REST API 사용방법 | gabriel yang |
Oct 8, 2024 | Request 라이브러리 사용방법 | gabriel yang |
Oct 8, 2024 | Requests 라이브러리로 베이직 인증 API 호출하기 | gabriel yang |
Oct 7, 2024 | Python으로 SSH를 제어하기 | gabriel yang |
Oct 7, 2024 | Pytest로 테스트하기 | gabriel yang |
Oct 7, 2024 | Pytest 기본적인 테스트 실행 방법 | gabriel yang |
Oct 7, 2024 | Pytest Fixture 개념과 필요성 | gabriel yang |
Oct 7, 2024 | CI 환경에서 Pytest 사용하기 | gabriel yang |
Oct 7, 2024 | 파이썬에서 테스트 코드 리팩토링 기법 | gabriel yang |
Oct 5, 2024 | Python os 모듈을 이용한 파일과 폴더 관리 | gabriel yang |
Oct 5, 2024 | Python shutil 모듈을 이용한 파일과 폴더 관리 | gabriel yang |
Oct 5, 2024 | MongoDB 데이터베이스 백업 및 복원 방법 | gabriel yang |
Oct 5, 2024 | Python jira 라이브러리를 이용해 JIRA를 관리하는 방법 | gabriel yang |
Oct 4, 2024 | Python에서 JSON 파일 읽기 및 데이터 활용 | gabriel yang |
Oct 4, 2024 | JSON 데이터 수정하기 | gabriel yang |
Oct 4, 2024 | JSON 파일의 구성요소와 구조 | gabriel yang |
Oct 4, 2024 | Python에서 Git명령 사용하기 | gabriel yang |
Oct 3, 2024 | 파이썬으로 클래스 정의하는 방법 | gabriel yang |
Oct 3, 2024 | 파이썬의 상속(Inheritance) 이해하기 | gabriel yang |
Oct 3, 2024 | 파이썬 클래스의 self 이해하기 | gabriel yang |
Oct 3, 2024 | 파이썬에서 추상 클래스(Abstract Class) 사용하기 | gabriel yang |
Oct 3, 2024 | 파이썬 클래스 활용 예시 | gabriel yang |
Oct 3, 2024 | 파이썬 클래스와 모듈 | gabriel yang |
Oct 3, 2024 | 파이썬에서 Lock을 사용하는 이유와 방법 | gabriel yang |
Oct 2, 2024 | Python에서 zip()을 사용하는 이유와 방법 | gabriel yang |
Oct 2, 2024 | 파이썬 시퀀스 슬라이싱 | gabriel yang |
Oct 2, 2024 | 파이썬에서 Generator를 사용하는 이유와 사용법 | gabriel yang |
Oct 2, 2024 | 파이썬의 @property 기능 | gabriel yang |
Oct 2, 2024 | 파이썬 시퀀스 언패킹 | gabriel yang |
Oct 2, 2024 | 파이썬 데이터 정렬 | gabriel yang |
Oct 2, 2024 | 파이썬 딕셔너리 key 예외처리하기 | gabriel yang |
Oct 2, 2024 | 파이썬의 defaultdict 사용법 | gabriel yang |
Oct 2, 2024 | 파이썬의 try-except 사용법 | gabriel yang |
Oct 2, 2024 | 파이썬의 클로저(Clsure) 사용법 | gabriel yang |
Oct 2, 2024 | 파이썬의 가변인자 사용법 | gabriel yang |
Oct 2, 2024 | 파이썬의 컴프리헨션(Comprehension) 사용법 | gabriel yang |
Oct 1, 2024 | 파이썬 가상환경 설정 | gabriel yang |
Oct 1, 2024 | 파이썬을 이용한 테스트 자동화 | gabriel yang |
Oct 1, 2024 | 파이썬 docstring을 사용하는 이유와 방법 | gabriel yang |
Oct 1, 2024 |
파이썬 yield 제너레이터와 효율적인 반복 처리
|
gabriel yang |
Oct 1, 2024 | 파이썬 데이터 언패킹 | gabriel yang |
Oct 1, 2024 | Python에서 enumerate를 사용하는 이유와 사용 방법 | gabriel yang |
Sep 30, 2024 | Python 설치 및 기본 파이썬 버전 설정 | gabriel yang |
Sep 30, 2024 | Python을 이용한 폴더 전체 복사 방법 | gabriel yang |
Sep 30, 2024 | Public Key와 Private Key의 역할 | gabriel yang |
Sep 23, 2024 | Python의 @dataclass 데코레이터 | gabriel yang |
Aug 29, 2024 | Google Colab에서 라이브러리 설치하기 | gabriel yang |
Aug 1, 2024 | 환경변수 설정하고 Python에서 읽어오기 | gabriel yang |
Jan 1, 2024 | 코루틴(coroutine)과 이벤트 루프 | gabriel yang |