CherryPy-퀵 가이드

CherryPy는 Python 개발자를 위해 HTTP 프로토콜에 친숙한 인터페이스를 제공하는 Python의 웹 프레임 워크입니다. 웹 애플리케이션 라이브러리라고도합니다.

CherryPy는 Python의 장점을 동적 언어로 사용하여 HTTP 프로토콜을 API로 모델링하고 바인딩합니다. 깨끗한 인터페이스와 안정적인 플랫폼을 제공하는 가장 오래된 Python 용 웹 프레임 워크 중 하나입니다.

CherryPy의 역사

Remi Delon은 2002 년 6 월 말에 CherryPy의 첫 번째 버전을 출시했습니다. 이것이 성공적인 Python 웹 라이브러리의 시작점이었습니다. Remi는 웹 애플리케이션 개발을위한 최고의 대안 중 하나로 Python을 신뢰 한 프랑스 해커입니다.

Remi가 개발 한 프로젝트는 접근 방식에 관심이있는 많은 개발자를 매료 시켰습니다. 접근 방식에는 다음과 같은 기능이 포함되었습니다.

  • CherryPy는 모델-뷰-컨트롤러 패턴에 가깝습니다.

  • CherryPy 클래스는 전체 애플리케이션과 자체 내장 웹 서버를 포함하는 자체 포함 Python 모듈을 생성하기 위해 CherryPy 엔진에 의해 처리 및 컴파일되어야합니다.

  • CherryPy는 URL과 쿼리 문자열을 Python 메서드 호출에 매핑 할 수 있습니다.

http://somehost.net/echo?message=hello would map to echo(message='hello')

CherryPy 프로젝트 개발 2 년 동안 커뮤니티의 지원을 받았으며 Remi는 여러 개선 된 버전을 출시했습니다.

2004 년 6 월, 프로젝트의 미래와 동일한 아키텍처를 계속 유지해야하는지에 대한 논의가 시작되었습니다. 여러 프로젝트 담당자의 브레인 스토밍과 토론을 통해 곧 CherryPy2의 핵심 부분이 된 객체 게시 엔진 및 필터 개념이 탄생했습니다. 2004 년 10 월에 CherryPy 2 알파의 첫 번째 버전이 개념 증명으로 출시되었습니다. 이러한 핵심 아이디어. CherryPy 2.0은 진정한 성공이었습니다. 그러나 그 디자인은 여전히 ​​개선 될 수 있고 리팩토링이 필요하다는 것을 인식했습니다.

피드백을 기반으로 논의한 후 CherryPy의 API는 우아함을 향상시키기 위해 더욱 수정되어 2005 년 10 월 CherryPy 2.1.0이 출시되었습니다. 다양한 변경 후 팀은 2006 년 4 월 CherryPy 2.2.0을 출시했습니다.

CherryPy의 강점

CherryPy의 다음 기능은 강점으로 간주됩니다.

간단

CherryPy에서 프로젝트를 개발하는 것은 Python의 규칙과 들여 쓰기에 따라 몇 줄의 코드로 개발 된 간단한 작업입니다.

CherryPy는 또한 매우 모듈 식입니다. 기본 구성 요소는 올바른 논리 개념으로 잘 관리되고 상위 클래스는 하위 클래스로 확장 가능합니다.

CherryPy는 Python의 모든 기능을 활용합니다. 또한 세계적 수준의 응용 프로그램을 개발하는 데 필요한 강력한 확장 지점 인 도구와 플러그인을 제공합니다.

오픈 소스

CherryPy는 오픈 소스 Python 웹 프레임 워크 (오픈 소스 BSD 라이선스에 따라 라이선스가 부여됨)로,이 프레임 워크를 제로 비용으로 상업적으로 사용할 수 있습니다.

커뮤니티 도움말

다양한 유형의 질문과 답변으로 완벽한 지원을 제공하는 헌신적 인 커뮤니티가 있습니다. 커뮤니티는 초급부터 고급까지 개발자에게 완전한 지원을 제공하려고 노력합니다.

전개

응용 프로그램을 배포하는 비용 효율적인 방법이 있습니다. CherryPy에는 애플리케이션을 호스팅하기위한 자체 프로덕션 준비 HTTP 서버가 포함되어 있습니다. CherryPy는 모든 WSGI 호환 게이트웨이에 배포 할 수도 있습니다.

CherryPy는 대부분의 오픈 소스 프로젝트와 같은 패키지로 제공되며 다음과 같이 다양한 방법으로 다운로드 및 설치할 수 있습니다.

  • Tarball 사용
  • easy_install 사용
  • Subversion 사용

요구 사항

CherryPy 프레임 워크 설치를위한 기본 요구 사항은 다음과 같습니다.

  • 버전 2.4 이상의 Python
  • 체리 파이 3.0

Python 모듈을 설치하는 것은 쉬운 프로세스로 간주됩니다. 설치에는 다음 명령 사용이 포함됩니다.

python setup.py build
python setup.py install

Python 패키지는 다음 기본 디렉토리에 저장됩니다.

  • UNIX 또는 Linux에서
/usr/local/lib/python2.4/site-packages
or
/usr/lib/python2.4/site-packages
  • Microsoft Windows에서
C:\Python or C:\Python2x
  • Mac OS에서는
Python:Lib:site-package

Tarball을 사용하여 설치

Tarball은 파일 또는 디렉토리의 압축 된 아카이브입니다. CherryPy 프레임 워크는 각 릴리스 (알파, 베타 및 안정)에 대해 Tarball을 제공합니다.

라이브러리의 완전한 소스 코드가 포함되어 있습니다. 이름은 UNIX 및 기타 운영 체제에서 사용되는 유틸리티에서 가져옵니다.

다음은 tar ball을 사용하여 CherryPy를 설치하기 위해 따라야 할 단계입니다.

Step 1 − 다음에서 사용자 요구 사항에 따라 버전을 다운로드합니다. http://download.cherrypy.org/

Step 2− Tarball이 다운로드 된 디렉토리를 검색하고 압축을 풉니 다. Linux 운영 체제의 경우 다음 명령을 입력하십시오-

tar zxvf cherrypy-x.y.z.tgz

Microsoft Windows의 경우 사용자는 7-Zip 또는 Winzip과 같은 유틸리티를 사용하여 그래픽 인터페이스를 통해 아카이브의 압축을 풀 수 있습니다.

Step 3 − 새로 생성 된 디렉토리로 이동하고 다음 명령어를 사용하여 CherryPy를 빌드합니다. −

python setup.py build

전역 설치의 경우 다음 명령을 사용해야합니다.

python setup.py install

easy_install을 사용한 설치

PEAK (Python Enterprise Application Kit)는 Easy Install이라는 Python 모듈을 제공합니다. 이는 Python 패키지의 배포를 용이하게합니다. 이 모듈은 Python 애플리케이션 및 제품을 다운로드, 빌드 및 배포하는 절차를 단순화합니다.

Easy Install은 CherryPy를 설치하기 전에 시스템에 설치해야합니다.

Step 1 − ez_setup.py 모듈을 http://peak.telecommunity.com 컴퓨터에서 관리 권한을 사용하여 실행합니다 : python ez_setup.py.

Step 2 − Easy Install을 설치하려면 다음 명령을 사용합니다.

easy_install product_name

Step 3− easy_install은 주어진 제품을 찾기 위해 PyPI (Python Package Index)를 검색합니다. PyPI는 모든 Python 제품에 대한 중앙 집중식 정보 저장소입니다.

