aboutsummaryrefslogtreecommitdiff
path: root/src/feature/control/control_proto.h
blob: c95e1824a1ce857c846bcfb5c855c3038870df7b (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
/* Copyright (c) 2001 Matej Pfajfar.
 * Copyright (c) 2001-2004, Roger Dingledine.
 * Copyright (c) 2004-2006, Roger Dingledine, Nick Mathewson.
 * Copyright (c) 2007-2021, The Tor Project, Inc. */
/* See LICENSE for licensing information */

/**
 * \file control_proto.h
 * \brief Header file for control_proto.c.
 *
 * See @ref replylines for details about the key-value abstraction for
 * generating reply lines.
 **/

#ifndef TOR_CONTROL_PROTO_H
#define TOR_CONTROL_PROTO_H

#include "lib/encoding/confline.h"

/**
 * @defgroup replylines Control reply lines
 * @brief Key-value structures for control reply lines
 *
 * Control reply lines are config_line_t key-value structures with
 * some additional information to help formatting, such as the numeric
 * result code specified in the control protocol and flags affecting
 * the way kvline_encode() formats the @a kvline.
 *
 * Generally, modules implementing control commands will work with
 * smartlists of these structures, using functions like
 * control_reply_add_str() for adding a reply line consisting of a
 * single string, or control_reply_add_one_kv() and
 * control_reply_append_kv() for composing a line containing one or
 * more key-value pairs.
 *
 * @{
 */
/** @brief A reply line for the control protocol.
 *
 * This wraps config_line_t with some additional information that's
 * useful when generating control reply lines.
 */
typedef struct control_reply_line_t {
  int code;                     /**< numeric code */
  int flags;                    /**< kvline encoding flags */
  config_line_t *kvline;        /**< kvline */
} control_reply_line_t;

void control_reply_line_free_(control_reply_line_t *line);
/**
 * @brief Free and null a control_reply_line_t
 *
 * @param line pointer to control_reply_line_t to free
 */
#define control_reply_line_free(line)                   \
  FREE_AND_NULL(control_reply_line_t,                   \
                control_reply_line_free_, (line))
/** @} */

void connection_write_str_to_buf(const char *s, control_connection_t *conn);
void connection_printf_to_buf(control_connection_t *conn,
                                     const char *format, ...)
  CHECK_PRINTF(2,3);

size_t write_escaped_data(const char *data, size_t len, char **out);
size_t read_escaped_data(const char *data, size_t len, char **out);
void send_control_done(control_connection_t *conn);

MOCK_DECL(void, control_write_reply, (control_connection_t *conn, int code,
                                      int c, const char *s));
void control_vprintf_reply(control_connection_t *conn, int code, int c,
                           const char *fmt, va_list ap)
  CHECK_PRINTF(4, 0);
void control_write_endreply(control_connection_t *conn, int code,
                            const char *s);
void control_printf_endreply(control_connection_t *conn, int code,
                             const char *fmt, ...)
  CHECK_PRINTF(3, 4);
void control_write_midreply(control_connection_t *conn, int code,
                            const char *s);
void control_printf_midreply(control_connection_t *conn, int code,
                             const char *fmt,
                             ...)
  CHECK_PRINTF(3, 4);
void control_write_datareply(control_connection_t *conn, int code,
                             const char *s);
void control_printf_datareply(control_connection_t *conn, int code,
                              const char *fmt,
                              ...)
  CHECK_PRINTF(3, 4);
void control_write_data(control_connection_t *conn, const char *data);

/** @addtogroup replylines
 * @{
 */
void control_write_reply_line(control_connection_t *conn,
                              const control_reply_line_t *line, bool lastone);
void control_write_reply_lines(control_connection_t *conn, smartlist_t *lines);

void control_reply_add_one_kv(smartlist_t *reply, int code, int flags,
                              const char *key, const char *val);
void control_reply_append_kv(smartlist_t *reply, const char *key,
                             const char *val);
void control_reply_add_str(smartlist_t *reply, int code, const char *s);
void control_reply_add_printf(smartlist_t *reply, int code,
                              const char *fmt, ...)
  CHECK_PRINTF(3, 4);
void control_reply_add_done(smartlist_t *reply);

void control_reply_clear(smartlist_t *reply);
void control_reply_free_(smartlist_t *reply);

/** @brief Free and null a smartlist of control_reply_line_t.
 *
 * @param r pointer to smartlist_t of control_reply_line_t to free */
#define control_reply_free(r) \
  FREE_AND_NULL(smartlist_t, control_reply_free_, (r))
/** @} */

#endif /* !defined(TOR_CONTROL_PROTO_H) */