моя контактная информация
Почтамезофия@protonmail.com
2024-07-12
한어Русский языкEnglishFrançaisIndonesianSanskrit日本語DeutschPortuguêsΕλληνικάespañolItalianoSuomalainenLatina
start_response
Метод анализаВ этой статье мы подробно разберем WSGI-сервер. start_response
метод.Этот метод отвечает за обработку кода состояния и заголовков ответа HTTP и возвратwrite
Функция, используемая для отправки данных ответа. Мы объясним, как работает этот метод, построчно и предоставим некоторую информацию, которая поможет понять его возможности.
WSGI (Интерфейс шлюза веб-сервера) — это стандартный интерфейс, используемый для подключения веб-сервера к веб-приложению или платформе. Через WSGI вы можете обмениваться данными между сервером и приложением и обрабатывать HTTP-запросы и ответы.
При реализации сервера WSGIstart_response
Этот метод является ключевой частью, которая вызывается приложением WSGI для установки состояния ответа HTTP и заголовков ответа, а также возвращает функцию для записи данных ответа.
start_response
Реализация методаНиже приводится типичный start_response
Реализация метода:
def start_response(status, response_headers, exc_info=None):
if exc_info:
try:
if headers_sent:
reraise(*exc_info)
finally:
exc_info = None
elif headers_set:
raise AssertionError("Headers already set")
headers_set[:] = [status, response_headers]
return write
exc_info
параметрif exc_info:
try:
if headers_sent:
reraise(*exc_info)
finally:
exc_info = None
exc_info
Аргумент обычно содержит кортеж исключений в формате(type, value, traceback)
, используемый для обработки исключений.еслиexc_info
Не дляNone
, указывая на наличие исключения, которое необходимо обработать.try
Внутриблочные проверкиheaders_sent
Был ли отправлен заголовок ответа.Если заголовки ответа были отправлены, вызовитеreraise(*exc_info)
Повторно создайте исключение.finally
блок гарантирует, что после обработки исключенияexc_info
установлен вNone
во избежание дублирования обработки.elif headers_set:
raise AssertionError("Headers already set")
headers_set
Список уже содержит значения (т.е. заголовок ответа задан), поднимаетAssertionError
аномальный.Эта проверка гарантируетstart_response
Может быть вызван только один раз для установки заголовков ответа.headers_set[:] = [status, response_headers]
status
иresponse_headers
назначен наheaders_set
список.Целью этого является сохранениеheaders_set
ссылку, но обновляет ее содержимое.status
— это строка, представляющая код состояния и сообщение HTTP-ответа, например"200 OK"
。response_headers
представляет собой список, содержащий кортежи, каждый кортеж представляет собой пару ключ-значение заголовка ответа, например[("Content-Type", "text/html"), ("Content-Length", "123")]
。write
функцияreturn write
start_response
Метод возвращаетwrite
функция. Эта функция вызывается приложением WSGI для отправки данных ответа.Вот полный пример, показывающий, как использовать start_response
Методы обработки ответов WSGI:
import socketserver
class MyTCPHandler(socketserver.BaseRequestHandler):
def handle(self):
self.headers_set = []
self.headers_sent = []
def start_response(status, response_headers, exc_info=None):
if exc_info:
try:
if self.headers_sent:
raise exc_info[1]
finally:
exc_info = None
elif self.headers_set:
raise AssertionError("Headers already set")
self.headers_set[:] = [status, response_headers]
return self.write
def write(data):
assert self.headers_set, "write() before start_response"
if not self.headers_sent:
status, response_headers = self.headers_sent[:] = self.headers_set
try:
code, msg = status.split(None, 1)
except ValueError:
code, msg = status, ""
code = int(code)
self.send_response(code, msg)
header_keys = set()
for key, value in response_headers:
self.send_header(key, value)
key = key.lower()
header_keys.add(key)
if not (
"content-length" in header_keys
or self.environ["REQUEST_METHOD"] == "HEAD"
or code < 200
or code in (204, 304)
):
self.close_connection = True
self.send_header("Connection", "close")
if "server" not in header_keys:
self.send_header("Server", self.version_string())
if "date" not in header_keys:
self.send_header("Date", self.date_time_string())
self.end_headers()
assert isinstance(data, bytes), "applications must write bytes"
self.wfile.write(data)
self.wfile.flush()
self.write = write
self.environ = self.make_environ()
try:
result = self.server.app(self.environ, start_response)
try:
for data in result:
write(data)
if not self.headers_sent:
write(b"")
finally:
if hasattr(result, "close"):
result.close()
except Exception as e:
self.send_error(500, str(e))
if __name__ == "__main__":
HOST, PORT = "localhost", 9999
with socketserver.TCPServer((HOST, PORT), MyTCPHandler) as server:
print("Server started at {}:{}".format(HOST, PORT))
server.serve_forever()
В этом руководстве мы подробно проанализировали сервер WSGI. start_response
Метод, который объясняет, как он обрабатывает код состояния и заголовки ответа HTTP, а также возвращает функцию для записи данных ответа. Понимание этого содержания поможет вам лучше понять спецификацию WSGI и реализовать индивидуальный сервер WSGI. Надеюсь, этот урок будет вам полезен.Более подробную информацию и примеры см.Официальная документация。