다음 명령을 사용하여 최신 버전의 CherryPy를 배포하십시오.

easy_install cherrypy

Step 4 − easy_install은 CherryPy를 다운로드하여 Python 환경에 전역 적으로 빌드 및 설치합니다.

Subversion을 사용하여 설치

Subversion을 사용한 CherryPy 설치는 다음과 같은 상황에서 권장됩니다.

  • 기능이 존재하거나 버그가 수정되었으며 개발중인 코드에서만 사용할 수 있습니다.

  • 개발자가 CherryPy 자체에서 작업 할 때.

  • 사용자가 버전 관리 저장소의 기본 분기에서 분기가 필요한 경우.

  • 이전 릴리스의 버그 수정을 위해.

Subversioning의 기본 원칙은 저장소를 등록하고 일련의 변경 사항이 포함 된 각 버전을 추적하는 것입니다.

Subversion을 사용하여 CherryPy를 설치하는 방법을 이해하려면 다음 단계를 따르십시오.

Step 1 − 프로젝트의 최신 버전을 사용하려면 Subversion 저장소에있는 트렁크 폴더를 확인해야합니다.

Step 2 − 셸에서 다음 명령을 입력합니다.

svn co http://svn.cherrypy.org/trunk cherrypy

Step 3 − 이제 CherryPy 디렉토리를 생성하고 여기에 전체 소스 코드를 다운로드합니다.

설치 테스트

Java와 같은 응용 프로그램과 동일한 방식으로 응용 프로그램이 시스템에 제대로 설치되었는지 여부를 확인해야합니다.

이전 장에서 언급 한 세 가지 방법 중 하나를 선택하여 사용자 환경에 CherryPy를 설치하고 배포 할 수 있습니다. CherryPy는 다음과 같이 Python 셸에서 가져올 수 있어야합니다.

import cherrypy

cherrypy.__version__
'3.0.0'

CherryPy가 로컬 시스템의 Python 환경에 전역 적으로 설치되지 않은 경우 PYTHONPATH 환경 변수를 설정해야합니다. 그렇지 않으면 다음과 같은 방식으로 오류가 표시됩니다.

import cherrypy

Traceback (most recent call last):
File "<stdin>", line 1, in ?
ImportError: No module named cherrypy

CherryPy의 작동을 이해하기 위해 정의해야하는 몇 가지 중요한 키워드가 있습니다. 키워드와 정의는 다음과 같습니다-

S. 아니 키워드 및 정의
1.

Web Server

HTTP 프로토콜을 다루는 인터페이스입니다. 목표는 응답을받을 수 있도록 HTTP 요청을 애플리케이션 서버로 변환하는 것입니다.

2.

Application

정보를 수집하는 소프트웨어입니다.

삼.

Application server

하나 이상의 응용 프로그램을 보유하는 구성 요소입니다.

4.

Web application server

웹 서버와 애플리케이션 서버의 조합입니다.

다음 예제는 CherryPy의 샘플 코드를 보여줍니다-

import cherrypy

class demoExample:
   def index(self):
   return "Hello World!!!"
   index.exposed = True
cherrypy.quickstart(demoExample())

이제 코드가 어떻게 작동하는지 이해하겠습니다.

  • 이름이 지정된 패키지 CherryPy 적절한 기능을 보장하기 위해 항상 지정된 클래스로 가져옵니다.

  • 위의 예에서 이름이 지정된 함수 index 매개 변수 "Hello World !!!"를 반환합니다.

  • 마지막 줄은 웹 서버를 시작하고 지정된 클래스 (여기서는 demoExample)를 호출하고 기본 함수 인덱스에 언급 된 값을 반환합니다.

예제 코드는 다음 출력을 반환합니다.

CherryPy는 자체 웹 (HTTP) 서버와 함께 제공됩니다. 그렇기 때문에 CherryPy는 자체 포함되어 있으며 사용자가 라이브러리를 얻은 후 몇 분 이내에 CherryPy 애플리케이션을 실행할 수 있습니다.

그만큼 web server 모든 요청과 응답이 추적되는 애플리케이션의 게이트웨이 역할을합니다.

웹 서버를 시작하려면 사용자는 다음 호출을해야합니다.

cherryPy.server.quickstart()

그만큼 internal engine of CherryPy 다음 활동에 대한 책임이 있습니다.

  • 요청 및 응답 개체의 생성 및 관리.
  • CherryPy 프로세스 제어 및 관리.

CherryPy – 구성

프레임 워크는 HTTP 서버를 매개 변수화 할 수있는 자체 구성 시스템과 함께 제공됩니다. 구성 설정은 INI 형식에 가까운 구문이있는 텍스트 파일 또는 완전한 Python 사전으로 저장할 수 있습니다.

CherryPy 서버 인스턴스를 구성하려면 개발자가 설정의 전역 섹션을 사용해야합니다.

global_conf = {
   'global': {
      'server.socket_host': 'localhost',
      'server.socket_port': 8080,
   },
}

application_conf = {
   '/style.css': {
      'tools.staticfile.on': True,
      'tools.staticfile.filename': os.path.join(_curdir, 'style.css'),
   }
}

This could be represented in a file like this:
[global]
server.socket_host = "localhost"
server.socket_port = 8080
[/style.css]
tools.staticfile.on = True
tools.staticfile.filename = "/full/path/to.style.css"

HTTP 준수

CherryPy는 천천히 발전하고 있지만 나중에 HTTP / 1.1 지원으로 전송되는 HTTP / 1.0 지원과 함께 HTTP 사양 컴파일이 포함됩니다.

CherryPy는 모든 필수 및 필수 수준을 구현하지만 사양의 모든 필수 수준을 구현하지 않으므로 HTTP / 1.1을 조건부로 준수한다고합니다. 따라서 CherryPy는 HTTP / 1.1의 다음 기능을 지원합니다.

  • 클라이언트가 HTTP / 1.1을 지원한다고 주장하는 경우 지정된 프로토콜 버전으로 이루어진 모든 요청에서 헤더 필드를 보내야합니다. 완료되지 않으면 CherryPy는 요청 처리를 즉시 중지합니다.

  • CherryPy는 모든 구성에서 사용되는 날짜 헤더 필드를 생성합니다.

  • CherryPy는 클라이언트 지원으로 응답 상태 코드 (100)를 처리 할 수 ​​있습니다.

  • CherryPy의 내장 HTTP 서버는 Connection : Keep-Alive 헤더를 사용하여 HTTP / 1.1의 기본값 인 영구 연결을 지원합니다.

  • CherryPy는 올바르게 청크 된 요청과 응답을 처리합니다.

  • CherryPy는 If-Modified-Since 및 If-Unmodified-Since 헤더의 두 가지 고유 한 방식으로 요청을 지원하고 그에 따라 요청에 따라 응답을 보냅니다.

  • CherryPy는 모든 HTTP 메소드를 허용합니다.

  • CherryPy는 클라이언트와 서버에 설정된 설정 간의 HTTP 버전 조합을 처리합니다.

다중 스레드 응용 프로그램 서버

CherryPy는 멀티 스레딩 개념을 기반으로 설계되었습니다. 개발자가 CherryPy 네임 스페이스에 값을 가져 오거나 설정할 때마다 다중 스레드 환경에서 수행됩니다.

cherrypy.request와 cherrypy.response는 모두 스레드 데이터 컨테이너이며, 이는 애플리케이션이 런타임에 프록시를 통해 프록시되는 요청을 파악하여 독립적으로 호출 함을 의미합니다.

