blob: 45471b5b741a19515b5d0caea2dd01dd3e9fce25 [file] [log] [blame]
Elijah Newrenbc5c5ec2023-05-16 06:33:57 +00001#include "git-compat-util.h"
Elijah Newrenf394e092023-03-21 06:25:54 +00002#include "gettext.h"
Jeff Hostetlerd2bd8622022-03-25 18:02:45 +00003#include "simple-ipc.h"
4#include "fsmonitor-ipc.h"
Elijah Newrend1cbe1e2023-04-22 20:17:20 +00005#include "repository.h"
Jeff Hostetlerd2bd8622022-03-25 18:02:45 +00006#include "run-command.h"
7#include "strbuf.h"
8#include "trace2.h"
9
10#ifndef HAVE_FSMONITOR_DAEMON_BACKEND
11
12/*
13 * A trivial implementation of the fsmonitor_ipc__ API for unsupported
14 * platforms.
15 */
16
17int fsmonitor_ipc__is_supported(void)
18{
19 return 0;
20}
21
Jeff King4cb5e0b2023-09-18 18:32:38 -040022const char *fsmonitor_ipc__get_path(struct repository *r UNUSED)
Jeff Hostetlerd2bd8622022-03-25 18:02:45 +000023{
24 return NULL;
25}
26
27enum ipc_active_state fsmonitor_ipc__get_state(void)
28{
29 return IPC_STATE__OTHER_ERROR;
30}
31
Jeff King4cb5e0b2023-09-18 18:32:38 -040032int fsmonitor_ipc__send_query(const char *since_token UNUSED,
33 struct strbuf *answer UNUSED)
Jeff Hostetlerd2bd8622022-03-25 18:02:45 +000034{
35 return -1;
36}
37
Jeff King4cb5e0b2023-09-18 18:32:38 -040038int fsmonitor_ipc__send_command(const char *command UNUSED,
39 struct strbuf *answer UNUSED)
Jeff Hostetlerd2bd8622022-03-25 18:02:45 +000040{
41 return -1;
42}
43
44#else
45
46int fsmonitor_ipc__is_supported(void)
47{
48 return 1;
49}
50
Jeff Hostetlerd2bd8622022-03-25 18:02:45 +000051enum ipc_active_state fsmonitor_ipc__get_state(void)
52{
Eric DeCosta6beb2682022-10-04 17:32:27 +000053 return ipc_get_active_state(fsmonitor_ipc__get_path(the_repository));
Jeff Hostetlerd2bd8622022-03-25 18:02:45 +000054}
55
56static int spawn_daemon(void)
57{
René Scharfed82dbbd2022-10-30 12:52:40 +010058 struct child_process cmd = CHILD_PROCESS_INIT;
Jeff Hostetlerd2bd8622022-03-25 18:02:45 +000059
René Scharfed82dbbd2022-10-30 12:52:40 +010060 cmd.git_cmd = 1;
61 cmd.no_stdin = 1;
62 cmd.trace2_child_class = "fsmonitor";
63 strvec_pushl(&cmd.args, "fsmonitor--daemon", "start", NULL);
64
65 return run_command(&cmd);
Jeff Hostetlerd2bd8622022-03-25 18:02:45 +000066}
67
68int fsmonitor_ipc__send_query(const char *since_token,
69 struct strbuf *answer)
70{
71 int ret = -1;
72 int tried_to_spawn = 0;
73 enum ipc_active_state state = IPC_STATE__OTHER_ERROR;
74 struct ipc_client_connection *connection = NULL;
75 struct ipc_client_connect_options options
76 = IPC_CLIENT_CONNECT_OPTIONS_INIT;
77 const char *tok = since_token ? since_token : "";
78 size_t tok_len = since_token ? strlen(since_token) : 0;
79
80 options.wait_if_busy = 1;
81 options.wait_if_not_found = 0;
82
83 trace2_region_enter("fsm_client", "query", NULL);
84 trace2_data_string("fsm_client", NULL, "query/command", tok);
85
86try_again:
Eric DeCosta6beb2682022-10-04 17:32:27 +000087 state = ipc_client_try_connect(fsmonitor_ipc__get_path(the_repository),
88 &options, &connection);
Jeff Hostetlerd2bd8622022-03-25 18:02:45 +000089
90 switch (state) {
91 case IPC_STATE__LISTENING:
92 ret = ipc_client_send_command_to_connection(
93 connection, tok, tok_len, answer);
94 ipc_client_close_connection(connection);
95
96 trace2_data_intmax("fsm_client", NULL,
97 "query/response-length", answer->len);
98 goto done;
99
100 case IPC_STATE__NOT_LISTENING:
101 case IPC_STATE__PATH_NOT_FOUND:
102 if (tried_to_spawn)
103 goto done;
104
105 tried_to_spawn++;
106 if (spawn_daemon())
107 goto done;
108
109 /*
110 * Try again, but this time give the daemon a chance to
111 * actually create the pipe/socket.
112 *
113 * Granted, the daemon just started so it can't possibly have
114 * any FS cached yet, so we'll always get a trivial answer.
115 * BUT the answer should include a new token that can serve
116 * as the basis for subsequent requests.
117 */
118 options.wait_if_not_found = 1;
119 goto try_again;
120
121 case IPC_STATE__INVALID_PATH:
122 ret = error(_("fsmonitor_ipc__send_query: invalid path '%s'"),
Eric DeCosta6beb2682022-10-04 17:32:27 +0000123 fsmonitor_ipc__get_path(the_repository));
Jeff Hostetlerd2bd8622022-03-25 18:02:45 +0000124 goto done;
125
126 case IPC_STATE__OTHER_ERROR:
127 default:
128 ret = error(_("fsmonitor_ipc__send_query: unspecified error on '%s'"),
Eric DeCosta6beb2682022-10-04 17:32:27 +0000129 fsmonitor_ipc__get_path(the_repository));
Jeff Hostetlerd2bd8622022-03-25 18:02:45 +0000130 goto done;
131 }
132
133done:
134 trace2_region_leave("fsm_client", "query", NULL);
135
136 return ret;
137}
138
139int fsmonitor_ipc__send_command(const char *command,
140 struct strbuf *answer)
141{
142 struct ipc_client_connection *connection = NULL;
143 struct ipc_client_connect_options options
144 = IPC_CLIENT_CONNECT_OPTIONS_INIT;
145 int ret;
146 enum ipc_active_state state;
147 const char *c = command ? command : "";
148 size_t c_len = command ? strlen(command) : 0;
149
150 strbuf_reset(answer);
151
152 options.wait_if_busy = 1;
153 options.wait_if_not_found = 0;
154
Eric DeCosta6beb2682022-10-04 17:32:27 +0000155 state = ipc_client_try_connect(fsmonitor_ipc__get_path(the_repository),
156 &options, &connection);
Jeff Hostetlerd2bd8622022-03-25 18:02:45 +0000157 if (state != IPC_STATE__LISTENING) {
158 die(_("fsmonitor--daemon is not running"));
159 return -1;
160 }
161
162 ret = ipc_client_send_command_to_connection(connection, c, c_len,
163 answer);
164 ipc_client_close_connection(connection);
165
166 if (ret == -1) {
167 die(_("could not send '%s' command to fsmonitor--daemon"), c);
168 return -1;
169 }
170
171 return 0;
172}
173
174#endif