summaryrefslogtreecommitdiffabout
path: root/src
Unidiff
Diffstat (limited to 'src') (more/less context) (ignore whitespace changes)
-rw-r--r--src/eyefiservice.cc1
-rw-r--r--src/eyefiworker.cc2
2 files changed, 3 insertions, 0 deletions
diff --git a/src/eyefiservice.cc b/src/eyefiservice.cc
index 9650ec1..0bf26c4 100644
--- a/src/eyefiservice.cc
+++ b/src/eyefiservice.cc
@@ -1,231 +1,232 @@
1#include <cassert> 1#include <cassert>
2#include <iostream> 2#include <iostream>
3#include <fstream> 3#include <fstream>
4#include <stdexcept> 4#include <stdexcept>
5#include <iterator> 5#include <iterator>
6#include <syslog.h> 6#include <syslog.h>
7#include <sys/wait.h> 7#include <sys/wait.h>
8#include <autosprintf.h> 8#include <autosprintf.h>
9#include "eyekinfig.h" 9#include "eyekinfig.h"
10#include "eyetil.h" 10#include "eyetil.h"
11#include "soapeyefiService.h" 11#include "soapeyefiService.h"
12 12
13static binary_t session_nonce; 13static binary_t session_nonce;
14 14
15static bool detached_child() { 15static bool detached_child() {
16 pid_t p = fork(); 16 pid_t p = fork();
17 if(p<0) throw std::runtime_error("failed to fork()"); 17 if(p<0) throw std::runtime_error("failed to fork()");
18 if(!p) { 18 if(!p) {
19 p = fork(); 19 p = fork();
20 if(p<0) { 20 if(p<0) {
21 syslog(LOG_ERR,"Failed to re-fork child process"); 21 syslog(LOG_ERR,"Failed to re-fork child process");
22 _exit(-1); 22 _exit(-1);
23 } 23 }
24 if(!p) { 24 if(!p) {
25 setsid(); 25 setsid();
26 for(int i=getdtablesize();i>=0;--i) close(i); 26 for(int i=getdtablesize();i>=0;--i) close(i);
27 int i=open("/dev/null",O_RDWR); assert(i==0); 27 int i=open("/dev/null",O_RDWR); assert(i==0);
28 i = dup(i); assert(i==1); 28 i = dup(i); assert(i==1);
29 i = dup(i); assert(i==2); 29 i = dup(i); assert(i==2);
30 return true; 30 return true;
31 } 31 }
32 _exit(0); 32 _exit(0);
33 } 33 }
34 int rc; 34 int rc;
35 if(waitpid(p,&rc,0)<0) throw std::runtime_error("failed to waitpid()"); 35 if(waitpid(p,&rc,0)<0) throw std::runtime_error("failed to waitpid()");
36 if(!WIFEXITED(rc)) throw std::runtime_error("error in forked process"); 36 if(!WIFEXITED(rc)) throw std::runtime_error("error in forked process");
37 if(WEXITSTATUS(rc)) throw std::runtime_error("forked process signalled error"); 37 if(WEXITSTATUS(rc)) throw std::runtime_error("forked process signalled error");
38 return false; 38 return false;
39} 39}
40 40
41int eyefiService::StartSession( 41int eyefiService::StartSession(
42 std::string macaddress,std::string cnonce, 42 std::string macaddress,std::string cnonce,
43 int transfermode,long transfermodetimestamp, 43 int transfermode,long transfermodetimestamp,
44 struct rns__StartSessionResponse &r ) try { 44 struct rns__StartSessionResponse &r ) try {
45#ifndef NDEBUG 45#ifndef NDEBUG
46 syslog(LOG_DEBUG, 46 syslog(LOG_DEBUG,
47 "StartSession request from %s with cnonce=%s, transfermode=%d, transfermodetimestamp=%ld", 47 "StartSession request from %s with cnonce=%s, transfermode=%d, transfermodetimestamp=%ld",
48 macaddress.c_str(), cnonce.c_str(), transfermode, transfermodetimestamp ); 48 macaddress.c_str(), cnonce.c_str(), transfermode, transfermodetimestamp );
49#endif 49#endif
50 eyekinfig_t eyekinfig(macaddress); 50 eyekinfig_t eyekinfig(macaddress);
51 r.credential = binary_t(macaddress+cnonce+eyekinfig.get_upload_key()).md5().hex(); 51 r.credential = binary_t(macaddress+cnonce+eyekinfig.get_upload_key()).md5().hex();
52 52
53 r.snonce = session_nonce.make_nonce().hex(); 53 r.snonce = session_nonce.make_nonce().hex();
54 r.transfermode=transfermode; 54 r.transfermode=transfermode;
55 r.transfermodetimestamp=transfermodetimestamp; 55 r.transfermodetimestamp=transfermodetimestamp;
56 r.upsyncallowed=false; 56 r.upsyncallowed=false;
57 57
58 std::string cmd = eyekinfig.get_on_start_session(); 58 std::string cmd = eyekinfig.get_on_start_session();
59 if(!cmd.empty()) { 59 if(!cmd.empty()) {
60 if(detached_child()) { 60 if(detached_child()) {
61 putenv( gnu::autosprintf("EYEFI_MACADDRESS=%s",macaddress.c_str()) ); 61 putenv( gnu::autosprintf("EYEFI_MACADDRESS=%s",macaddress.c_str()) );
62 putenv( gnu::autosprintf("EYEFI_TRANSFERMODE=%d",transfermode) ); 62 putenv( gnu::autosprintf("EYEFI_TRANSFERMODE=%d",transfermode) );
63 putenv( gnu::autosprintf("EYEFI_TRANSFERMODETIMESTAMP=%ld",transfermodetimestamp) ); 63 putenv( gnu::autosprintf("EYEFI_TRANSFERMODETIMESTAMP=%ld",transfermodetimestamp) );
64 char *argv[] = { (char*)"/bin/sh", (char*)"-c", (char*)cmd.c_str(), 0 }; 64 char *argv[] = { (char*)"/bin/sh", (char*)"-c", (char*)cmd.c_str(), 0 };
65 execv("/bin/sh",argv); 65 execv("/bin/sh",argv);
66 syslog(LOG_ERR,"Failed to execute '%s'",cmd.c_str()); 66 syslog(LOG_ERR,"Failed to execute '%s'",cmd.c_str());
67 _exit(-1); 67 _exit(-1);
68 } 68 }
69 } 69 }
70 return SOAP_OK; 70 return SOAP_OK;
71}catch(std::runtime_error& e) { 71}catch(std::runtime_error& e) {
72 syslog(LOG_ERR,"error while processing StartSession: %s",e.what()); 72 syslog(LOG_ERR,"error while processing StartSession: %s",e.what());
73} 73}
74 74
75int eyefiService::GetPhotoStatus( 75int eyefiService::GetPhotoStatus(
76 std::string credential, std::string macaddress, 76 std::string credential, std::string macaddress,
77 std::string filename, long filesize, std::string filesignature, 77 std::string filename, long filesize, std::string filesignature,
78 int flags, 78 int flags,
79 struct rns__GetPhotoStatusResponse &r ) { 79 struct rns__GetPhotoStatusResponse &r ) {
80#ifndef NDEBUG 80#ifndef NDEBUG
81 syslog(LOG_DEBUG, 81 syslog(LOG_DEBUG,
82 "GetPhotoStatus request from %s with credential=%s, filename=%s, filesize=%ld, filesignature=%s, flags=%d; session nonce=%s", 82 "GetPhotoStatus request from %s with credential=%s, filename=%s, filesize=%ld, filesignature=%s, flags=%d; session nonce=%s",
83 macaddress.c_str(), credential.c_str(), filename.c_str(), filesize, filesignature.c_str(), flags, 83 macaddress.c_str(), credential.c_str(), filename.c_str(), filesize, filesignature.c_str(), flags,
84 session_nonce.hex().c_str() ); 84 session_nonce.hex().c_str() );
85#endif 85#endif
86 86
87 std::string computed_credential = binary_t(macaddress+eyekinfig_t(macaddress).get_upload_key()+session_nonce.hex()).md5().hex(); 87 std::string computed_credential = binary_t(macaddress+eyekinfig_t(macaddress).get_upload_key()+session_nonce.hex()).md5().hex();
88 88
89#ifndef NDEBUG 89#ifndef NDEBUG
90 syslog(LOG_DEBUG, " computed credential=%s", computed_credential.c_str()); 90 syslog(LOG_DEBUG, " computed credential=%s", computed_credential.c_str());
91#endif 91#endif
92 92
93 if (credential != computed_credential) throw std::runtime_error("card authentication failed"); 93 if (credential != computed_credential) throw std::runtime_error("card authentication failed");
94 94
95 r.fileid = 1; r.offset = 0; 95 r.fileid = 1; r.offset = 0;
96 return SOAP_OK; 96 return SOAP_OK;
97} 97}
98 98
99int eyefiService::MarkLastPhotoInRoll( 99int eyefiService::MarkLastPhotoInRoll(
100 std::string macaddress, int mergedelta, 100 std::string macaddress, int mergedelta,
101 struct rns__MarkLastPhotoInRollResponse &r ) { 101 struct rns__MarkLastPhotoInRollResponse &r ) {
102#ifndef NDEBUG 102#ifndef NDEBUG
103 syslog(LOG_DEBUG, 103 syslog(LOG_DEBUG,
104 "MarkLastPhotoInRoll request from %s with mergedelta=%d", 104 "MarkLastPhotoInRoll request from %s with mergedelta=%d",
105 macaddress.c_str(), mergedelta ); 105 macaddress.c_str(), mergedelta );
106#endif 106#endif
107 std::string cmd = eyekinfig_t(macaddress).get_on_mark_last_photo_in_roll(); 107 std::string cmd = eyekinfig_t(macaddress).get_on_mark_last_photo_in_roll();
108 if(!cmd.empty()) { 108 if(!cmd.empty()) {
109 if(detached_child()) { 109 if(detached_child()) {
110 putenv( gnu::autosprintf("EYEFI_MACADDRESS=%s",macaddress.c_str()) ); 110 putenv( gnu::autosprintf("EYEFI_MACADDRESS=%s",macaddress.c_str()) );
111 putenv( gnu::autosprintf("EYEFI_MERGEDELTA=%d",mergedelta) ); 111 putenv( gnu::autosprintf("EYEFI_MERGEDELTA=%d",mergedelta) );
112 char *argv[] = { (char*)"/bin/sh", (char*)"-c", (char*)cmd.c_str(), 0 }; 112 char *argv[] = { (char*)"/bin/sh", (char*)"-c", (char*)cmd.c_str(), 0 };
113 execv("/bin/sh",argv); 113 execv("/bin/sh",argv);
114 syslog(LOG_ERR,"Failed to execute '%s'",cmd.c_str()); 114 syslog(LOG_ERR,"Failed to execute '%s'",cmd.c_str());
115 _exit(-1); 115 _exit(-1);
116 } 116 }
117 } 117 }
118 keep_alive = 0;
118 return SOAP_OK; 119 return SOAP_OK;
119} 120}
120 121
121int eyefiService::UploadPhoto( 122int eyefiService::UploadPhoto(
122 int fileid, std::string macaddress, 123 int fileid, std::string macaddress,
123 std::string filename, long filesize, std::string filesignature, 124 std::string filename, long filesize, std::string filesignature,
124 std::string encryption, int flags, 125 std::string encryption, int flags,
125 struct rns__UploadPhotoResponse& r ) { 126 struct rns__UploadPhotoResponse& r ) {
126#ifndef NDEBUG 127#ifndef NDEBUG
127 syslog(LOG_DEBUG, 128 syslog(LOG_DEBUG,
128 "UploadPhoto request from %s with fileid=%d, filename=%s, filesize=%ld," 129 "UploadPhoto request from %s with fileid=%d, filename=%s, filesize=%ld,"
129 " filesignature=%s, encryption=%s, flags=%04X", 130 " filesignature=%s, encryption=%s, flags=%04X",
130 macaddress.c_str(), fileid, filename.c_str(), filesize, 131 macaddress.c_str(), fileid, filename.c_str(), filesize,
131 filesignature.c_str(), encryption.c_str(), flags ); 132 filesignature.c_str(), encryption.c_str(), flags );
132#endif 133#endif
133 eyekinfig_t eyekinfig(macaddress); 134 eyekinfig_t eyekinfig(macaddress);
134 135
135 umask(eyekinfig.get_umask()); 136 umask(eyekinfig.get_umask());
136 137
137 std::string td = eyekinfig.get_targetdir(); 138 std::string td = eyekinfig.get_targetdir();
138 tmpdir_t indir(td+"/.incoming.XXXXXX"); 139 tmpdir_t indir(td+"/.incoming.XXXXXX");
139 140
140 std::string jf,lf; 141 std::string jf,lf;
141 binary_t digest, idigest; 142 binary_t digest, idigest;
142 143
143 for(soap_multipart::iterator i=mime.begin(),ie=mime.end();i!=ie;++i) { 144 for(soap_multipart::iterator i=mime.begin(),ie=mime.end();i!=ie;++i) {
144#ifndef NDEBUG 145#ifndef NDEBUG
145 syslog(LOG_DEBUG, 146 syslog(LOG_DEBUG,
146 " MIME attachment with id=%s, type=%s, size=%ld", 147 " MIME attachment with id=%s, type=%s, size=%ld",
147 (*i).id, (*i).type, (long)(*i).size ); 148 (*i).id, (*i).type, (long)(*i).size );
148#endif 149#endif
149 150
150 if((*i).id && !strcmp((*i).id,"INTEGRITYDIGEST")) { 151 if((*i).id && !strcmp((*i).id,"INTEGRITYDIGEST")) {
151 std::string idigestr((*i).ptr,(*i).size); 152 std::string idigestr((*i).ptr,(*i).size);
152#ifndef NDEBUG 153#ifndef NDEBUG
153 syslog(LOG_DEBUG, " INTEGRITYDIGEST=%s", idigestr.c_str()); 154 syslog(LOG_DEBUG, " INTEGRITYDIGEST=%s", idigestr.c_str());
154#endif 155#endif
155 idigest.from_hex(idigestr); 156 idigest.from_hex(idigestr);
156 } 157 }
157 if( (*i).id && !strcmp((*i).id,"FILENAME") ) { 158 if( (*i).id && !strcmp((*i).id,"FILENAME") ) {
158 assert( (*i).type && !strcmp((*i).type,"application/x-tar") ); 159 assert( (*i).type && !strcmp((*i).type,"application/x-tar") );
159#ifdef III_SAVE_TARS 160#ifdef III_SAVE_TARS
160 std::string tarfile = indir.get_file(filename); 161 std::string tarfile = indir.get_file(filename);
161 { 162 {
162 std::ofstream(tarfile.c_str(),std::ios::out|std::ios::binary).write((*i).ptr,(*i).size); 163 std::ofstream(tarfile.c_str(),std::ios::out|std::ios::binary).write((*i).ptr,(*i).size);
163 } 164 }
164#endif 165#endif
165 166
166 if(!jf.empty()) throw std::runtime_error("already seen tarball"); 167 if(!jf.empty()) throw std::runtime_error("already seen tarball");
167 if(!digest.empty()) throw std::runtime_error("already have integrity digest"); 168 if(!digest.empty()) throw std::runtime_error("already have integrity digest");
168 digest = integrity_digest((*i).ptr,(*i).size,eyekinfig.get_upload_key()); 169 digest = integrity_digest((*i).ptr,(*i).size,eyekinfig.get_upload_key());
169#ifndef NDEBUG 170#ifndef NDEBUG
170 syslog(LOG_DEBUG," computed integrity digest=%s", digest.hex().c_str()); 171 syslog(LOG_DEBUG," computed integrity digest=%s", digest.hex().c_str());
171#endif 172#endif
172 173
173 tarchive_t a((*i).ptr,(*i).size); 174 tarchive_t a((*i).ptr,(*i).size);
174 while(a.read_next_header()) { 175 while(a.read_next_header()) {
175 std::string f = indir.get_file(a.entry_pathname()); 176 std::string f = indir.get_file(a.entry_pathname());
176 std::string::size_type fl = f.length(); 177 std::string::size_type fl = f.length();
177 if(fl<4) continue; 178 if(fl<4) continue;
178 const char *s = f.c_str()+fl-4; 179 const char *s = f.c_str()+fl-4;
179 if(!strcasecmp(s,".JPG")) 180 if(!strcasecmp(s,".JPG"))
180 jf = f; 181 jf = f;
181 else if(!strcasecmp(s,".log")) 182 else if(!strcasecmp(s,".log"))
182 lf = f; 183 lf = f;
183 else continue; 184 else continue;
184 int fd=open(f.c_str(),O_CREAT|O_WRONLY,0666); 185 int fd=open(f.c_str(),O_CREAT|O_WRONLY,0666);
185 if(fd<0) 186 if(fd<0)
186 throw std::runtime_error(gnu::autosprintf("failed to create output file '%s'",f.c_str())); 187 throw std::runtime_error(gnu::autosprintf("failed to create output file '%s'",f.c_str()));
187 if(!a.read_data_into_fd(fd)) 188 if(!a.read_data_into_fd(fd))
188 throw std::runtime_error(gnu::autosprintf("failed to untar file into '%s'",f.c_str())); 189 throw std::runtime_error(gnu::autosprintf("failed to untar file into '%s'",f.c_str()));
189 close(fd); 190 close(fd);
190 } 191 }
191 } 192 }
192 } 193 }
193 194
194 if(jf.empty()) throw std::runtime_error("haven't seen jpeg file"); 195 if(jf.empty()) throw std::runtime_error("haven't seen jpeg file");
195 if(digest!=idigest) throw std::runtime_error("integrity digest verification failed"); 196 if(digest!=idigest) throw std::runtime_error("integrity digest verification failed");
196 197
197 std::string::size_type ls = jf.rfind('/'); 198 std::string::size_type ls = jf.rfind('/');
198 // XXX: actually, lack of '/' signifies error here 199 // XXX: actually, lack of '/' signifies error here
199 std::string jbn = (ls==std::string::npos)?jf:jf.substr(ls+1); 200 std::string jbn = (ls==std::string::npos)?jf:jf.substr(ls+1);
200 ls = lf.rfind('/'); 201 ls = lf.rfind('/');
201 std::string lbn = (ls==std::string::npos)?lf:lf.substr(ls+1); 202 std::string lbn = (ls==std::string::npos)?lf:lf.substr(ls+1);
202 std::string tjf,tlf; 203 std::string tjf,tlf;
203 bool success = false; 204 bool success = false;
204 for(int i=0;i<32767;++i) { 205 for(int i=0;i<32767;++i) {
205 const char *fmt = i ? "%1$s/(%3$05d)%2$s" : "%1$s/%2$s"; 206 const char *fmt = i ? "%1$s/(%3$05d)%2$s" : "%1$s/%2$s";
206 tjf = (const char*)gnu::autosprintf(fmt,td.c_str(),jbn.c_str(),i); 207 tjf = (const char*)gnu::autosprintf(fmt,td.c_str(),jbn.c_str(),i);
207 if(!lf.empty()) tlf = (const char*)gnu::autosprintf(fmt,td.c_str(),lbn.c_str(),i); 208 if(!lf.empty()) tlf = (const char*)gnu::autosprintf(fmt,td.c_str(),lbn.c_str(),i);
208 if( (!link(jf.c_str(),tjf.c_str())) && (lf.empty()) || !link(lf.c_str(),tlf.c_str()) ) { 209 if( (!link(jf.c_str(),tjf.c_str())) && (lf.empty()) || !link(lf.c_str(),tlf.c_str()) ) {
209 unlink(jf.c_str()); 210 unlink(jf.c_str());
210 if(!lf.empty()) unlink(lf.c_str()); 211 if(!lf.empty()) unlink(lf.c_str());
211 success=true; 212 success=true;
212 break; 213 break;
213 } 214 }
214 } 215 }
215 std::string cmd = eyekinfig.get_on_upload_photo(); 216 std::string cmd = eyekinfig.get_on_upload_photo();
216 if(success && !cmd.empty()) { 217 if(success && !cmd.empty()) {
217 if(detached_child()) { 218 if(detached_child()) {
218 putenv( gnu::autosprintf("EYEFI_UPLOADED_ORIG=%s",jbn.c_str()) ); 219 putenv( gnu::autosprintf("EYEFI_UPLOADED_ORIG=%s",jbn.c_str()) );
219 putenv( gnu::autosprintf("EYEFI_MACADDRESS=%s",macaddress.c_str()) ); 220 putenv( gnu::autosprintf("EYEFI_MACADDRESS=%s",macaddress.c_str()) );
220 putenv( gnu::autosprintf("EYEFI_UPLOADED=%s",tjf.c_str()) ); 221 putenv( gnu::autosprintf("EYEFI_UPLOADED=%s",tjf.c_str()) );
221 if(!lf.empty()) putenv( gnu::autosprintf("EYEFI_LOG=%s",tlf.c_str()) ); 222 if(!lf.empty()) putenv( gnu::autosprintf("EYEFI_LOG=%s",tlf.c_str()) );
222 char *argv[] = { (char*)"/bin/sh", (char*)"-c", (char*)cmd.c_str(), 0 }; 223 char *argv[] = { (char*)"/bin/sh", (char*)"-c", (char*)cmd.c_str(), 0 };
223 execv("/bin/sh",argv); 224 execv("/bin/sh",argv);
224 syslog(LOG_ERR,"Failed to execute '%s'",cmd.c_str()); 225 syslog(LOG_ERR,"Failed to execute '%s'",cmd.c_str());
225 _exit(-1); 226 _exit(-1);
226 } 227 }
227 } 228 }
228 229
229 r.success = true; 230 r.success = true;
230 return SOAP_OK; 231 return SOAP_OK;
231} 232}
diff --git a/src/eyefiworker.cc b/src/eyefiworker.cc
index 4038658..90c2bc2 100644
--- a/src/eyefiworker.cc
+++ b/src/eyefiworker.cc
@@ -1,26 +1,28 @@
1#include <signal.h> 1#include <signal.h>
2#include <stdexcept> 2#include <stdexcept>
3#include "eyefiworker.h" 3#include "eyefiworker.h"
4 4
5eyefiworker::eyefiworker() 5eyefiworker::eyefiworker()
6 : eyefiService(SOAP_IO_STORE|SOAP_IO_KEEPALIVE) { 6 : eyefiService(SOAP_IO_STORE|SOAP_IO_KEEPALIVE) {
7 bind_flags = SO_REUSEADDR; max_keep_alive = 0; 7 bind_flags = SO_REUSEADDR; max_keep_alive = 0;
8 socket_flags = MSG_NOSIGNAL;
8 } 9 }
9 10
10int eyefiworker::run(int port) { 11int eyefiworker::run(int port) {
11 if(!soap_valid_socket(bind(0,port,5))) 12 if(!soap_valid_socket(bind(0,port,5)))
12 throw std::runtime_error("failed to bind()"); 13 throw std::runtime_error("failed to bind()");
13 signal(SIGCHLD,SIG_IGN); 14 signal(SIGCHLD,SIG_IGN);
14 while(true) { 15 while(true) {
15 if(!soap_valid_socket(accept())) 16 if(!soap_valid_socket(accept()))
16 throw std::runtime_error("failed to accept()"); 17 throw std::runtime_error("failed to accept()");
17 pid_t p = fork(); 18 pid_t p = fork();
18 if(p<0) throw std::runtime_error("failed to fork()"); 19 if(p<0) throw std::runtime_error("failed to fork()");
19 if(!p) { 20 if(!p) {
21 recv_timeout = 600; send_timeout = 120;
20 (void)serve(); 22 (void)serve();
21 soap_destroy(this); soap_end(this); soap_done(this); 23 soap_destroy(this); soap_end(this); soap_done(this);
22 _exit(0); 24 _exit(0);
23 } 25 }
24 close(socket); socket = SOAP_INVALID_SOCKET; 26 close(socket); socket = SOAP_INVALID_SOCKET;
25 } 27 }
26} 28}