스레드 패턴을 사용하는 애플리케이션 서버는 스레드 사용이 동기화 요구 사항으로 인해 문제의 가능성을 증가시키는 것으로 간주되기 때문에 높게 평가되지 않습니다.

다른 대안은-

다중 프로세스 패턴

각 요청은 자체 Python 프로세스에서 처리됩니다. 여기에서 서버의 성능과 안정성이 더 좋다고 생각할 수 있습니다.

비동기 패턴

여기에서 새 연결을 수락하고 데이터를 클라이언트로 다시 보내는 것은 요청 프로세스에서 비동기 적으로 수행됩니다. 이 기술은 효율성으로 잘 알려져 있습니다.

URL 발송

CherryPy 커뮤니티는 더 유연하고 디스패처를위한 다른 솔루션이 높이 평가되기를 원합니다. CherryPy 3는 다른 기본 제공 디스패처를 제공하고 자체 디스패처를 작성하고 사용하는 간단한 방법을 제공합니다.

  • HTTP 메서드를 개발하는 데 사용되는 응용 프로그램입니다. (GET, POST, PUT 등)
  • URL에서 경로를 정의하는 것 – Routes Dispatcher

HTTP 메소드 디스패처

일부 응용 프로그램에서 URI는 리소스에서 서버가 수행하는 작업과 독립적입니다.

예를 들면http://xyz.com/album/delete/10

URI에는 클라이언트가 수행하려는 작업이 포함됩니다.

기본적으로 CherryPy 디스패처는 다음과 같은 방식으로 매핑됩니다.

album.delete(12)

위에서 언급 한 디스패처는 올바르게 언급되었지만 다음과 같은 방법으로 독립적으로 만들 수 있습니다.

http://xyz.com/album/10

사용자는 서버가 정확한 페이지를 어떻게 전달하는지 궁금 할 수 있습니다. 이 정보는 HTTP 요청 자체에 의해 전달됩니다. 클라이언트에서 서버로 요청이있을 때 CherryPy는 가장 적합한 핸들러로 보이며 핸들러는 URI가 대상으로하는 리소스를 나타냅니다.

DELETE /album/12 HTTP/1.1

라우트 디스패처

다음은 디스패치에 필요한 메서드의 매개 변수 목록입니다.

  • name 매개 변수는 연결할 경로의 고유 한 이름입니다.

  • 경로는 URI와 일치하는 패턴입니다.

  • 컨트롤러는 페이지 핸들러를 포함하는 인스턴스입니다.

  • Routes 디스패처를 사용하면 URI와 일치하는 패턴을 연결하고 특정 페이지 핸들러를 연결합니다.

작동 원리를 이해하기 위해 예를 들어 보겠습니다.

import random
import string
import cherrypy

class StringMaker(object):
   @cherrypy.expose
   def index(self):
      return "Hello! How are you?"
   
   @cherrypy.expose
   def generate(self, length=9):
      return ''.join(random.sample(string.hexdigits, int(length)))
		
if __name__ == '__main__':
   cherrypy.quickstart(StringMaker ())

위 코드의 출력을 얻으려면 아래 단계를 따르십시오-

Step 1 − 위에서 언급 한 파일을 tutRoutes.py.

Step 2 − 다음 URL을 방문하세요 −

http://localhost:8080/generate?length=10

Step 3 − 다음 출력을 받게됩니다 −

CherryPy 내에서 기본 제공 도구는 CherryPy 라이브러리를 호출하는 단일 인터페이스를 제공합니다. CherryPy에 정의 된 도구는 다음과 같은 방법으로 구현할 수 있습니다.

  • 구성 설정에서
  • Python 데코레이터 또는 페이지 핸들러의 특수 _cp_config 속성을 통해
  • 모든 함수 내에서 적용 할 수있는 Python 콜 러블

기본 인증 도구

이 도구의 목적은 애플리케이션에서 설계된 애플리케이션에 기본 인증을 제공하는 것입니다.

인수

이 도구는 다음 인수를 사용합니다-

이름 기본 기술
왕국 해당 없음 영역 값을 정의하는 문자열입니다.
사용자 해당 없음 형태의 사전-username : password 또는 그러한 사전을 반환하는 Python 호출 가능 함수.
암호화 없음 클라이언트가 반환 한 암호를 암호화하고 사용자 사전에 제공된 암호화 된 암호와 비교하는 데 사용되는 Python 호출 가능.

작동 원리를 이해하기 위해 예를 들어 보겠습니다.

import sha
import cherrypy

class Root:
@cherrypy.expose
def index(self):

return """
<html>
   <head></head>
   <body>
      <a href = "admin">Admin </a>
   </body>
</html>
""" 

class Admin:

@cherrypy.expose
def index(self):
return "This is a private area"

if __name__ == '__main__':
def get_users():
# 'test': 'test'
return {'test': 'b110ba61c4c0873d3101e10871082fbbfd3'}
def encrypt_pwd(token):

return sha.new(token).hexdigest()
   conf = {'/admin': {'tools.basic_auth.on': True,
      tools.basic_auth.realm': 'Website name',
      'tools.basic_auth.users': get_users,
      'tools.basic_auth.encrypt': encrypt_pwd}}
   root = Root()
root.admin = Admin()
cherrypy.quickstart(root, '/', config=conf)

그만큼 get_users함수는 하드 코딩 된 사전을 반환하지만 데이터베이스 또는 다른 곳에서 값을 가져옵니다. 클래스 관리자는 CherryPy의 인증 내장 도구를 사용하는이 기능을 포함합니다. 인증은 비밀번호와 사용자 ID를 암호화합니다.

침입자가 암호를 인코딩하고 디코딩 할 수 있기 때문에 기본 인증 도구는 실제로 안전하지 않습니다.

캐싱 도구

이 도구의 목적은 CherryPy 생성 콘텐츠의 메모리 캐싱을 제공하는 것입니다.

인수

이 도구는 다음 인수를 사용합니다-

이름 기본 기술
invalid_methods ( "POST", "PUT", "DELETE") 캐시되지 않을 HTTP 메서드 문자열의 튜플입니다. 이러한 메서드는 또한 리소스의 캐시 된 복사본을 무효화 (삭제)합니다.
cache_Class MemoryCache 캐싱에 사용할 클래스 객체

디코딩 도구

이 도구의 목적은 들어오는 요청 매개 변수를 디코딩하는 것입니다.

인수

이 도구는 다음 인수를 사용합니다-

이름 기본 기술
부호화 없음 콘텐츠 유형 헤더를 찾습니다.
Default_encoding "UTF-8" 제공되거나 발견되지 않은 경우 사용할 기본 인코딩입니다.

작동 원리를 이해하기 위해 예를 들어 보겠습니다.

import cherrypy
from cherrypy import tools

class Root:
@cherrypy.expose
def index(self):

return """ 
<html>
   <head></head>
   <body>
      <form action = "hello.html" method = "post">
         <input type = "text" name = "name" value = "" />
         <input type = ”submit” name = "submit"/>
      </form>
   </body>
</html>
"""

@cherrypy.expose
@tools.decode(encoding='ISO-88510-1')
def hello(self, name):
return "Hello %s" % (name, )
if __name__ == '__main__':
cherrypy.quickstart(Root(), '/')

위의 코드는 사용자로부터 문자열을 받아 사용자를 "hello.html"페이지로 리디렉션하여 지정된 이름의 "Hello"로 표시합니다.

위 코드의 출력은 다음과 같습니다.

hello.html

풀 스택 응용 프로그램은 일부 명령 또는 파일 실행을 통해 새 응용 프로그램을 만드는 기능을 제공합니다.

