Какова цель закрытия сокета в этом примере Common Lisp?

Я нашел этот пример из Поваренной книги Common Lisp, который показывает, как запустить TCP-сервер с usocket.

В этом примере создается объект сокета и устанавливается соединение, а затем выполняется запись в сокет. В случае ошибки запись сокета оборачивается защитой от размотки, которая закрывает сокет, чтобы его можно было использовать повторно. Я переписал пример, чтобы вызвать ошибку, но когда я запускаю его несколько раз, я получаюUSOCKET:ADDRESS-IN-USE-ERROR. Поведение будет таким же, если я удалюsocket-close вызовы функций.

(load "~/quicklisp/setup.lisp")
(ql:quickload "usocket")

(let* ((socket (usocket:socket-listen "localhost" 8080))
       (connection (usocket:socket-accept socket)))
        (unwind-protect
          (progn
            (error "error 1"))
          (progn
            (usocket:socket-close connection)
            (usocket:socket-close socket)
            (print "Error clean up"))))
Unhandled USOCKET:ADDRESS-IN-USE-ERROR in thread #<SB-THREAD:THREAD "main thread" RUNNING
                                                    {10005E85B3}>:
  Condition USOCKET:ADDRESS-IN-USE-ERROR was signalled.

Backtrace for: #<SB-THREAD:THREAD "main thread" RUNNING {10005E85B3}>
0: (SB-DEBUG::DEBUGGER-DISABLED-HOOK #<USOCKET:ADDRESS-IN-USE-ERROR {1003FDC063}> #<unused argument> :QUIT T)
1: (SB-DEBUG::RUN-HOOK *INVOKE-DEBUGGER-HOOK* #<USOCKET:ADDRESS-IN-USE-ERROR {1003FDC063}>)
2: (INVOKE-DEBUGGER #<USOCKET:ADDRESS-IN-USE-ERROR {1003FDC063}>)
3: (ERROR #<USOCKET:ADDRESS-IN-USE-ERROR {1003FDC063}>)
4: (USOCKET:SOCKET-LISTEN "localhost" 8080 :REUSEADDRESS NIL :REUSE-ADDRESS NIL :BACKLOG 5 :ELEMENT-TYPE CHARACTER)
5: ((LAMBDA NIL :IN "/home/sam/test/serve.lisp"))
6: (SB-INT:SIMPLE-EVAL-IN-LEXENV (LET* ((SOCKET (USOCKET:SOCKET-LISTEN "localhost" 8080)) (CONNECTION (USOCKET:SOCKET-ACCEPT SOCKET))) (UNWIND-PROTECT (PROGN (ERROR "error 1")) (PROGN (USOCKET:SOCKET-CLOSE CONNECTION) (USOCKET:SOCKET-CLOSE SOCKET) (PRINT "Error clean up")))) #<NULL-LEXENV>)
7: (EVAL-TLF (LET* ((SOCKET (USOCKET:SOCKET-LISTEN "localhost" 8080)) (CONNECTION (USOCKET:SOCKET-ACCEPT SOCKET))) (UNWIND-PROTECT (PROGN (ERROR "error 1")) (PROGN (USOCKET:SOCKET-CLOSE CONNECTION) (USOCKET:SOCKET-CLOSE SOCKET) (PRINT "Error clean up")))) 2 NIL)
8: ((LABELS SB-FASL::EVAL-FORM :IN SB-INT:LOAD-AS-SOURCE) (LET* ((SOCKET (USOCKET:SOCKET-LISTEN "localhost" 8080)) (CONNECTION (USOCKET:SOCKET-ACCEPT SOCKET))) (UNWIND-PROTECT (PROGN (ERROR "error 1")) (PROGN (USOCKET:SOCKET-CLOSE CONNECTION) (USOCKET:SOCKET-CLOSE SOCKET) (PRINT "Error clean up")))) 2)
9: ((LAMBDA (SB-KERNEL:FORM &KEY :CURRENT-INDEX &ALLOW-OTHER-KEYS) :IN SB-INT:LOAD-AS-SOURCE) (LET* ((SOCKET (USOCKET:SOCKET-LISTEN "localhost" 8080)) (CONNECTION (USOCKET:SOCKET-ACCEPT SOCKET))) (UNWIND-PROTECT (PROGN (ERROR "error 1")) (PROGN (USOCKET:SOCKET-CLOSE CONNECTION) (USOCKET:SOCKET-CLOSE SOCKET) (PRINT "Error clean up")))) :CURRENT-INDEX 2)
10: (SB-C::%DO-FORMS-FROM-INFO #<CLOSURE (LAMBDA (SB-KERNEL:FORM &KEY :CURRENT-INDEX &ALLOW-OTHER-KEYS) :IN SB-INT:LOAD-AS-SOURCE) {1001B7128B}> #<SB-C::SOURCE-INFO {1001B71243}> SB-C::INPUT-ERROR-IN-LOAD)
11: (SB-INT:LOAD-AS-SOURCE #<SB-SYS:FD-STREAM for "file /home/sam/test/serve.lisp" {1001B66763}> :VERBOSE NIL :PRINT NIL :CONTEXT "loading")
12: ((FLET SB-FASL::THUNK :IN LOAD))
13: (SB-FASL::CALL-WITH-LOAD-BINDINGS #<CLOSURE (FLET SB-FASL::THUNK :IN LOAD) {7FFFF63E769B}> #<SB-SYS:FD-STREAM for "file /home/sam/test/serve.lisp" {1001B66763}>)
14: ((FLET SB-FASL::LOAD-STREAM :IN LOAD) #<SB-SYS:FD-STREAM for "file /home/sam/test/serve.lisp" {1001B66763}> NIL)
15: (LOAD #<SB-SYS:FD-STREAM for "file /home/sam/test/serve.lisp" {1001B66763}> :VERBOSE NIL :PRINT NIL :IF-DOES-NOT-EXIST T :EXTERNAL-FORMAT :DEFAULT)
16: ((FLET SB-IMPL::LOAD-SCRIPT :IN SB-IMPL::PROCESS-SCRIPT) #<SB-SYS:FD-STREAM for "file /home/sam/test/serve.lisp" {1001B66763}>)
17: ((FLET SB-UNIX::BODY :IN SB-IMPL::PROCESS-SCRIPT))
18: ((FLET "WITHOUT-INTERRUPTS-BODY-3" :IN SB-IMPL::PROCESS-SCRIPT))
19: (SB-IMPL::PROCESS-SCRIPT "serve.lisp")
20: (SB-IMPL::TOPLEVEL-INIT)
21: ((FLET SB-UNIX::BODY :IN SAVE-LISP-AND-DIE))
22: ((FLET "WITHOUT-INTERRUPTS-BODY-36" :IN SAVE-LISP-AND-DIE))
23: ((LABELS SB-IMPL::RESTART-LISP :IN SAVE-LISP-AND-DIE))

unhandled condition in --disable-debugger mode, quitting

1 ответ

Решение

Причина, по которой вы получаете это, заключается в природе протокола TCP: соединение находится в состоянии, называемом TIME-WAIT в конечном автомате TCP, описанном в RFC793. Схема конечного автомата приведена на странице 23 RFC793.

Интересная часть конечного автомата - это когда одна сторона (которую я назову "ты") хочет закрыть соединение - это называется "активное закрытие", и в данном случае это то, что вы инициируете socket-closeзвонки. Я назову другой конец "они". Обычная последовательность событий для активного закрытия:

  1. вы отправляете им пакет FIN;
  2. они подтверждают ваш FIN и, в свою очередь, отправляют FIN;
  3. ВЫ ПОДТВЕРЖДАЕТЕ их FIN.

Теперь важно помнить, что любой из этих пакетов (их ACK и FIN обычно являются одним и тем же пакетом, и я думаю, что всегда) может быть потерян, и конечный автомат должен восстановиться после этого.

Есть один особенно интересный пакет, который является последним ACK: он особенно интересен, потому что это последний когда-либо отправленный пакет, а это означает, что у вас нет возможности узнать, дошел ли он до них.

Итак, рассмотрим ситуацию с обоих концов

С их стороны: они отправили FIN и ждут вашего ACK на него. Сейчас:

  • либо ACK прибывает в должное время, и в этом случае они знают, что все заведено, и могут демонтировать все, что связано с подключением.
  • или, после ожидания в течение установленного времени, ACK не приходит, поэтому они должны предположить, что либо их FIN был потерян, либо ваш ACK для их FIN был потерян, и поэтому они должны повторно отправить FIN и вернуться к ожиданию ACK.

С вашей стороны: вы получили их FIN и отправили последний ACK. Но вы понятия не имеете, дошел ли до них этот ACK. Таким образом, вы ждете установленное время, чтобы дать им возможность понять, что ACK не получил их, и повторно отправить их FIN. Во время этого ожидания вы не можете разорвать соединение, потому что в любой момент вы можете получить еще один FIN.

Это состояние ожидания известно как TIME-WAIT, и во время него конечную точку соединения нельзя использовать повторно. И это проблема, которую вы видите.

Вам нужно сидеть в режиме TIME-WAIT в течение вдвое большего максимального времени жизни сегмента (MSL): MSL - это время, в течение которого пакет может находиться в сети.

Есть и другие состояния ожидания, которые, конечно, могут возникнуть до TIME-WAIT, если предыдущие пакеты теряются. Но TIME-WAIT - единственное, что происходит всегда.

TIME-WAIT часто называют TIME_WAIT из-за языков, которые не могут обрабатывать дефисы в именах.


Другие вопросы по тегам