t5000: use check_mtime()
[git/debian.git] / fsmonitor-ipc.c
blob19d772f0f3ae3737d70e69837ff0d9aa3021da62
1 #include "cache.h"
2 #include "fsmonitor.h"
3 #include "simple-ipc.h"
4 #include "fsmonitor-ipc.h"
5 #include "run-command.h"
6 #include "strbuf.h"
7 #include "trace2.h"
9 #ifndef HAVE_FSMONITOR_DAEMON_BACKEND
12 * A trivial implementation of the fsmonitor_ipc__ API for unsupported
13 * platforms.
16 int fsmonitor_ipc__is_supported(void)
18 return 0;
21 const char *fsmonitor_ipc__get_path(struct repository *r)
23 return NULL;
26 enum ipc_active_state fsmonitor_ipc__get_state(void)
28 return IPC_STATE__OTHER_ERROR;
31 int fsmonitor_ipc__send_query(const char *since_token,
32 struct strbuf *answer)
34 return -1;
37 int fsmonitor_ipc__send_command(const char *command,
38 struct strbuf *answer)
40 return -1;
43 #else
45 int fsmonitor_ipc__is_supported(void)
47 return 1;
50 enum ipc_active_state fsmonitor_ipc__get_state(void)
52 return ipc_get_active_state(fsmonitor_ipc__get_path(the_repository));
55 static int spawn_daemon(void)
57 struct child_process cmd = CHILD_PROCESS_INIT;
59 cmd.git_cmd = 1;
60 cmd.no_stdin = 1;
61 cmd.trace2_child_class = "fsmonitor";
62 strvec_pushl(&cmd.args, "fsmonitor--daemon", "start", NULL);
64 return run_command(&cmd);
67 int fsmonitor_ipc__send_query(const char *since_token,
68 struct strbuf *answer)
70 int ret = -1;
71 int tried_to_spawn = 0;
72 enum ipc_active_state state = IPC_STATE__OTHER_ERROR;
73 struct ipc_client_connection *connection = NULL;
74 struct ipc_client_connect_options options
75 = IPC_CLIENT_CONNECT_OPTIONS_INIT;
76 const char *tok = since_token ? since_token : "";
77 size_t tok_len = since_token ? strlen(since_token) : 0;
79 options.wait_if_busy = 1;
80 options.wait_if_not_found = 0;
82 trace2_region_enter("fsm_client", "query", NULL);
83 trace2_data_string("fsm_client", NULL, "query/command", tok);
85 try_again:
86 state = ipc_client_try_connect(fsmonitor_ipc__get_path(the_repository),
87 &options, &connection);
89 switch (state) {
90 case IPC_STATE__LISTENING:
91 ret = ipc_client_send_command_to_connection(
92 connection, tok, tok_len, answer);
93 ipc_client_close_connection(connection);
95 trace2_data_intmax("fsm_client", NULL,
96 "query/response-length", answer->len);
97 goto done;
99 case IPC_STATE__NOT_LISTENING:
100 case IPC_STATE__PATH_NOT_FOUND:
101 if (tried_to_spawn)
102 goto done;
104 tried_to_spawn++;
105 if (spawn_daemon())
106 goto done;
109 * Try again, but this time give the daemon a chance to
110 * actually create the pipe/socket.
112 * Granted, the daemon just started so it can't possibly have
113 * any FS cached yet, so we'll always get a trivial answer.
114 * BUT the answer should include a new token that can serve
115 * as the basis for subsequent requests.
117 options.wait_if_not_found = 1;
118 goto try_again;
120 case IPC_STATE__INVALID_PATH:
121 ret = error(_("fsmonitor_ipc__send_query: invalid path '%s'"),
122 fsmonitor_ipc__get_path(the_repository));
123 goto done;
125 case IPC_STATE__OTHER_ERROR:
126 default:
127 ret = error(_("fsmonitor_ipc__send_query: unspecified error on '%s'"),
128 fsmonitor_ipc__get_path(the_repository));
129 goto done;
132 done:
133 trace2_region_leave("fsm_client", "query", NULL);
135 return ret;
138 int fsmonitor_ipc__send_command(const char *command,
139 struct strbuf *answer)
141 struct ipc_client_connection *connection = NULL;
142 struct ipc_client_connect_options options
143 = IPC_CLIENT_CONNECT_OPTIONS_INIT;
144 int ret;
145 enum ipc_active_state state;
146 const char *c = command ? command : "";
147 size_t c_len = command ? strlen(command) : 0;
149 strbuf_reset(answer);
151 options.wait_if_busy = 1;
152 options.wait_if_not_found = 0;
154 state = ipc_client_try_connect(fsmonitor_ipc__get_path(the_repository),
155 &options, &connection);
156 if (state != IPC_STATE__LISTENING) {
157 die(_("fsmonitor--daemon is not running"));
158 return -1;
161 ret = ipc_client_send_command_to_connection(connection, c, c_len,
162 answer);
163 ipc_client_close_connection(connection);
165 if (ret == -1) {
166 die(_("could not send '%s' command to fsmonitor--daemon"), c);
167 return -1;
170 return 0;
173 #endif