web2py 프레임 워크와 같은 Python 애플리케이션을 고려하십시오. 전체 프로젝트 / 애플리케이션은 MVC 프레임 워크 측면에서 생성됩니다. 마찬가지로 CherryPy를 사용하면 사용자가 요구 사항에 따라 코드 레이아웃을 설정하고 구성 할 수 있습니다.

이 장에서는 CherryPy 애플리케이션을 생성하고 실행하는 방법에 대해 자세히 알아 봅니다.

파일 시스템

응용 프로그램의 파일 시스템은 다음 스크린 샷에 나와 있습니다.

다음은 파일 시스템에있는 다양한 파일에 대한 간략한 설명입니다.

  • config.py− 모든 응용 프로그램에는 구성 파일과이를로드하는 방법이 필요합니다. 이 기능은 config.py에서 정의 할 수 있습니다.

  • controllers.py− MVC는 사용자에게 인기있는 디자인 패턴입니다. controllers.py는 cherrypy.tree에 마운트 될 모든 객체가 구현되는 곳입니다 .

  • models.py −이 파일은 일부 서비스 또는 영구 데이터 저장을 위해 데이터베이스와 직접 상호 작용합니다.

  • server.py −이 파일은로드 밸런싱 프록시와 제대로 작동하는 프로덕션 준비 웹 서버와 상호 작용합니다.

  • Static − 모든 CSS 및 이미지 파일을 포함합니다.

  • Views − 주어진 응용 프로그램에 대한 모든 템플릿 파일을 포함합니다.

CherryPy 애플리케이션을 만드는 단계를 자세히 알아 보겠습니다.

Step 1 − 해당 응용 프로그램을 포함해야하는 응용 프로그램을 만듭니다.

Step 2− 디렉토리 안에 프로젝트에 해당하는 python 패키지를 생성합니다. gedit 디렉토리를 만들고 동일한 디렉토리에 _init_.py 파일을 포함합니다.

Step 3 − 패키지 안에 다음 내용이 포함 된 controllers.py 파일을 포함합니다.

#!/usr/bin/env python

import cherrypy

class Root(object):

   def __init__(self, data):
      self.data = data

   @cherrypy.expose
   def index(self):
      return 'Hi! Welcome to your application'

def main(filename):
   data = {} # will be replaced with proper functionality later

   # configuration file
   cherrypy.config.update({
      'tools.encode.on': True, 'tools.encode.encoding': 'utf-8',
      'tools.decode.on': True,
      'tools.trailing_slash.on': True,
      'tools.staticdir.root': os.path.abspath(os.path.dirname(__file__)),
   })

   cherrypy.quickstart(Root(data), '/', {
      '/media': {
         'tools.staticdir.on': True,
         'tools.staticdir.dir': 'static'
      }
   })
	
if __name__ == '__main__':
main(sys.argv[1])

Step 4− 사용자가 양식을 통해 값을 입력하는 응용 프로그램을 고려하십시오. 애플리케이션에 index.html과 submit.html의 두 가지 형식을 포함 해 보겠습니다.

Step 5 − 위의 컨트롤러 용 코드에서 우리는 index(), 이는 기본 함수이며 특정 컨트롤러가 호출되면 먼저로드됩니다.

Step 6 −의 구현 index() 방법은 다음과 같이 변경할 수 있습니다-

@cherrypy.expose
   def index(self):
      tmpl = loader.load('index.html')
	 
      return tmpl.generate(title='Sample').render('html', doctype='html')

Step 7− 이것은 주어진 응용 프로그램을 시작할 때 index.html을로드하고 주어진 출력 스트림으로 보냅니다. index.html 파일은 다음과 같습니다-

index.html

<!DOCTYPE html >
<html>
   <head>
      <title>Sample</title>
   </head>
	
   <body class = "index">
      <div id = "header">
         <h1>Sample Application</h1>
      </div>
		
      <p>Welcome!</p>
		
      <div id = "footer">
         <hr>
      </div>
		
   </body>
	
</html>

Step 8 − 루트 클래스에 메소드를 추가하는 것이 중요합니다. controller.py 이름 및 제목과 같은 값을 허용하는 양식을 작성하려는 경우.

@cherrypy.expose
   def submit(self, cancel = False, **value):
	
      if cherrypy.request.method == 'POST':
         if cancel:
            raise cherrypy.HTTPRedirect('/') # to cancel the action
         link = Link(**value)
         self.data[link.id] = link
         raise cherrypy.HTTPRedirect('/')
      tmp = loader.load('submit.html')
      streamValue = tmp.generate()
		
      return streamValue.render('html', doctype='html')

Step 9 − submit.html에 포함될 코드는 다음과 같습니다 −

<!DOCTYPE html>
   <head>
      <title>Input the new link</title>
   </head>
	
   <body class = "submit">
      <div id = " header">
         <h1>Submit new link</h1>
      </div>
		
      <form action = "" method = "post">
         <table summary = "">
            <tr>
               <th><label for = " username">Your name:</label></th>
               <td><input type = " text" id = " username" name = " username" /></td>
            </tr>
				
            <tr>
               <th><label for = " url">Link URL:</label></th>
               <td><input type = " text" id=" url" name= " url" /></td>
            </tr>
				
            <tr>
               <th><label for = " title">Title:</label></th>
               <td><input type = " text" name = " title" /></td>
            </tr>
				
            <tr>
               <td></td>
               <td>
                  <input type = " submit" value = " Submit" />
                  <input type = " submit" name = " cancel" value = "Cancel" />
               </td>
            </tr>
				
         </table>
			
      </form>
      <div id = "footer">
      </div>
		
   </body>
	
</html>

Step 10 − 다음 출력을 받게됩니다 −

여기서 메소드 이름은 "POST"로 정의됩니다. 파일에 지정된 방법을 교차 검증하는 것은 항상 중요합니다. 방법에 "POST"방법이 포함 된 경우 해당 필드의 데이터베이스에서 값을 다시 확인해야합니다.

메소드에 "GET"메소드가 포함 된 경우 저장할 값이 URL에 표시됩니다.

웹 서비스는 개방형 프로토콜 및 표준을 포함하는 응용 프로그램 또는 시스템 간의 데이터 교환을 지원하는 웹 기반 구성 요소 집합입니다. 웹에서 게시, 사용 및 찾을 수 있습니다.

웹 서비스는 RWS (RESTfUL Web Service), WSDL, SOAP 등과 같은 다양한 유형입니다.

REST — 표현 상태 전송

원격 프로 시저를 호출하는 대신 상태를 조작하는 데 사용할 수있는 클라이언트에서 서버로 상태를 전송하는 원격 액세스 프로토콜 유형입니다.

  • 특정 인코딩이나 구조 및 유용한 오류 메시지를 반환하는 방법을 정의하지 않습니다.

  • HTTP "동사"를 사용하여 상태 전송 작업을 수행합니다.

  • 리소스는 URL을 사용하여 고유하게 식별됩니다.

  • API가 아니라 API 전송 계층입니다.

REST는 네트워크에서 리소스 명명법을 유지하고 이러한 리소스에 대한 작업을 수행하는 통합 메커니즘을 제공합니다. 각 리소스는 하나 이상의 식별자로 식별됩니다. REST 인프라가 HTTP 기반으로 구현 된 경우 이러한 식별자는Uniform Resource Identifiers (URIs).

다음은 URI 세트의 두 가지 일반적인 하위 집합입니다-

