/* * Copyright (C) Igor Sysoev */ #include #include #include #include ngx_int_t ngx_pop3_parse_command(ngx_imap_session_t *s) { u_char ch, *p, *c; ngx_str_t *arg; enum { sw_start = 0, sw_spaces_before_argument, sw_argument, sw_almost_done, sw_done } state; state = s->state; p = s->buffer->pos; while (p < s->buffer->last && state < sw_done) { ch = *p++; switch (state) { /* POP3 command */ case sw_start: if (ch == ' ' || ch == CR || ch == LF) { c = s->buffer->start; if (p - 1 - c == 4) { if (c[0] == 'U' && c[1] == 'S' && c[2] == 'E' && c[3] == 'R') { s->command = NGX_POP3_USER; } else if (c[0] == 'P' && c[1] == 'A' && c[2] == 'S' && c[3] == 'S') { s->command = NGX_POP3_PASS; } else if (c[0] == 'Q' && c[1] == 'U' && c[2] == 'I' && c[3] == 'T') { s->command = NGX_POP3_QUIT; #if 0 } else if (c[0] == 'N' && c[1] == 'O' && c[2] == 'O' && c[3] == 'P') { s->command = NGX_POP3_NOOP; #endif } else { s->state = sw_start; return NGX_IMAP_PARSE_INVALID_COMMAND; } } else { s->state = sw_start; return NGX_IMAP_PARSE_INVALID_COMMAND; } switch (ch) { case ' ': state = sw_spaces_before_argument; break; case CR: state = sw_almost_done; break; case LF: state = sw_done; break; } break; } if (ch < 'A' || ch > 'Z') { s->state = sw_start; return NGX_IMAP_PARSE_INVALID_COMMAND; } break; /* the spaces before the argument */ case sw_spaces_before_argument: switch (ch) { case ' ': break; case CR: state = sw_almost_done; s->arg_end = p - 1; break; case LF: state = sw_done; s->arg_end = p - 1; break; default: if (s->args.nelts > 2) { s->state = sw_start; return NGX_IMAP_PARSE_INVALID_COMMAND; } state = sw_argument; s->arg_start = p - 1; break; } break; /* the argument */ case sw_argument: switch (ch) { case ' ': case CR: case LF: if (!(arg = ngx_array_push(&s->args))) { return NGX_ERROR; } arg->len = p - 1 - s->arg_start; arg->data = s->arg_start; s->arg_start = NULL; switch (ch) { case ' ': state = sw_spaces_before_argument; break; case CR: state = sw_almost_done; break; case LF: state = sw_done; break; } break; default: break; } break; /* end of request line */ case sw_almost_done: switch (ch) { case LF: state = sw_done; break; default: s->state = sw_start; return NGX_IMAP_PARSE_INVALID_COMMAND; } break; /* suppress warning */ case sw_done: break; } } s->buffer->pos = p; if (state == sw_done) { if (s->arg_start) { if (!(arg = ngx_array_push(&s->args))) { return NGX_ERROR; } arg->len = s->arg_end - s->arg_start; arg->data = s->arg_start; s->arg_start = NULL; } s->state = sw_start; return NGX_OK; } else { s->state = state; return NGX_AGAIN; } }