summaryrefslogtreecommitdiffabout
path: root/src/process.cc
authorMichael Krelin <hacker@klever.net>2008-08-31 11:50:48 (UTC)
committer Michael Krelin <hacker@klever.net>2008-08-31 11:50:48 (UTC)
commit6a68fcae465905e904eea248ffe6b1ed6b8297a4 (patch) (unidiff)
tree5f070058ca6cf9948ae9af229694d3d746d2f1c5 /src/process.cc
parent26757537328da204e4bf952a80579307440f62c9 (diff)
downloaddudki-6a68fcae465905e904eea248ffe6b1ed6b8297a4.zip
dudki-6a68fcae465905e904eea248ffe6b1ed6b8297a4.tar.gz
dudki-6a68fcae465905e904eea248ffe6b1ed6b8297a4.tar.bz2
don't break in the absence of sendmail
If for whatever reason notify child isn't properly spawned, don't die and still try to launch process in question Signed-off-by: Michael Krelin <hacker@klever.net>
Diffstat (limited to 'src/process.cc') (more/less context) (ignore whitespace changes)
-rw-r--r--src/process.cc7
1 files changed, 6 insertions, 1 deletions
diff --git a/src/process.cc b/src/process.cc
index 6d3b2a2..4807b98 100644
--- a/src/process.cc
+++ b/src/process.cc
@@ -109,147 +109,152 @@ void process::launch(const string& id,configuration& config) {
109 throw runtime_error(string(__PRETTY_FUNCTION__)+": failed to chroot()"); 109 throw runtime_error(string(__PRETTY_FUNCTION__)+": failed to chroot()");
110 } 110 }
111 if(!user.empty()) { 111 if(!user.empty()) {
112 if((getuid()!=uid) && setuid(uid)) 112 if((getuid()!=uid) && setuid(uid))
113 throw runtime_error(string(__PRETTY_FUNCTION__)+": failed to setuid()"); 113 throw runtime_error(string(__PRETTY_FUNCTION__)+": failed to setuid()");
114 } 114 }
115 char *argv[] = { "/bin/sh", "-c", (char*)restart_cmd.c_str(), NULL }; 115 char *argv[] = { "/bin/sh", "-c", (char*)restart_cmd.c_str(), NULL };
116 close(0); close(1); close(2); 116 close(0); close(1); close(2);
117 execv("/bin/sh",argv); 117 execv("/bin/sh",argv);
118 }catch(exception& e) { 118 }catch(exception& e) {
119 syslog(LOG_ERR,"Error trying to launch process '%s': %s",id.c_str(),e.what()); 119 syslog(LOG_ERR,"Error trying to launch process '%s': %s",id.c_str(),e.what());
120 } 120 }
121 _exit(-1); 121 _exit(-1);
122 } 122 }
123 // parent 123 // parent
124 int rv; 124 int rv;
125 if(waitpid(p,&rv,0)<0) 125 if(waitpid(p,&rv,0)<0)
126 throw runtime_error(string(__PRETTY_FUNCTION__)+": failed to waitpid()"); 126 throw runtime_error(string(__PRETTY_FUNCTION__)+": failed to waitpid()");
127} 127}
128 128
129void process::do_notify(const string& id,const string& event,const string& description,configuration& config) { 129void process::do_notify(const string& id,const string& event,const string& description,configuration& config) {
130 string the_notify; 130 string the_notify;
131 if(!notify.empty()) 131 if(!notify.empty())
132 the_notify=notify; 132 the_notify=notify;
133 else if(!config.notify.empty()) 133 else if(!config.notify.empty())
134 the_notify=config.notify; 134 the_notify=config.notify;
135 else 135 else
136 return; 136 return;
137 try { 137 try {
138 string::size_type colon = the_notify.find(':'); 138 string::size_type colon = the_notify.find(':');
139 if(colon==string::npos) 139 if(colon==string::npos)
140 throw runtime_error("invalid notify action specification"); 140 throw runtime_error("invalid notify action specification");
141 string nschema = the_notify.substr(0,colon); 141 string nschema = the_notify.substr(0,colon);
142 string ntarget = the_notify.substr(colon+1); 142 string ntarget = the_notify.substr(colon+1);
143 if(nschema=="mailto") { 143 if(nschema=="mailto") {
144 notify_mailto(ntarget,id,event,description,config); 144 notify_mailto(ntarget,id,event,description,config);
145 }else 145 }else
146 throw runtime_error("unrecognized notification schema"); 146 throw runtime_error("unrecognized notification schema");
147 }catch(exception& e) { 147 }catch(exception& e) {
148 syslog(LOG_ERR,"Notification error: %s",e.what()); 148 syslog(LOG_ERR,"Notification error: %s",e.what());
149 } 149 }
150} 150}
151 151
152void process::notify_mailto(const string& email,const string& id,const string& event,const string& description,configuration& config) { 152void process::notify_mailto(const string& email,const string& id,const string& event,const string& description,configuration& config) {
153 int files[2]; 153 int files[2];
154 if(pipe(files)) 154 if(pipe(files))
155 throw runtime_error("Failed to pipe()"); 155 throw runtime_error("Failed to pipe()");
156 pid_t pid = vfork(); 156 pid_t pid = vfork();
157 if(pid==-1) { 157 if(pid==-1) {
158 close(files[0]); 158 close(files[0]);
159 close(files[1]); 159 close(files[1]);
160 throw runtime_error("Failed to vfork()"); 160 throw runtime_error("Failed to vfork()");
161 } 161 }
162 if(!pid) { 162 if(!pid) {
163 // child 163 // child
164 if(dup2(files[0],0)!=0) 164 if(dup2(files[0],0)!=0)
165 _exit(-1); 165 _exit(-1);
166 close(1); 166 close(1);
167 close(files[0]); 167 close(files[0]);
168 close(files[1]); 168 close(files[1]);
169 execl("/usr/sbin/sendmail","usr/sbin/sendmail","-i",email.c_str(),NULL); 169 execl("/usr/sbin/sendmail","usr/sbin/sendmail","-i",email.c_str(),NULL);
170 _exit(-1); 170 _exit(-1);
171 } 171 }
172 // parent 172 // parent
173 int status;
174 if(waitpid(pid,&status,WNOHANG)) {
175 close(files[0]);
176 close(files[1]);
177 throw runtime_error("vfork()ed sendmail child exited unexpectedly");
178 }
173 close(files[0]); 179 close(files[0]);
174 FILE *mta = fdopen(files[1],"w"); 180 FILE *mta = fdopen(files[1],"w");
175 for(headers_t::const_iterator i=mailto_headers.begin();i!=mailto_headers.end();++i) { 181 for(headers_t::const_iterator i=mailto_headers.begin();i!=mailto_headers.end();++i) {
176 fprintf(mta,"%s: %s\n",i->first.c_str(),i->second.c_str()); 182 fprintf(mta,"%s: %s\n",i->first.c_str(),i->second.c_str());
177 } 183 }
178 for(headers_t::const_iterator i=config.mailto_headers.begin();i!=config.mailto_headers.end();++i) { 184 for(headers_t::const_iterator i=config.mailto_headers.begin();i!=config.mailto_headers.end();++i) {
179 if(mailto_headers.find(i->first)!=mailto_headers.end()) 185 if(mailto_headers.find(i->first)!=mailto_headers.end())
180 continue; 186 continue;
181 fprintf(mta,"%s: %s\n",i->first.c_str(),i->second.c_str()); 187 fprintf(mta,"%s: %s\n",i->first.c_str(),i->second.c_str());
182 } 188 }
183 fprintf(mta, 189 fprintf(mta,
184 "Subject: [%s] %s\n\n" 190 "Subject: [%s] %s\n\n"
185 "%s\n" 191 "%s\n"
186 "---\n" 192 "---\n"
187 "This message was sent automatically by the 'dudki' daemon\n", 193 "This message was sent automatically by the 'dudki' daemon\n",
188 id.c_str(), event.c_str(), 194 id.c_str(), event.c_str(),
189 description.c_str() ); 195 description.c_str() );
190 fclose(mta); 196 fclose(mta);
191 int status;
192 waitpid(pid,&status,0); 197 waitpid(pid,&status,0);
193 // TODO: check the return code 198 // TODO: check the return code
194} 199}
195 200
196void process::signal(int signum) const { 201void process::signal(int signum) const {
197 if(!pidfile.empty()) { 202 if(!pidfile.empty()) {
198 ifstream pids(pidfile.c_str(),ios::in); 203 ifstream pids(pidfile.c_str(),ios::in);
199 if(!pids) 204 if(!pids)
200 throw runtime_error("no pidfile found"); 205 throw runtime_error("no pidfile found");
201 pid_t pid = 0; 206 pid_t pid = 0;
202 pids >> pid; 207 pids >> pid;
203 pids.close(); 208 pids.close();
204 if(!pid) 209 if(!pid)
205 throw runtime_error("no pid in pidfile"); 210 throw runtime_error("no pid in pidfile");
206 if(kill(pid,signum)) 211 if(kill(pid,signum))
207 throw runtime_error("failed to signal process"); 212 throw runtime_error("failed to signal process");
208 }else if(!process_name.empty()) { 213 }else if(!process_name.empty()) {
209 if(procpids.empty()) 214 if(procpids.empty())
210 gather_proc_info(); 215 gather_proc_info();
211 pair<multimap<string,pid_t>::const_iterator,multimap<string,pid_t>::const_iterator> range = procpids.equal_range(process_name); 216 pair<multimap<string,pid_t>::const_iterator,multimap<string,pid_t>::const_iterator> range = procpids.equal_range(process_name);
212 int count = 0; 217 int count = 0;
213 for(multimap<string,pid_t>::const_iterator i=range.first;i!=range.second;++i) { 218 for(multimap<string,pid_t>::const_iterator i=range.first;i!=range.second;++i) {
214 pid_t pid = i->second; 219 pid_t pid = i->second;
215 if(kill(i->second,signum)) 220 if(kill(i->second,signum))
216 throw runtime_error("failed to signal process"); 221 throw runtime_error("failed to signal process");
217 count++; 222 count++;
218 } 223 }
219 if(!count) 224 if(!count)
220 throw runtime_error("no running instance detected"); 225 throw runtime_error("no running instance detected");
221 }else 226 }else
222 throw runtime_error("nothing is known about the process"); 227 throw runtime_error("nothing is known about the process");
223} 228}
224 229
225void process::prepare_herd() { 230void process::prepare_herd() {
226 procpids.clear(); 231 procpids.clear();
227} 232}
228void process::unprepare_herd() { 233void process::unprepare_herd() {
229 procpids.clear(); 234 procpids.clear();
230} 235}
231void process::gather_proc_info() { 236void process::gather_proc_info() {
232 vector<pid_t> allpids; 237 vector<pid_t> allpids;
233 DIR *pd = opendir("/proc"); 238 DIR *pd = opendir("/proc");
234 if(!pd) 239 if(!pd)
235 throw runtime_error("failed to open /proc"); 240 throw runtime_error("failed to open /proc");
236 struct dirent *pde; 241 struct dirent *pde;
237 pid_t selfpid = getpid(); 242 pid_t selfpid = getpid();
238 while(pde=readdir(pd)) { 243 while(pde=readdir(pd)) {
239 errno=0; 244 errno=0;
240 pid_t pid = atoi(pde->d_name); 245 pid_t pid = atoi(pde->d_name);
241 if((!pid) || pid==selfpid) 246 if((!pid) || pid==selfpid)
242 continue; 247 continue;
243 allpids.push_back(pid); 248 allpids.push_back(pid);
244 } 249 }
245 closedir(pd); 250 closedir(pd);
246 char s[256]; 251 char s[256];
247 procpids.clear(); 252 procpids.clear();
248 for(vector<pid_t>::const_iterator i=allpids.begin();i!=allpids.end();++i) { 253 for(vector<pid_t>::const_iterator i=allpids.begin();i!=allpids.end();++i) {
249 int r = snprintf(s,sizeof(s),"/proc/%d/stat",*i); 254 int r = snprintf(s,sizeof(s),"/proc/%d/stat",*i);
250 if(r>=sizeof(s) || r<1) 255 if(r>=sizeof(s) || r<1)
251 continue; 256 continue;
252 string cmd; 257 string cmd;
253 ifstream ss(s,ios::in); 258 ifstream ss(s,ios::in);
254 if(ss) { 259 if(ss) {
255 getline(ss,cmd); 260 getline(ss,cmd);