부분 집합 전체 형식
URL 유니폼 리소스 로케이터 http://www.gmail.com/
항아리 균일 한 리소스 이름 urn : isbn : 0-201-71088-9 urn : uuid : 13e8cf26-2a25-11db-8693-000ae4ea7d46

CherryPy 아키텍처의 구현을 이해하기 전에 CherryPy의 아키텍처에 중점을 두겠습니다.

CherryPy는 다음 세 가지 구성 요소를 포함합니다.

  • cherrypy.engine − 프로세스 시작 / 해체 및 이벤트 처리를 제어합니다.

  • cherrypy.server − WSGI 또는 HTTP 서버를 설정하고 제어합니다.

  • cherrypy.tools − HTTP 요청 처리와 직교하는 유틸리티 도구 상자.

CherryPy를 통한 REST 인터페이스

RESTful 웹 서비스는 다음의 도움으로 CherryPy 아키텍처의 각 섹션을 구현합니다.

  • Authentication
  • Authorization
  • Structure
  • Encapsulation
  • 오류 처리

입증

인증은 우리가 상호 작용하는 사용자를 확인하는 데 도움이됩니다. CherryPy에는 각 인증 방법을 처리하는 도구가 포함되어 있습니다.

def authenticate():
   if not hasattr(cherrypy.request, 'user') or cherrypy.request.user is None:
      # < Do stuff to look up your users >
		
      cherrypy.request.authorized = False # This only authenticates. 
         Authz must be handled separately.
		
      cherrypy.request.unauthorized_reasons = []
      cherrypy.request.authorization_queries = []
		
cherrypy.tools.authenticate = \
   cherrypy.Tool('before_handler', authenticate, priority=10)

위의 함수 authenticate ()는 클라이언트 또는 사용자의 존재를 확인하는 데 도움이됩니다. 기본 제공 도구는 프로세스를 체계적으로 완료하는 데 도움이됩니다.

권한 부여

권한 부여는 URI를 통해 프로세스의 온 전성을 유지하는 데 도움이됩니다. 이 프로세스는 사용자 토큰 리드별로 개체를 변형하는데도 도움이됩니다.

def authorize_all():
   cherrypy.request.authorized = 'authorize_all'
	
cherrypy.tools.authorize_all = cherrypy.Tool('before_handler', authorize_all, priority=11)

def is_authorized():
   if not cherrypy.request.authorized:
      raise cherrypy.HTTPError("403 Forbidden",
         ','.join(cherrypy.request.unauthorized_reasons))
			
cherrypy.tools.is_authorized = cherrypy.Tool('before_handler', is_authorized, 
priority = 49)

cherrypy.config.update({
   'tools.is_authorized.on': True,
   'tools.authorize_all.on': True
})

내장 된 권한 부여 도구는 이전 예에서 언급했듯이 체계적인 방식으로 루틴을 처리하는 데 도움이됩니다.

구조

API 구조를 유지하면 애플리케이션의 URI 매핑 작업 부하를 줄이는 데 도움이됩니다. 항상 API를 검색 가능하고 깔끔하게 유지해야합니다. CherryPy 프레임 워크 용 API의 기본 구조는 다음과 같아야합니다.

  • 계정 및 사용자
  • Autoresponder
  • Contact
  • File
  • Folder
  • 목록 및 필드
  • 메시지 및 배치

캡슐화

캡슐화는 가볍고 사람이 읽을 수 있고 다양한 클라이언트가 액세스 할 수있는 API를 만드는 데 도움이됩니다. 생성, 검색, 업데이트 및 삭제와 함께 항목 목록에는 API 캡슐화가 필요합니다.

오류 처리

이 프로세스는 API가 특정 본능에서 실행되지 않는 경우 오류를 관리합니다. 예를 들어 400은 잘못된 요청이고 403은 무단 요청입니다.

데이터베이스, 유효성 검사 또는 응용 프로그램 오류에 대한 예로 다음을 고려하십시오.

import cherrypy
import json

def error_page_default(status, message, traceback, version):
   ret = {
      'status': status,
      'version': version,
      'message': [message],
      'traceback': traceback
   }
	
   return json.dumps(ret)
	
class Root:
   _cp_config = {'error_page.default': error_page_default}
	
@cherrypy.expose
   def index(self):
      raise cherrypy.HTTPError(500, "Internal Sever Error")
cherrypy.quickstart(Root())

위의 코드는 다음 출력을 생성합니다.

접근 도구가 내장되어있어 CherryPy를 통해 API (Application Programming Interface) 관리가 용이합니다.

HTTP 방법

리소스에서 작동하는 HTTP 메소드 목록은 다음과 같습니다.

S. 아니 HTTP 방법 및 작동
1.

HEAD

리소스 메타 데이터를 검색합니다.

2.

GET

리소스 메타 데이터 및 콘텐츠를 검색합니다.

삼.

POST

요청 본문에 포함 된 데이터를 사용하여 새 리소스를 만들도록 서버에 요청합니다.

4.

PUT

기존 리소스를 요청 본문에 포함 된 리소스로 바꾸도록 서버에 요청합니다.

5.

DELETE

해당 URI로 식별되는 리소스를 제거하도록 서버에 요청합니다.

6.

OPTIONS

전역 적으로 또는 구체적으로 리소스에 대한 기능에 대한 세부 정보를 반환하도록 서버에 요청합니다.

아톰 게시 프로토콜 (APP)

APP는 웹 리소스의 게시 및 편집을 허용하기 위해 HTTP 위에 애플리케이션 수준 프로토콜로 Atom 커뮤니티에서 발생했습니다. APP 서버와 클라이언트 간의 메시지 단위는 Atom XML 문서 형식을 기반으로합니다.

Atom Publishing Protocol은 HTTP 및 해당 메커니즘과 Atom XML 문서 형식을 메시지 단위로 사용하여 APP 서비스와 사용자 에이전트 간의 작업 집합을 정의합니다.

APP는 먼저 APP 서비스에서 제공하는 다양한 컬렉션의 URI를 사용자 에이전트에 제공하는 서비스 문서를 정의합니다.

APP 작동 방식을 보여주는 예를 들어 보겠습니다.

<?xml version = "1.0" encoding = "UTF-8"?>
<service xmlns = "http://purl.org/atom/app#" xmlns:atom = "http://www.w3.org/2005/Atom">
   
   <workspace>
      <collection href = "http://host/service/atompub/album/">
         <atom:title> Albums</atom:title>
         <categories fixed = "yes">
            <atom:category term = "friends" />
         </categories>
      </collection>
      
      <collection href = "http://host/service/atompub/film/">
         <atom:title>Films</atom:title>
         <accept>image/png,image/jpeg</accept>
      </collection>
   </workspace>
	
</service>

APP는 다음 표에 설명 된대로 HTTP 메서드를 사용하여 컬렉션의 구성원 또는 컬렉션 자체에 대해 기본 CRUD 작업을 수행하는 방법을 지정합니다.

조작 HTTP 방법 상태 코드 함유량
검색 가져 오기 200 자원을 나타내는 Atom 항목
창조하다 우편 201 Location 및 Content-Location 헤더를 통해 새로 생성 된 리소스의 URI
최신 정보 놓다 200 자원을 나타내는 Atom 항목
지우다 지우다 200 없음

프레젠테이션 계층은이를 통해 전달되는 통신이 의도 한 수신자를 대상으로하는지 확인합니다. CherryPy는 다양한 템플릿 엔진에 의해 프레젠테이션 레이어의 작동을 유지합니다.

