source: gsdl/trunk/src/recpt/depositoraction.cpp@ 15453

Last change on this file since 15453 was 15453, checked in by mdewsnip, 16 years ago

Removed some unnecessary inclusions of header files from colservr/protocol.

  • Property svn:keywords set to Author Date Id Revision
File size: 28.3 KB
Line 
1/**********************************************************************
2 *
3 * depositoraction.cpp --
4 * Copyright (C) 2000 The New Zealand Digital Library Project
5 *
6 * A component of the Greenstone digital library software
7 * from the New Zealand Digital Library Project at the
8 * University of Waikato, New Zealand.
9 *
10 * This program is free software; you can redistribute it and/or modify
11 * it under the terms of the GNU General Public License as published by
12 * the Free Software Foundation; either version 2 of the License, or
13 * (at your option) any later version.
14 *
15 * This program is distributed in the hope that it will be useful,
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 * GNU General Public License for more details.
19 *
20 * You should have received a copy of the GNU General Public License
21 * along with this program; if not, write to the Free Software
22 * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
23 *
24 *********************************************************************/
25
26#include "gsdl_modules_cfg.h"
27#ifdef GSDL_USE_DEPOSITOR_ACTION
28
29// note that the collectoraction relies on having direct access to a
30// collections configuration file. this breaks the separation between
31// receptionist and collection server and so is not suitable (at least
32// in its current form) for use when collection servers are separate
33// from the receptionist (e.g. when using the CORBA protocol).
34
35// following line required to get fstream.filedesc() on darwin (Mac OS X)
36#define _STREAM_COMPAT 1
37// required for utsname on solaris???
38#define _XOPEN_SOURCE 1
39#define _XOPEN_SOURCE_EXTENDED 1
40
41#include "depositoraction.h"
42#include "OIDtools.h"
43#include "fileutil.h"
44#include "cfgread.h"
45#include "gsdltools.h"
46#include "gsdltimes.h"
47#include "argdb.h"
48#include "cgiutils.h"
49
50#include <stdio.h>
51//#include <strings.h> // seems to be needed for gcc 2.95, but not particularly happy it
52#include <fcntl.h>
53#include <time.h>
54
55#if !defined (__WIN32__)
56#include <sys/utsname.h>
57#include <unistd.h>
58#endif
59
60#include <new>
61//#include <string>
62#include <vector>
63#include <stdexcept>
64#include <iostream>
65//#include <cstdlib>
66
67depositoraction::depositoraction ()
68 : wizardaction()
69{
70 macro_prefix = "di1"; //deposit item
71 lastpage = 0;
72
73 cgiarginfo arg_ainfo;
74 arg_ainfo.shortname = "a";
75 arg_ainfo.longname = "action";
76 arg_ainfo.multiplechar = true;
77 arg_ainfo.defaultstatus = cgiarginfo::weak;
78 arg_ainfo.argdefault = "depositor";
79 arg_ainfo.savedarginfo = cgiarginfo::must;
80 argsinfo.addarginfo (NULL, arg_ainfo);
81
82 arg_ainfo.shortname = "p";
83 arg_ainfo.longname = "page";
84 arg_ainfo.multiplechar = true;
85 arg_ainfo.defaultstatus = cgiarginfo::weak;
86 arg_ainfo.argdefault = "select";
87 arg_ainfo.savedarginfo = cgiarginfo::must;
88 argsinfo.addarginfo (NULL, arg_ainfo);
89
90 //furthest page that has been visited
91 arg_ainfo.shortname = "di1lastpage";
92 arg_ainfo.longname = "depositor specific";
93 arg_ainfo.multiplechar = true;
94 arg_ainfo.defaultstatus = cgiarginfo::weak;
95 arg_ainfo.argdefault = "0";
96 arg_ainfo.savedarginfo = cgiarginfo::must;
97 argsinfo.addarginfo (NULL, arg_ainfo);
98
99 // the fileupload info that cgiwrapper produces - we parse the fileupload_t
100 // and set di1userfile, di1userfilesize
101 arg_ainfo.shortname = "di1userfileinfo";
102 arg_ainfo.longname = "depositor specific";
103 arg_ainfo.fileupload = true;
104 arg_ainfo.savedarginfo = cgiarginfo::mustnot;
105 argsinfo.addarginfo (NULL, arg_ainfo);
106
107 // essential: reset fileupload to default as we are not setting
108 // it for each arg
109 arg_ainfo.fileupload = false;
110
111 //the name of the file to be added
112 arg_ainfo.shortname = "di1userfile";
113 arg_ainfo.longname = "depositor specific";
114 arg_ainfo.multiplechar = true;
115 arg_ainfo.defaultstatus = cgiarginfo::weak;
116 arg_ainfo.argdefault = g_EmptyText;
117 arg_ainfo.savedarginfo = cgiarginfo::must;
118 argsinfo.addarginfo (NULL, arg_ainfo);
119
120 //the file size
121 arg_ainfo.shortname = "di1userfilesize";
122 arg_ainfo.longname = "depositor specific";
123 arg_ainfo.multiplechar = true;
124 arg_ainfo.defaultstatus = cgiarginfo::weak;
125 arg_ainfo.argdefault = "0";
126 arg_ainfo.savedarginfo = cgiarginfo::must;
127 argsinfo.addarginfo (NULL, arg_ainfo);
128
129 //the file timestamp
130 arg_ainfo.shortname = "di1timestamp";
131 arg_ainfo.longname = "depositor specific";
132 arg_ainfo.multiplechar = true;
133 arg_ainfo.defaultstatus = cgiarginfo::weak;
134 arg_ainfo.argdefault = "0";
135 arg_ainfo.savedarginfo = cgiarginfo::must;
136 argsinfo.addarginfo (NULL, arg_ainfo);
137
138 // temporary directory name for this collector
139 // session
140 arg_ainfo.shortname = "di1tmp";
141 arg_ainfo.longname = "depositor specific";
142 arg_ainfo.multiplechar = true;
143 arg_ainfo.defaultstatus = cgiarginfo::weak;
144 arg_ainfo.argdefault = g_EmptyText;
145 arg_ainfo.savedarginfo = cgiarginfo::must;
146 argsinfo.addarginfo (NULL, arg_ainfo);
147
148// arg_ainfo.shortname = "di1fullname";
149// arg_ainfo.longname = "depositor specific";
150// arg_ainfo.multiplechar = true;
151// arg_ainfo.defaultstatus = cgiarginfo::weak;
152// arg_ainfo.argdefault = g_EmptyText;
153// arg_ainfo.savedarginfo = cgiarginfo::mustnot; // saved on disk
154// argsinfo.addarginfo (NULL, arg_ainfo);
155
156 arg_ainfo.shortname = "di1dirname";
157 arg_ainfo.longname = "depositor specific";
158 arg_ainfo.multiplechar = true;
159 arg_ainfo.defaultstatus = cgiarginfo::weak;
160 arg_ainfo.argdefault = g_EmptyText;
161 arg_ainfo.savedarginfo = cgiarginfo::must;
162 argsinfo.addarginfo (NULL, arg_ainfo);
163
164 arg_ainfo.shortname = "di1contactemail";
165 arg_ainfo.longname = "depositor specific";
166 arg_ainfo.multiplechar = true;
167 arg_ainfo.defaultstatus = cgiarginfo::weak;
168 arg_ainfo.argdefault = g_EmptyText;
169 arg_ainfo.savedarginfo = cgiarginfo::mustnot; // saved on disk
170 argsinfo.addarginfo (NULL, arg_ainfo);
171
172// arg_ainfo.shortname = "di1clone";
173// arg_ainfo.longname = "depositor specific";
174// arg_ainfo.multiplechar = false;
175// arg_ainfo.defaultstatus = cgiarginfo::weak;
176// arg_ainfo.argdefault = "0";
177// arg_ainfo.savedarginfo = cgiarginfo::mustnot; // saved on disk
178// argsinfo.addarginfo (NULL, arg_ainfo);
179
180// arg_ainfo.shortname = "di1clonecol";
181// arg_ainfo.longname = "depositor specific";
182// arg_ainfo.multiplechar = true;
183// arg_ainfo.defaultstatus = cgiarginfo::weak;
184// arg_ainfo.argdefault = g_EmptyText;
185// arg_ainfo.savedarginfo = cgiarginfo::mustnot; // saved on disk
186// argsinfo.addarginfo (NULL, arg_ainfo);
187
188// arg_ainfo.shortname = "cfgfile";
189// arg_ainfo.longname = "configuration file contents";
190// arg_ainfo.multiplechar = true;
191// arg_ainfo.defaultstatus = cgiarginfo::weak;
192// arg_ainfo.argdefault = g_EmptyText;
193// arg_ainfo.savedarginfo = cgiarginfo::mustnot;
194// argsinfo.addarginfo (NULL, arg_ainfo);
195
196// // will be set if we arrived at the "configure collection" page
197// // via the "changing an existing collection" page
198// arg_ainfo.shortname = "di1econf";
199// arg_ainfo.longname = "depositor specific";
200// arg_ainfo.multiplechar = false;
201// arg_ainfo.defaultstatus = cgiarginfo::weak;
202// arg_ainfo.argdefault = "0";
203// arg_ainfo.savedarginfo = cgiarginfo::must;
204// argsinfo.addarginfo (NULL, arg_ainfo);
205
206 // wizard uses this to see if we are working with an existing collection:
207 // set the default to 1
208 arg_ainfo.shortname = "di1esrce";
209 arg_ainfo.longname = "depositor specific";
210 arg_ainfo.multiplechar = false;
211 arg_ainfo.defaultstatus = cgiarginfo::weak;
212 arg_ainfo.argdefault = "1";
213 arg_ainfo.savedarginfo = cgiarginfo::must;
214 argsinfo.addarginfo (NULL, arg_ainfo);
215
216// arg_ainfo.shortname = "di1input";
217// arg_ainfo.longname = "depositor specific";
218// arg_ainfo.multiplechar = true;
219// arg_ainfo.multiplevalue = true;
220// arg_ainfo.defaultstatus = cgiarginfo::weak;
221// arg_ainfo.argdefault = g_EmptyText;
222// arg_ainfo.savedarginfo = cgiarginfo::mustnot; // saved on disk
223// argsinfo.addarginfo (NULL, arg_ainfo);
224
225// //we don't need this in the depositor
226// arg_ainfo.shortname = "di1inputtype";
227// arg_ainfo.longname = "depositor specific";
228// arg_ainfo.multiplechar = true;
229// arg_ainfo.multiplevalue = true;
230// arg_ainfo.defaultstatus = cgiarginfo::weak;
231// arg_ainfo.argdefault = g_EmptyText;
232// arg_ainfo.savedarginfo = cgiarginfo::mustnot; // saved on disk
233// argsinfo.addarginfo (NULL, arg_ainfo);
234
235}
236
237depositoraction::~depositoraction () {
238}
239
240
241bool depositoraction::check_cgiargs (cgiargsinfoclass &argsinfo, cgiargsclass &args, recptprotolistclass * protos, ostream &logout) {
242
243 wizardaction::check_cgiargs(argsinfo,args,protos,logout);
244
245 text_t &current_page = args["p"];
246
247 // note that the "bildstatus" and "bildframe1" pages don't actually do anything
248 // functional so we don't need to worry about authenticating them (it's the
249 // underlying "bild" page that does the building (and creates the frameset))
250 // This helps us overcome a bit of a problem we have with multiple pages trying
251 // to read from the key.db database at the same time.
252
253 //right now, anyone can do anything until it's time to build
254 if (current_page == "select") {
255 //authenticate the user if authentication is available
256 args["uan"] = 1;
257 args["ug"] = "all-collections-editor";
258 }
259
260
261 //Check to see if a file was specified
262 //if so, upload it and set the di1userfile, di1userfilesize, and
263 //di1timestamp args.
264 //every time a new file is specified, a new timestamped folder is
265 //created.
266
267 // This doesn't create a problem but is inefficient (wasted space)
268 // Consider recoding at some point in the future.
269
270 fileupload_t *fileupload = args.getargfile("di1userfileinfo");
271
272 if (fileupload != NULL) {
273
274 if (!(*fileupload).tmp_name.empty() && file_exists((*fileupload).tmp_name)) {
275 // create the timestamp
276 time_t timestamp = time(NULL);
277 text_t timestamp_str(timestamp);
278
279 args["di1timestamp"] = timestamp_str;
280
281 // set filename and size from the fileupload struct
282 args["di1userfilesize"] = (*fileupload).size;
283 args["di1userfile"] = (*fileupload).name;
284 // copy the file into its temporary location
285 text_t tmpdir = filename_cat(gsdlhome,"tmp",args["di1tmp"],timestamp_str);
286 bool tflag = mk_dir(tmpdir);
287 text_t tmpfile = filename_cat(tmpdir,(*fileupload).name);
288 if (!file_copy((*fileupload).tmp_name, tmpfile)) {
289 cerr << "Depositor error: cannot save uploaded fileto "<<tmpfile<<endl;
290 } else {
291 cerr << "Depositor: saved uploaded file to "<<tmpfile<<endl;
292 }
293 }
294 }
295
296 if (current_page == "select") {
297
298 //make sure the last page arg is re-set to zero
299 //since a new collection is being chosen
300 args["di1lastpage"] = "0";
301
302 // assign (and create) a temporary directory This will create a new
303 // tbuild directory every time the user goes back to select a new
304 // collection. It causes no problems except wasted space. Should be
305 // addressed in the future
306 if (assign_tmpname (args, logout)==false) {
307 // there was an error creating the tmp dir
308 message="tmpfail";
309 return true; // true because we could still parse the arguments
310 }
311
312 // clean up any old builds left laying about in the tmp directory
313 // (note that it's possible this could take some time if there's a huge
314 // partially built collecton laying about so we'll make it an asynchronous
315 // system call)
316 gsdl_system ("perl -S cleantmp.pl", false, logout);
317
318 }
319
320 if (current_page == "bildstatus" || current_page == "bildcancel") {
321 // if .final file exists then build has finished
322 text_t fbld = filename_cat (gsdlhome, "tmp", args[macro_prefix + "tmp"], args[macro_prefix + "dirname"] + ".bld.final");
323 if (file_exists (fbld)) {
324 char *fbldc = fbld.getcstr();
325 ifstream fbld_in (fbldc);
326 if (fbld_in) {
327 failcode = fbld_in.get();
328 fbld_in.close();
329 if (failcode == '0') {
330 // success - we need to create and configure a collection server for the
331 // newly built collection (for fastcgi and local library where
332 // initialization isn't going to be redone when the user clicks the
333 // "view your new collection" button
334 create_colserver (args[macro_prefix + "dirname"], logout);
335 current_page = "bilddone";
336 }
337 else current_page = "bildfail";
338 } else {
339 // assume build failed (we shouldn't get here though ... right?)
340 current_page = "bildfail";
341 }
342 delete []fbldc;
343 }
344 }
345
346 //is it a step page?
347 text_t::const_iterator here = current_page.begin();
348 text_t::const_iterator end = current_page.end();
349 text_t stepstring = substr(here,here+4);
350
351 //if so, increment the step count
352 //this should not be going up more than one at a time.
353 if(stepstring == "step") {
354 text_t currpage_textt = substr(here+4,here+5);
355 int currpage = currpage_textt.getint();
356
357 text_t lastpage_textt = args["di1lastpage"];
358 int lastpage = lastpage_textt.getint();
359 if (currpage > lastpage) {
360 lastpage++;
361 text_t lastpage_textt(lastpage);
362 args["di1lastpage"] = lastpage_textt;
363 }
364 }
365 return true;
366}
367
368
369void depositoraction::define_internal_macros (displayclass &disp, cgiargsclass &args,
370 recptprotolistclass *protos, ostream &logout) {
371
372 // define_internal_macros sets some/all of the following macros (depending
373 // on cgiargs):
374 //
375 // _pagescriptextra_
376 // _header_
377 // _depositorbar_
378 // _textfailmsg_
379 // _di1userfile_
380 // _di1userfilesize_
381
382 text_t &depositor_page = args["p"];
383
384 // set _pagescriptextra_ macro to _cpagescriptextra_
385 disp.setmacro ("pagescriptextra", "depositor", "_" + depositor_page + "scriptextra_");
386
387 if (depositor_page == "bildstatus" || depositor_page == "bilddone" ||
388 depositor_page == "bildfail" || depositor_page == "bildframe1" ||
389 depositor_page == "select") {
390 disp.setmacro ("header", "depositor", "_" + depositor_page + "header_");
391 }
392
393 if (depositor_page == "bildstatus") {
394 set_statusline (disp, args, logout);
395 }
396
397 if (depositor_page == "select") {
398 set_fullnamemenu (disp, args, protos, logout);
399 }
400
401 //how many pages in collection?
402 //we need a _depositorbar_ for each
403
404 text_t numsteps_str;
405 disp.expandstring("depositor", "_numsteps_", numsteps_str);
406 int numsteps = numsteps_str.getint();
407
408 //is the page a step page?
409
410 text_t::const_iterator here = depositor_page.begin();
411 text_t::const_iterator end = depositor_page.end();
412 text_t stepstring = substr(here,here+4);
413
414 if((stepstring) == "step" || (depositor_page == "depositonly") ) {
415 disp.setmacro("di1userfile","depositor",args["di1userfile"]);
416 disp.setmacro("di1userfilesize","depositor",args["di1userfilesize"]);
417 }
418
419 //set up the depositor bar
420 text_t depositorbar = "<table class=wizardbar border=0 cellspacing=4 cellpadding=0><tr>\n";
421
422 if(stepstring == "step") {
423
424 here = depositor_page.begin();
425 text_t stepnums = substr(here+4,here+5);
426 int stepnum = stepnums.getint();
427
428 text_t lastpage_textt = args["di1lastpage"];
429 int lastpage = lastpage_textt.getint();
430
431 // again, begin with the select bar...
432 //the first button - selecting a collection
433 for(int i = 1; i <= numsteps; i++) {
434 text_t numstr(i);
435 if(i <= lastpage && i == stepnum) {
436 depositorbar += get_button (args,depositor_page, "yellow", "step"+numstr, false);
437 } else if (i <= lastpage) {
438 depositorbar += get_button (args,depositor_page, "yellow", "step"+numstr, true);
439 } else if((i == lastpage+1)){
440 depositorbar += get_button (args,depositor_page, "green", "step"+numstr, true);
441 } else {
442 depositorbar += get_button (args,depositor_page, "grey", "step"+numstr, false);
443 }
444 depositorbar += "<td>_icongreyarrow_</td>\n";
445
446 }
447
448 //the build and preview pages are always last
449 if(lastpage == numsteps) {
450 depositorbar += get_button (args,depositor_page, "green", "laststep", true);
451 } else {
452 depositorbar += get_button (args,depositor_page, "grey", "laststep", false);
453 }
454
455 text_t laststep_textt;
456 disp.expandstring("depositor", "_laststep_", laststep_textt);
457 if(laststep_textt == "bild") {
458 depositorbar += "<td>_icongreyarrow_</td>\n";
459 depositorbar += get_button (args,depositor_page, "grey", "view", false);
460 }
461 depositorbar += "</tr><tr>";
462
463 for(int j = 1; j <= 2*(stepnum-1); j++) {
464 depositorbar += "<td></td>";
465 }
466 depositorbar += "<td align=center>_icongreyuparrow_</td>\n";
467
468 depositorbar += "</tr></table>\n";
469
470 }
471
472 if ((depositor_page == "bildcancel") || (depositor_page == "bildfail")) {
473
474 for(int i = 1; i <= numsteps; i++) {
475
476 text_t numstr(i);
477 depositorbar += get_button (args,depositor_page, "yellow", "step"+numstr, true);
478 depositorbar += "<td>_icongreyarrow_</td>\n";
479
480 }
481
482 //the build and preview pages are always last
483 depositorbar += get_button (args,depositor_page, "green", "laststep", true);
484 depositorbar += "<td>_icongreyarrow_</td>\n";
485 depositorbar += get_button (args,depositor_page, "grey", "view", false);
486
487 depositorbar += "</tr><tr>";
488
489 for(int j = 1; j <= 2*numsteps; j++) {
490 depositorbar += "<td></td>";
491 }
492 depositorbar += "<td align=center>_icongreyuparrow_</td>\n";
493
494 depositorbar += "</tr></table>\n";
495
496 }
497
498 if(depositor_page == "bilddone"){
499
500 for(int i = 1; i <= numsteps; i++) {
501
502 text_t numstr(i);
503 depositorbar += get_button (args, depositor_page, "grey", "step"+numstr, false);
504 depositorbar += "<td>_icongreyarrow_</td>\n";
505
506 }
507
508 //the build and preview pages are always last
509 depositorbar += get_button (args,depositor_page, "yellow", "laststep", false);
510 depositorbar += "<td>_icongreyarrow_</td>\n";
511 depositorbar += get_button (args,depositor_page, "green", "view", true);
512
513 depositorbar += "</tr><tr>";
514
515 for(int j = 1; j <= 2*numsteps; j++) {
516 depositorbar += "<td></td>";
517 }
518 depositorbar += "<td align=center>_icongreyuparrow_</td>\n";
519
520 depositorbar += "</tr></table>\n";
521 }
522 disp.setmacro ("depositorbar", "depositor", depositorbar);
523
524 if (depositor_page == "bildfail") {
525
526 text_t textfailmsg = "_textfailmsg";
527 textfailmsg.push_back(failcode);
528 textfailmsg.push_back('_');
529 disp.setmacro("textfailmsg", "depositor", textfailmsg);
530
531 text_t bldlog = filename_cat(gsdlhome, "tmp", args["di1tmp"], args["di1dirname"] + ".bld");
532 text_t rawlog = file_tail (bldlog, 6, 0);
533 // we'll shove in some <br> tags where \n's occur
534 text_t faillog;
535 text_t::const_iterator here = rawlog.begin();
536 text_t::const_iterator end = rawlog.end();
537 while (here != end) {
538 if (*here == '\n') faillog += "<br>";
539 faillog.push_back (*here);
540 ++here;
541 }
542 disp.setmacro ("faillog", "depositor", dm_safe(faillog));
543 }
544}
545
546
547//basic framework
548bool depositoraction::do_action (cgiargsclass &args, recptprotolistclass * /*protos*/,
549 browsermapclass * /*browsers*/, displayclass &disp,
550 outconvertclass &outconvert, ostream &textout,
551 ostream &logout) {
552 // make sure the depositor is enabled
553 if (disabled) {
554 textout << outconvert
555 << "<html>\n"
556 << "<head>\n"
557 << "<title>Depositor disabled</title>\n"
558 << "</head>\n"
559 << "<body bgcolor=\"#ffffff\" text=\"#000000\" link=\"#006666\" "
560 << "alink=\"#cc9900\" vlink=\"#666633\">\n"
561 << "<h2>Facility disabled</h2>\n"
562 << "Sorry, the Depositor end-user collecation update facility is currently disabled\n"
563 << "\n</body>\n"
564 << "</html>\n";
565 return true;
566 }
567
568 text_t &depositor_page = args["p"];
569 text_t &collection = args["di1dirname"];
570
571 // make sure we have perl (we won't bother with this check for the
572 // building status pages to avoid slowing things down unneccessarily)
573 if (depositor_page != "bildstatus" && depositor_page != "bildframe1" && !perl_ok(logout)) {
574 textout << outconvert
575 << "<html>\n"
576 << "<head>\n"
577 << "<title>Perl not found</title>\n"
578 << "</head>\n"
579 << "<body bgcolor=\"#ffffff\" text=\"#000000\" link=\"#006666\" "
580 << "alink=\"#cc9900\" vlink=\"#666633\">\n"
581 << "<h2>Perl not found</h2>\n"
582 << "Greenstone could not detect perl on this system. It is therefore not\n"
583 << "possible to build a Greenstone collection, either from the Collector or the \n"
584 << "command-line tools, or to use the Collector for any other task.\n"
585 << "<p>Please refer to the Greenstone Installer's Guide for details on\n"
586 << "installing perl on your system.\n"
587 << "\n</body>\n"
588 << "</html>\n";
589 return true;
590
591 }
592
593 text_t::const_iterator here = depositor_page.begin();
594 text_t::const_iterator end = depositor_page.end();
595 text_t stepstring = substr(here,here+4);
596
597 if ((depositor_page == "select") || (stepstring == "step")) {
598 textout << outconvert << disp << ("_depositor:header_\n")
599 << ("_depositor:" + depositor_page + "content_\n")
600 << ("_depositor:footer_\n");
601
602 }
603
604 if ((depositor_page == "bild") || (depositor_page == "depositoronly")) {
605
606 text_t filename_textt = args["di1userfile"];
607 text_t timestamp_str = args["di1timestamp"];
608 //check to make sure a file was uploaded
609 //if more than one upload occurred, the last one is taken
610 if(filename_textt == "") {
611 textout << outconvert << disp << "<p> no file available</p>\n";
612 } else {
613
614 text_t col_dirname = filename_cat(gsdlhome,"collect",args[macro_prefix+"dirname"]);
615 text_t import_dirname = filename_cat(col_dirname,"import");
616 if(!directory_exists(import_dirname)) {
617 bool flag = mk_dir(import_dirname);
618 if(!flag) {
619 cerr << "Error: unable to make directory " << import_dirname << endl;
620 return true;
621 }
622 }
623
624 text_t dirname = filename_cat(import_dirname,timestamp_str);
625 bool flag = mk_dir(dirname);
626 if(!flag) {
627 cerr << "Error: unable to make timestamp directory " << dirname << endl;
628 return true;
629 }
630
631 text_t filename_textt = args["di1userfile"];
632 text_t tmpdir = filename_cat(gsdlhome,"tmp",args["di1tmp"],timestamp_str);
633 text_t tmpfile = filename_cat(tmpdir,filename_textt);
634 text_t filename = filename_cat(dirname, filename_textt);
635 if(!file_copy(tmpfile, filename)) {
636 cerr << "Unable to copy " << tmpfile << " to " << filename << endl;
637 }
638
639 //write the metadata file
640 write_metadata_file(args, filename_textt, timestamp_str);
641
642 if(depositor_page == "bild"){
643
644 //create a manifest file
645 write_manifest_file(args, filename_textt, timestamp_str);
646
647 // do the work (download, import, build)
648 gsdl_build (args, logout);
649
650 if (message.empty()) {
651 // bild page is a frameset so we don't want headers and stuff
652 textout << outconvert << disp << ("_depositor:bildcontent_\n");
653 }
654 }
655 }
656 }
657
658 if (message.empty()) {
659
660 if (depositor_page != "bild" && stepstring != "step" && depositor_page != "select") {
661 // output page ("bild" page was already output above)
662 textout << outconvert << disp << ("_depositor:header_\n")
663 << ("_depositor:" + depositor_page + "content_\n")
664 << ("_depositor:footer_\n");
665 }
666 } else {
667 // message was set somewhere (probably an error), output message page
668 textout << outconvert << disp << ("_depositor:header_\n")
669 << ("_depositor:" + message + "content_\n")
670 << ("_depositor:footer_\n");
671 message.clear();
672 }
673
674 return true;
675
676}
677
678
679//This function creates a metadata.xml file
680void depositoraction::write_metadata_file(cgiargsclass &args, text_t filename_str, text_t& timestamp_str)
681{
682
683 //build metadata file
684 //the front
685 text_t metadata_file = "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n";
686 metadata_file += "<!DOCTYPE DirectoryMetadata SYSTEM \"http://greenstone.org/dtd/DirectoryMetadata/1.0/DirectoryMetadata.dtd\">\n";
687 metadata_file += "<DirectoryMetadata>\n";
688 metadata_file += " <FileSet>\n";
689 metadata_file += " <FileName>";
690
691 // Generate 'regular expression' save version of filename
692 // for now that means escaping any '.' characters
693 // consider generalising as function in gsdltools.h/cpp?
694
695 text_t filename_re;
696 text_t::const_iterator here = filename_str.begin();
697 text_t::const_iterator end = filename_str.end();
698 while (here != end) {
699 if (*here == '.') {
700 filename_re.push_back('\\');
701 }
702
703 filename_re.push_back(*here);
704 ++here;
705 }
706
707 metadata_file += filename_re;
708
709 metadata_file += "</FileName>\n";
710 metadata_file += " <Description>\n";
711
712
713 cgiargsclass::const_iterator args_here = args.begin();
714 cgiargsclass::const_iterator args_end = args.end();
715
716 while (args_here != args_end) {
717 text_t args_name = (*args_here).first;
718
719 int prefix_len = macro_prefix.size();
720
721 text_t args_prefix = substr(args_name.begin(),args_name.begin()+prefix_len+3);
722
723 if (args_prefix == (macro_prefix+"md.")) {
724 text_t args_val = args[args_name];
725
726 text_t args_suffix = substr(args_name.begin()+prefix_len+3,args_name.end());
727
728 text_tarray mdvalues;
729 splitchar (args_val.begin(), args_val.end(), ',', mdvalues);
730 int numvalues = mdvalues.size();
731
732 for (int i = 0; i < numvalues; ++i) {
733 if (!mdvalues[i].empty()) {
734
735 decode_cgi_arg(mdvalues[i]);
736
737 metadata_file += " <Metadata mode=\"accumulate\" name=\"";
738 metadata_file += args_suffix;
739 metadata_file += "\">";
740 metadata_file += mdvalues[i];
741 metadata_file += "</Metadata>\n";
742
743 }
744 }
745
746 }
747
748
749 ++args_here;
750 }
751
752 //the end of the file
753 metadata_file += " </Description>\n";
754 metadata_file += " </FileSet>\n";
755 metadata_file += "</DirectoryMetadata>\n";
756
757 //create metadata.xml file
758 text_t metadata_path = filename_cat(gsdlhome,"collect", args[macro_prefix+"dirname"], "import", timestamp_str, "metadata.xml");
759 text_t my_path = filename_cat(gsdlhome,"tmp", "metadata.xml");
760
761 ofstream metadata(metadata_path.getcstr());
762 ofstream my_tmp(my_path.getcstr());
763
764 if(!metadata.is_open()) {
765 cerr << "Cannot open metadata.xml!" << endl;
766 } else {
767
768 //write metadata.xml
769 metadata.write(metadata_file.getcstr(), metadata_file.size());
770 my_tmp.write(metadata_file.getcstr(), metadata_file.size());
771 }
772}
773
774//This function creates a manifest.xml file
775void depositoraction::write_manifest_file(cgiargsclass &args, text_t filename,
776 text_t& timestamp_str)
777{
778
779 //build manifest file
780
781 //the front
782 text_t manifest_file = "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n";
783 manifest_file += "<Manifest>\n";
784 manifest_file += " <Index>\n";
785 manifest_file += " <Filename>\n";
786 manifest_file += " " + timestamp_str + "\n";
787 manifest_file += " </Filename>\n";
788
789 //the end of the file
790 manifest_file += " </Index>\n";
791 manifest_file += "</Manifest>\n";
792
793 //create manifest.xml file
794 text_t manifest_path = filename_cat(gsdlhome,"collect", args[macro_prefix+"dirname"], "manifest.xml");
795
796 char* manifest_cstr = manifest_path.getcstr();
797
798 ofstream manifest(manifest_cstr);
799
800 if(!manifest.is_open()) {
801 cerr << "Cannot open" << manifest_cstr << endl;
802 } else {
803
804 //write manifest.xml
805 char *manifest_file_cstr = manifest_file.getcstr();
806 manifest.write(manifest_file_cstr, manifest_file.size());
807 delete [] manifest_file_cstr;
808 }
809
810 delete [] manifest_cstr;
811}
812
813
814
815text_t depositoraction::get_button(cgiargsclass &args, const text_t &thispage,
816 const text_t &color,
817 const text_t &type, bool enabled)
818{
819
820 text_t::const_iterator here = type.begin();
821 text_t::const_iterator end = type.end();
822 text_t stepstring = substr(here,here+4);
823 if ((color != "green" && color != "grey" && color != "yellow") ||
824 (type != "select" && type != "laststep" && stepstring != "step" && type != "view"))
825 return g_EmptyText;
826
827 text_t href = "_http"+type+"_";
828 text_t target = "";
829
830 here = thispage.begin();
831 stepstring = substr(here,here+4);
832 if (thispage == "bildcancel" || thispage == "bildfail" || thispage == "select"|| stepstring == "step") {
833 // call the check submit macro instead of linking directly to the page
834
835 if(type == "laststep") {
836 href="\"javascript:check_submit('_"+type+"_');\"";
837 } else {
838 href="\"javascript:check_submit('"+type+"');\"";
839 }
840
841 } else if (type == "view") {
842 //target = " target=_top";
843 target = " target=_blank";
844 }
845
846 text_t tdclass = "wizardbar"+color;
847 if (enabled) {
848 // link to the appropriate page
849 return "<td class="+tdclass+"><a href="+href+target+">_text"+type+"_</a></td>";
850 }
851 else {
852 // just display the text
853 return "<td class="+tdclass+">_text"+type+"_</td>";
854 }
855}
856
857
858#endif //GSDL_USE_DEPOSITOR_ACTION
Note: See TracBrowser for help on using the repository browser.