템플릿 엔진은 비즈니스 로직의 도움을 받아 페이지의 입력을 가져온 다음 의도 된 대상만을 대상으로하는 최종 페이지로 처리합니다.

Kid — 템플릿 엔진

Kid 처리 할 템플릿의 이름 (필수)과 템플릿이 렌더링 될 때 전달할 데이터의 입력을 포함하는 간단한 템플릿 엔진입니다.

처음으로 템플릿을 만들 때 Kid는 템플릿의 캐시 된 버전으로 사용할 수있는 Python 모듈을 만듭니다.

그만큼 kid.Template 함수는 출력 내용을 렌더링하는 데 사용할 수있는 템플릿 클래스의 인스턴스를 반환합니다.

템플릿 클래스는 다음 명령 세트를 제공합니다-

S. 아니 명령 및 설명
1.

serialize

출력 내용을 문자열로 반환합니다.

2.

generate

출력 내용을 반복자로 반환합니다.

삼.

write

출력 내용을 파일 객체로 덤프합니다.

이 명령에 사용되는 매개 변수는 다음과 같습니다.

S. 아니 명령 및 설명
1.

encoding

출력 내용을 인코딩하는 방법을 알려줍니다.

2.

fragment

XML 프롤로그 또는 Doctype에 알려주는 부울 값입니다.

삼.

output

이 유형의 직렬화는 콘텐츠를 렌더링하는 데 사용됩니다.

방법을 이해하기 위해 예를 들어 보겠습니다. kid 작품-

<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01//EN" "http://www.w3.org/TR/html4/strict.dtd">
<html xmlns:py = "http://purl.org/kid/ns#">
   <head>
      <title>${title}</title> <link rel = "stylesheet" href = "style.css" /> </head> <body> <p>${message}</p>
   </body>
</html>

The next step after saving the file is to process the template via the Kid engine.

import kid

params = {'title': 'Hello world!!', 'message': 'CherryPy.'}
t = kid.Template('helloworld.kid', **params)
print t.serialize(output='html')

아이의 속성

다음은 Kid의 속성입니다-

XML 기반 템플릿 언어

XML 기반 언어입니다. Kid 템플릿은 적절한 이름 지정 규칙이있는 잘 구성된 XML 문서 여야합니다.

Kid는 XML 요소 내에 속성을 구현하여 요소에 도달하기 위해 따라야 할 작업에 대한 기본 엔진을 업데이트합니다. XML 문서 내의 다른 기존 속성과 겹치지 않도록 Kid는 자체 네임 스페이스를 도입했습니다.

<p py:if = "...">...</p>

가변 대체

Kid는 변수 대체 체계와 간단한 접근 방식 인 $ {variable-name}을 제공합니다.

변수는 요소의 속성 또는 요소의 텍스트 콘텐츠로 사용할 수 있습니다. Kid는 처형이 일어날 때마다 변수를 평가할 것입니다.

사용자가 $ {something}과 같은 리터럴 문자열의 출력이 필요한 경우 달러 기호를 두 배로 늘려 변수 대체를 사용하여 이스케이프 할 수 있습니다.

조건문

템플릿에서 다른 경우를 토글하기 위해 다음 구문이 사용됩니다.

<tag py:if = "expression">...</tag>

여기서 tag는 요소의 이름입니다 (예 : DIV 또는 SPAN).

표현식은 Python 표현식입니다. 부울로서 True로 평가되면 요소가 출력 내용에 포함되거나 그렇지 않으면 출력 내용의 일부가 아닙니다.

루핑 메커니즘

Kid에서 요소를 반복하려면 다음 구문이 사용됩니다.

<tag py:for = "expression">...</tag>

여기서 tag는 요소의 이름입니다. 표현식은 Python 표현식입니다 (예 : [...]의 값).

다음 코드는 루핑 메커니즘이 어떻게 작동하는지 보여줍니다.

<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01//EN" "http://www.w3.org/TR/html4/strict.dtd">
<html>
   <head>
      <title>${title}</title> <link rel = "stylesheet" href = "style.css" /> </head> <body> <table> <caption>A few songs</caption> <tr> <th>Artist</th> <th>Album</th> <th>Title</th> </tr> <tr py:for = "info in infos"> <td>${info['artist']}</td>
            <td>${info['album']}</td> <td>${info['song']}</td>
         </tr>
      </table>
   </body>
</html>

import kid

params = discography.retrieve_songs()
t = kid.Template('songs.kid', **params)
print t.serialize(output='html')

그만큼 output 루핑 메커니즘이있는 위의 코드는 다음과 같습니다.

2005 년까지 모든 웹 애플리케이션에서 따르는 패턴은 페이지 당 하나의 HTTP 요청을 관리하는 것이 었습니다. 한 페이지를 다른 페이지로 이동하려면 전체 페이지를로드해야했습니다. 이것은 더 큰 수준에서 성능을 저하시킵니다.

따라서 상승이 있었다 rich client applications AJAX, XML 및 JSON을 포함하는 데 사용되었습니다.

AJAX

AJAX (Asynchronous JavaScript and XML)는 빠르고 동적 인 웹 페이지를 만드는 기술입니다. AJAX를 사용하면 백그라운드에서 서버와 소량의 데이터를 교환하여 웹 페이지를 비동기 적으로 업데이트 할 수 있습니다. 이는 전체 페이지를 다시로드하지 않고도 웹 페이지의 일부를 업데이트 할 수 있음을 의미합니다.

Google지도, Gmail, YouTube 및 Facebook은 AJAX 애플리케이션의 몇 가지 예입니다.

Ajax는 JavaScript를 사용하여 HTTP 요청을 보내는 아이디어를 기반으로합니다. 보다 구체적으로 AJAX는 XMLHttpRequest 객체와 해당 API에 의존하여 이러한 작업을 수행합니다.

JSON

JSON은 JavaScript 애플리케이션이이를 평가하고 나중에 조작 할 수있는 JavaScript 개체로 변환 할 수있는 방식으로 직렬화 된 JavaScript 개체를 전달하는 방법입니다.

예를 들어 사용자가 JSON 형식으로 포맷 된 앨범 개체를 서버에 요청하면 서버는 다음과 같이 출력을 반환합니다.

{'description': 'This is a simple demo album for you to test', 'author': ‘xyz’}

이제 데이터는 JavaScript 연관 배열이며 설명 필드는 다음을 통해 액세스 할 수 있습니다.

data ['description'];

애플리케이션에 AJAX 적용

index.html 및 Jquery 플러그인이있는 "media"라는 폴더와 AJAX 구현이있는 파일이 포함 된 애플리케이션을 고려하십시오. 파일 이름을 "ajax_app.py"로 간주하겠습니다.

ajax_app.py

import cherrypy
import webbrowser
import os
import simplejson
import sys

MEDIA_DIR = os.path.join(os.path.abspath("."), u"media")

class AjaxApp(object):
   @cherrypy.expose
   def index(self):
      return open(os.path.join(MEDIA_DIR, u'index.html'))

   @cherrypy.expose
   def submit(self, name):
      cherrypy.response.headers['Content-Type'] = 'application/json'
      return simplejson.dumps(dict(title="Hello, %s" % name))
		
config = {'/media':
   {'tools.staticdir.on': True,
   'tools.staticdir.dir': MEDIA_DIR,}
}
			
def open_page():
webbrowser.open("http://127.0.0.1:8080/")
cherrypy.engine.subscribe('start', open_page)
cherrypy.tree.mount(AjaxApp(), '/', config=config)
cherrypy.engine.start()

“AjaxApp”클래스는 미디어 폴더에 포함 된“index.html”웹 페이지로 리디렉션됩니다.

<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" 
   " http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd">
	
<html xmlns = "http://www.w3.org/1999/xhtml" lang = "en" xml:lang = "en">
   <head>
      <title>AJAX with jQuery and cherrypy</title>
      <meta http-equiv = " Content-Type" content = " text/html; charset=utf-8" />
      <script type = " text/javascript" src = " /media/jquery-1.4.2.min.js"></script>
		
      <script type = " text/javascript">
         $(function() { // When the testform is submitted... $("#formtest").submit(function() {
         
               // post the form values via AJAX...
               $.post('/submit', {name: $("#name").val()}, function(data) {
         
                  // and set the title with the result
                  $("#title").html(data['title']) ;
               });
               return false ;
            });
         });
      </script>
		
   </head>
	
   <body>
      <h1 id = "title">What's your name?</h1>
      <form id = " formtest" action = " #" method = " post">
         <p>
            <label for = " name">Name:</label>
            <input type = " text" id = "name" /> <br />
            <input type = " submit" value = " Set" />
         </p>
      </form>
   </body>
	
</html>

AJAX에 대한 함수는 <script> 태그 내에 포함됩니다.

산출

위의 코드는 다음 출력을 생성합니다.

사용자가 값을 제출하면 AJAX 기능이 구현되고 화면이 아래와 같이 양식으로 리디렉션됩니다.

이 장에서는 CherryPy 프레임 워크에서 애플리케이션을 만드는 방법에 중점을 둡니다.

중히 여기다 PhotoblogCherryPy의 데모 애플리케이션을위한 애플리케이션입니다. Photoblog 응용 프로그램은 일반 블로그이지만 주요 텍스트는 텍스트 대신 사진입니다. Photoblog 응용 프로그램의 주요 특징은 개발자가 디자인과 구현에 더 집중할 수 있다는 것입니다.

기본 구조 – 엔티티 설계

엔티티는 애플리케이션의 기본 구조를 설계합니다. 다음은 Photoblog 응용 프로그램의 엔티티입니다-

  • Film
  • Photo
  • Album

다음은 엔티티 관계에 대한 기본 클래스 다이어그램입니다-

디자인 구조

이전 장에서 논의했듯이 프로젝트의 설계 구조는 다음 스크린 샷과 같습니다.

Photoblog 애플리케이션에 대한 하위 디렉토리가있는 주어진 애플리케이션을 고려하십시오. 하위 디렉토리는 controllers.py, models.py 및 server.py를 포함하는 Photo, Album 및 Film입니다.

기능적으로 Photoblog 애플리케이션은 기존 CRUD 인터페이스 (생성, 검색, 업데이트 및 삭제)를 통해 해당 엔티티를 조작하는 API를 제공합니다.

데이터베이스에 연결

스토리지 모듈에는 일련의 작업이 포함됩니다. 작업 중 하나 인 데이터베이스와의 연결.

완전한 애플리케이션이기 때문에 API와 생성, 검색, 업데이트 및 삭제 기능을 유지하려면 데이터베이스와의 연결이 필수입니다.

import dejavu

arena = dejavu.Arena()
from model import Album, Film, Photo
def connect():

conf = {'Connect': "host=localhost dbname=Photoblog user=test password=test"}
arena.add_store("main", "postgres", conf)
arena.register_all(globals())

위 코드의 영역은 기본 스토리지 관리자와 비즈니스 로직 계층 간의 인터페이스가됩니다.

연결 기능은 PostgreSQL RDBMS 용 아레나 객체에 스토리지 관리자를 추가합니다.

연결이 이루어지면 비즈니스 요구 사항에 따라 양식을 만들고 응용 프로그램 작업을 완료 할 수 있습니다.

응용 프로그램을 만들기 전에 가장 중요한 것은 entity mapping 및 응용 프로그램의 구조 설계.

테스트는 응용 프로그램이 다음을 위해 다른 관점에서 수행되는 프로세스입니다.

  • 문제 목록 찾기
  • 예상 결과와 실제 결과, 출력, 상태 등의 차이점을 찾습니다.
  • 구현 단계를 이해합니다.
  • 현실적인 목적에 유용한 응용 프로그램을 찾으십시오.

테스트의 목표는 개발자를 잘못하는 것이 아니라 도구를 제공하고 주어진 시간에 애플리케이션의 상태를 추정하기위한 품질을 개선하는 것입니다.

테스트는 미리 계획해야합니다. 이를 위해서는 테스트의 목적을 정의하고, 테스트 사례의 범위를 이해하고, 비즈니스 요구 사항 목록을 작성하고, 프로젝트의 여러 단계에 관련된 위험을 인식해야합니다.

테스트는 시스템 또는 애플리케이션에서 검증 할 다양한 측면으로 정의됩니다. 다음은 목록입니다common test approaches

  • Unit testing− 이는 일반적으로 개발자가 직접 수행합니다. 이것은 코드 단위가 예상대로 작동하는지 여부를 확인하는 것을 목표로합니다.

  • Usability testing− 개발자는 일반적으로 시스템에 대한 지식이없는 최종 사용자를 위해 애플리케이션을 작성하고 있다는 사실을 잊을 수 있습니다. 사용성 테스트는 제품의 장단점을 확인합니다.

  • Functional/Acceptance testing − 사용성 테스트는 애플리케이션 또는 시스템의 사용 가능 여부를 확인하는 반면, 기능 테스트는 지정된 모든 기능이 구현되었는지 확인합니다.

  • Load and performance testing− 이는 시스템이 수행 할 부하 및 성능 테스트에 적응할 수 있는지 이해하기 위해 수행됩니다. 이는 하드웨어 변경, SQL 쿼리 최적화 등으로 이어질 수 있습니다.

  • Regression testing − 제품의 연속 릴리스가 이전 기능을 손상시키지 않는지 확인합니다.

  • Reliability and resilience testing − 신뢰성 테스트는 하나 또는 여러 구성 요소를 분석하여 시스템 애플리케이션을 검증하는 데 도움이됩니다.

단위 테스트

Photoblog 응용 프로그램은 지속적으로 단위 테스트를 사용하여 다음을 확인합니다.

  • 새로운 기능이 올바르게 작동합니다.
  • 기존 기능은 새 코드 릴리스로 인해 손상되지 않습니다.
  • 결함이 수정되고 수정 된 상태로 유지됩니다.

Python은 단위 테스트에 대한 다른 접근 방식을 제공하는 표준 단위 테스트 모듈과 함께 제공됩니다.

Unittest

unittest는 Kent Beck과 Erich Gamma가 개발 한 Java 단위 테스트 패키지 인 JUnit에 기반을두고 있습니다. 단위 테스트는 단순히 정의 된 데이터를 반환합니다. 모의 객체를 정의 할 수 있습니다. 이러한 개체를 사용하면 전체 애플리케이션에 의존하지 않고도 디자인의 인터페이스에 대해 테스트 할 수 있습니다. 또한 다른 테스트가 포함 된 격리 모드에서 테스트를 실행하는 방법을 제공합니다.

다음과 같은 방법으로 더미 클래스를 정의 해 봅시다.

import unittest

class DummyTest(unittest.TestCase):
def test_01_forward(self):
dummy = Dummy(right_boundary=3)
   self.assertEqual(dummy.forward(), 1)
   self.assertEqual(dummy.forward(), 2)
   self.assertEqual(dummy.forward(), 3)
   self.assertRaises(ValueError, dummy.forward)

def test_02_backward(self):
dummy = Dummy(left_boundary=-3, allow_negative=True)
   self.assertEqual(dummy.backward(), -1)
   self.assertEqual(dummy.backward(), -2)
   self.assertEqual(dummy.backward(), -3)
   self.assertRaises(ValueError, dummy.backward)

def test_03_boundaries(self):
dummy = Dummy(right_boundary=3, left_boundary=-3,allow_negative=True)
   self.assertEqual(dummy.backward(), -1)
   self.assertEqual(dummy.backward(), -2)
   self.assertEqual(dummy.forward(), -1)
   self.assertEqual(dummy.backward(), -2)
   self.assertEqual(dummy.backward(), -3)

코드에 대한 설명은 다음과 같습니다.

  • 주어진 클래스에 대한 단위 테스트 기능을 제공하려면 unittest 모듈을 가져와야합니다.

  • unittest를 서브 클래 싱하여 클래스를 생성해야합니다.

  • 위 코드의 모든 방법은 테스트 단어로 시작합니다. 이 모든 메소드는 unittest 핸들러에 의해 호출됩니다.

  • assert / fail 메서드는 예외를 관리하기 위해 테스트 케이스에서 호출됩니다.

이것을 테스트 케이스를 실행하는 예제로 고려하십시오-

if __name__ == '__main__':
unittest.main()

테스트 케이스를 실행 한 결과 (출력)는 다음과 같습니다.

----------------------------------------------------------------------
Ran 3 tests in 0.000s
OK

기능 테스트

애플리케이션 기능이 요구 사항에 따라 형성되기 시작하면 일련의 기능 테스트를 통해 사양과 관련된 애플리케이션의 정확성을 검증 할 수 있습니다. 그러나 Selenium과 같은 타사 제품을 사용해야하는 성능 향상을 위해 테스트를 자동화해야합니다.

CherryPy는 기능 테스트 작성을 쉽게하기 위해 내장 함수와 같은 도우미 클래스를 제공합니다.

부하 테스트

작성중인 애플리케이션과 볼륨 측면에서 기대하는 바에 따라 특정 수준의 성능에 도달하지 못하게하는 애플리케이션의 잠재적 병목 현상을 감지하기 위해로드 및 성능 테스트를 실행해야 할 수 있습니다.

이 섹션에서는 FunkLoad 패키지에 포함되지 않은 성능 또는 부하 테스트를 수행하는 방법에 대해 자세히 설명하지 않습니다.

FunkLoad의 가장 기본적인 예는 다음과 같습니다.

from funkload.FunkLoadTestCase 
import FunkLoadTestCase

class LoadHomePage(FunkLoadTestCase):
def test_homepage(self):

server_url = self.conf_get('main', 'url')
nb_time = self.conf_getInt('test_homepage', 'nb_time')
home_page = "%s/" % server_url

for i in range(nb_time):
self.logd('Try %i' % i)
self.get(home_page, description='Get gome page')
if __name__ in ('main', '__main__'):

import unittest

unittest.main()

다음은 위 코드에 대한 자세한 설명입니다.

  • 테스트 케이스는 FunkLoadTestCase 클래스에서 상속 받아야 FunkLoad가 테스트 중에 일어나는 일을 추적하는 내부 작업을 수행 할 수 있습니다.

  • FunkLoad는 클래스 이름을 기반으로 파일을 검색하므로 클래스 이름이 중요합니다.

  • 설계된 테스트 케이스는 구성 파일에 직접 액세스 할 수 있습니다. Get () 및 post () 메서드는 응답을 얻기 위해 서버에 대해 간단히 호출됩니다.

이 장에서는 내장 된 CherryPy HTTP 서버를 통해 활성화 된 CherryPy 기반 애플리케이션 SSL에 대해 자세히 설명합니다.

구성

웹 애플리케이션에 필요한 다양한 수준의 구성 설정이 있습니다.

  • Web server − HTTP 서버에 연결된 설정

  • Engine − 엔진 호스팅과 관련된 설정

  • Application − 사용자가 사용하는 애플리케이션

전개

CherryPy 애플리케이션의 배포는 Python 시스템 경로에서 필요한 모든 패키지를 사용할 수있는 매우 쉬운 방법으로 간주됩니다. 공유 웹 호스팅 환경에서 웹 서버는 호스트 공급자가 필터링 작업을 수행 할 수 있도록 프런트 엔드에 상주합니다. 프런트 엔드 서버는 Apache 또는lighttpd.

이 섹션에서는 Apache 및 lighttpd 웹 서버 뒤에서 CherryPy 애플리케이션을 실행하기위한 몇 가지 솔루션을 제공합니다.

cherrypy
def setup_app():

class Root:
@cherrypy.expose
def index(self):
   # Return the hostname used by CherryPy and the remote
   # caller IP address
	
return "Hello there %s from IP: %s " %
(cherrypy.request.base, cherrypy.request.remote.ip)
cherrypy.config.update({'server.socket_port': 9091,
   'environment': 'production',
   'log.screen': False,
   'show_tracebacks': False})
	
cherrypy.tree.mount(Root())
if __name__ == '__main__':

setup_app()
cherrypy.server.quickstart()
cherrypy.engine.start()

SSL

SSL (Secure Sockets Layer)CherryPy 기반 애플리케이션에서 지원 될 수 있습니다. SSL 지원을 활성화하려면 다음 요구 사항을 충족해야합니다.

  • 사용자 환경에 PyOpenSSL 패키지를 설치하십시오.
  • 서버에 SSL 인증서와 개인 키가 있어야합니다.

인증서 및 개인 키 만들기

인증서 및 개인 키의 요구 사항을 다루겠습니다.

  • 먼저 사용자는 개인 키가 필요합니다.
openssl genrsa -out server.key 2048
  • 이 키는 암호로 보호되지 않으므로 보호 기능이 약합니다.
  • 다음 명령이 발행됩니다-
openssl genrsa -des3 -out server.key 2048
  • 프로그램에는 암호가 필요합니다. OpenSSL 버전에서 빈 문자열을 제공 할 수 있다면 그렇게하십시오. 그렇지 않으면 기본 암호를 입력하고 다음과 같이 생성 된 키에서 제거합니다.

openssl rsa -in server.key -out server.key
  • 인증서 생성은 다음과 같습니다.
openssl req -new -key server.key -out server.csr
  • 이 프로세스는 몇 가지 세부 정보를 입력하도록 요청합니다. 이렇게하려면 다음 명령을 실행해야합니다.

openssl x509 -req -days 60 -in server.csr -signkey
server.key -out server.crt
  • 새로 서명 된 인증서는 60 일 동안 유효합니다.

다음 코드는 구현을 보여줍니다-

import cherrypy
import os, os.path

localDir = os.path.abspath(os.path.dirname(__file__))
CA = os.path.join(localDir, 'server.crt')
KEY = os.path.join(localDir, 'server.key')
def setup_server():

class Root:
@cherrypy.expose
def index(self):
   return "Hello there!"
	
cherrypy.tree.mount(Root())
if __name__ == '__main__':

setup_server()
cherrypy.config.update({'server.socket_port': 8443,
   'environment': 'production',
   'log.screen': True,
   'server.ssl_certificate': CA,
   'server.ssl_private_key': KEY})
	
cherrypy.server.quickstart()
cherrypy.engine.start()

다음 단계는 서버를 시작하는 것입니다. 성공하면 화면에 다음 메시지가 표시됩니다.

HTTP Serving HTTPS on https://localhost:8443/