You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

774 lines
45 KiB

  1. <?php
  2. // Project: Web Reference Database (refbase) <http://www.refbase.net>
  3. // Copyright: Matthias Steffens <mailto:refbase@extracts.de> and the file's
  4. // original author(s).
  5. //
  6. // This code is distributed in the hope that it will be useful,
  7. // but WITHOUT ANY WARRANTY. Please see the GNU General Public
  8. // License for more details.
  9. //
  10. // File: ./import_csa_modify.php
  11. // Repository: $HeadURL: file:///svn/p/refbase/code/branches/bleeding-edge/import_csa_modify.php $
  12. // Author(s): Matthias Steffens <mailto:refbase@extracts.de>
  13. //
  14. // Created: 21-Nov-03, 22:46
  15. // Modified: $Date: 2012-02-27 20:25:30 +0000 (Mon, 27 Feb 2012) $
  16. // $Author: msteffens $
  17. // $Revision: 1337 $
  18. // This php script accepts input from 'import_csa.php' and will process any CSA full record data. In case of a single
  19. // record, the script will call 'record.php' with all provided fields pre-filled. The user can then verify the data,
  20. // add or modify any details as necessary and add the record to the database. Multiple records will be imported directly.
  21. // TODO: I18n
  22. // Incorporate some include files:
  23. include 'initialize/db.inc.php'; // 'db.inc.php' is included to hide username and password
  24. include 'includes/include.inc.php'; // include common functions
  25. include 'includes/import.inc.php'; // include common import functions
  26. include 'initialize/ini.inc.php'; // include common variables
  27. // --------------------------------------------------------------------
  28. // START A SESSION:
  29. // call the 'start_session()' function (from 'include.inc.php') which will also read out available session variables:
  30. start_session(true);
  31. // --------------------------------------------------------------------
  32. // Initialize preferred display language:
  33. // (note that 'locales.inc.php' has to be included *after* the call to the 'start_session()' function)
  34. include 'includes/locales.inc.php'; // include the locales
  35. // --------------------------------------------------------------------
  36. // Clear any errors that might have been found previously:
  37. $errors = array();
  38. // Write the (POST or GET) form variables into an array:
  39. foreach($_REQUEST as $varname => $value)
  40. $formVars[$varname] = $value;
  41. // --------------------------------------------------------------------
  42. // Get the referring page (or set a default one if no referrer is available):
  43. if (!empty($_SERVER['HTTP_REFERER'])) // if the referrer variable isn't empty
  44. $referer = $_SERVER['HTTP_REFERER']; // on error, redirect to calling page
  45. else
  46. $referer = "import_csa.php"; // on error, redirect to CSA import form
  47. // First of all, check if the user is logged in:
  48. if (!isset($_SESSION['loginEmail'])) // -> if the user isn't logged in
  49. {
  50. header("Location: user_login.php?referer=" . rawurlencode($referer)); // ask the user to login first, then he'll get directed back to the calling page (normally, 'import_csa.php')
  51. exit; // >>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>> !EXIT! <<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<
  52. }
  53. // now, check if the (logged in) user is allowed to import any record into the database:
  54. if (isset($_SESSION['user_permissions']) AND !preg_match("/allow_import|allow_batch_import/", $_SESSION['user_permissions'])) // if the 'user_permissions' session variable does NOT contain either 'allow_import' or 'allow_batch_import'...
  55. {
  56. // return an appropriate error message:
  57. $HeaderString = returnMsg($loc["NoPermission"] . $loc["NoPermission_ForImport"] . "!", "warning", "strong", "HeaderString"); // function 'returnMsg()' is defined in 'include.inc.php'
  58. if (!preg_match("/^cli/i", $client))
  59. header("Location: index.php"); // redirect back to main page ('index.php')
  60. exit; // >>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>> !EXIT! <<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<
  61. }
  62. // Extract form variables sent through POST:
  63. // Note: Although we could use the '$formVars' array directly below (e.g.: $formVars['sourceText'] etc., like in 'user_validation.php'), we'll read out
  64. // all variables individually again. This is done to enhance readability. (A smarter way of doing so seems to be the use of the 'extract()' function, but that
  65. // may expose yet another security hole...)
  66. // Get the form used by the user:
  67. $formType = $formVars['formType'];
  68. // Get the source text containing the CSA record(s):
  69. $sourceText = $formVars['sourceText'];
  70. // Check if source text originated from a PubMed import form (instead of 'import_csa.php')
  71. if ($formType == "importPUBMED")
  72. {
  73. $sourceText = trim($sourceText); // trim whitespace from poorly copied id's
  74. // Fetch PubMed XML data (by PubMed ID given in '$sourceText') and convert to CSA format:
  75. // (this allows for import of PubMed records via the import form provided by 'import_pubmed.php')
  76. $sourceText = PubmedToCsa($sourceText); // function 'PubmedToCsa()' is defined in 'import.inc.php'
  77. }
  78. // Check if the format of the pasted source data is in "ISI Web of Science" format (instead of "CSA" format):
  79. if ((!preg_match("/\s*Record \d+ of \d+\s*/", $sourceText)) and (preg_match("/\s*FN ISI Export Format\s*/", $sourceText)))
  80. {
  81. // Convert ISI WoS format to CSA format:
  82. // (this allows for import of ISI WoS records via the import form provided by 'import_csa.php')
  83. $sourceText = IsiToCsa($sourceText); // function 'IsiToCsa()' is defined in 'import.inc.php'
  84. }
  85. // Check whether we're supposed to display the original source data:
  86. if (isset($formVars['showSource']))
  87. $showSource = $formVars['showSource'];
  88. else
  89. $showSource = "";
  90. if (isset($_SESSION['user_permissions']) AND preg_match("/allow_batch_import/", $_SESSION['user_permissions'])) // if the 'user_permissions' session variable does contain 'allow_batch_import'...
  91. {
  92. // Check whether we're supposed to import all records ('all') or just particular ones ('only'):
  93. $importRecordsRadio = $formVars['importRecordsRadio'];
  94. // Get the record numbers of those records that shall be imported:
  95. // examples of recognized formats: '1-5' imports the first five records; '1 3 7' will import records 1, 3 and 7; '1-3 5-7 9' will import records 1, 2, 3, 5, 6, 7 and 9
  96. // (note that the first three records could be labelled e.g. as 'Record 12 of 52', 'Record 30 of 112' and 'Record 202 of 533' but they must be referred to as records '1-3'
  97. // in the 'importRecords' form)
  98. $importRecords = $formVars['importRecords'];
  99. }
  100. else // if the user is only allowed to import one record at a time, we'll always import the very first record
  101. {
  102. $importRecordsRadio = "only";
  103. $importRecords = "1";
  104. }
  105. // Check whether we're supposed to skip records with unrecognized data format:
  106. if (isset($formVars['skipBadRecords']))
  107. $skipBadRecords = $formVars['skipBadRecords'];
  108. else
  109. $skipBadRecords = "";
  110. // --------------------------------------------------------------------
  111. // Do some pre-processing of the data input:
  112. // Process record number input:
  113. $importRecordNumbersArray = array(); // initialize array variable which will hold all the record numbers that shall be imported
  114. if (!empty($importRecords))
  115. {
  116. // split input string on all but digits or the hyphen ("-") character:
  117. // (the 'PREG_SPLIT_NO_EMPTY' flag causes only non-empty pieces to be returned)
  118. $importRecordsArray = preg_split("/[^0-9-]+/", $importRecords, -1, PREG_SPLIT_NO_EMPTY); // this keeps only elements such as '1', '3-5', '3-5-9' or '3-' (we'll deal with the last two cases below)
  119. foreach ($importRecordsArray as $importRecordsElement)
  120. {
  121. if (preg_match("/\d+-\d+/", $importRecordsElement)) // if we're dealing with a range of record numbers (such as '1-5')
  122. {
  123. $importRecordsElementArray = preg_split("/-/", $importRecordsElement); // split input string on hyphen ("-") character
  124. // generate an array that includes all numbers from start number to end number:
  125. // (in case of incorrect input (such as '3-5-9') we'll only take the first two numbers and ignore anything else)
  126. $importRecordRangeArray = range($importRecordsElementArray[0], $importRecordsElementArray[1]);
  127. foreach ($importRecordRangeArray as $importRecordNumber) // append all record numbers within range to array
  128. $importRecordNumbersArray[] = $importRecordNumber;
  129. }
  130. else // this element contains just a single record number
  131. {
  132. // append this record number to array:
  133. $importRecordNumbersArray[] = preg_replace("/(\d+).*/", "\\1", $importRecordsElement); // we account for the case that '$importRecordsElement' contains something like '3-'
  134. }
  135. }
  136. }
  137. // validation will throw up an error if we're supposed to import only particular records but no record numbers were specified
  138. // Remove any duplicate record number(s) from the list of extracted record numbers:
  139. $importRecordNumbersArray = array_unique($importRecordNumbersArray);
  140. // Extract the first record identifier of the input data (e.g. "\nRecord 1 of ...\n"):
  141. // this will also allow to paste record data that don't start with "\nRecord 1 of ...\n" but with e.g. "\nRecord 3 of ...\n"
  142. // (note that I couldn't figure out a non-greedy regex pattern to reliably extract the *very first* record identifier
  143. // which is why I'm using this 'preg_split()' workaround here)
  144. // - split on record identifiers ("\nRecord ... of ...\n") and include the found identifiers (i.e. the delimiters) as additional array elements ('PREG_SPLIT_DELIM_CAPTURE' flag):
  145. $recordsAndRecordNumbersArray = preg_split("/\s*(Record \d+ of \d+)\s*/", $sourceText, -1, PREG_SPLIT_DELIM_CAPTURE);
  146. $firstRecordIdentifier = "Record 1 of \d+"; // establish a default search pattern for the regex replace action below (otherwise, if '$firstRecordIdentifier' would be empty, all of the '$sourceText' would be deleted!)
  147. for ($i=0; $i<(count($recordsAndRecordNumbersArray)); $i++)
  148. {
  149. if (preg_match("/^Record \d+ of \d+$/", $recordsAndRecordNumbersArray[$i]))
  150. {
  151. // - extract the first record identifier
  152. $firstRecordIdentifier = $recordsAndRecordNumbersArray[$i];
  153. break;
  154. }
  155. }
  156. // Remove any text preceeding the actual record data as well as the first record identifier ("\nRecord ... of ...\n"):
  157. // CAUTION: Note that record identifiers must be unique among the pasted records! If e.g. the record identifier in '$firstRecordIdentifier'
  158. // occurs twice, the line below will remove all records up to the *second* occurrence of this identifier!
  159. $trimmedSourceText = preg_replace("/.*\s*" . $firstRecordIdentifier . "(?=\D)\s*/s", "", $sourceText);
  160. // Split input text on the header text preceeding each CSA record (e.g. "\nRecord 4 of 52\n"):
  161. $recordArray = preg_split("/\s*Record \d+ of \d+\s*/", $trimmedSourceText);
  162. $recordsCount = count($recordArray); // count how many records are available
  163. // --------------------------------------------------------------------
  164. // VALIDATE data fields:
  165. // Verify that some source text was given:
  166. if (empty($sourceText)) // no source data given
  167. {
  168. $errors["sourceText"] = "Source data missing!";
  169. }
  170. // Validate the 'importRecords' text entry field...
  171. if ($importRecordsRadio == "only") // ...if we're supposed to import only particular records
  172. {
  173. // ...make sure that some records were specified and that they are actually available in the input data:
  174. if (empty($importRecords) OR !preg_match("/[0-9]/", $importRecords)) // partial import requested but no record numbers given
  175. {
  176. $errors["importRecords"] = "Record number(s) missing!";
  177. }
  178. else // if some record numbers were given, check that these numbers are actually available in the input data:
  179. {
  180. $availableRecordNumbersArray = range(1, $recordsCount); // construct an array of available record numbers
  181. // get all record numbers to import which are NOT available in the source data:
  182. $importRecordNumbersNotAvailableArray = array_diff($importRecordNumbersArray, $availableRecordNumbersArray); // get all unique array elements from '$importRecordNumbersArray' that are not present in '$availableRecordNumbersArray'
  183. // just FYI, the line below would get all record numbers to import which ARE actually available in the source data:
  184. // $importRecordNumbersAvailableArray = array_diff($importRecordNumbersArray, $importRecordNumbersNotAvailableArray); // get all unique array elements from '$importRecordNumbersArray' that are not present in '$importRecordNumbersNotAvailableArray'
  185. if (!empty($importRecordNumbersNotAvailableArray)) // the user did request to import some record(s) that don't exist in the pasted source data
  186. {
  187. if ($recordsCount == 1) // one record available
  188. $errors["importRecords"] = "Only one record available! You can only use record number '1'.";
  189. else // several records available
  190. $errors["importRecords"] = "Only " . $recordsCount . " records available! You can only use record numbers '1-" . $recordsCount . "'.";
  191. }
  192. }
  193. }
  194. // the user did enter some source text and did input some recognized record numbers
  195. if (!empty($sourceText))
  196. {
  197. $importRecordNumbersRecognizedFormatArray = array(); // initialize array variable which will hold all record numbers of those records that shall be imported AND which were of a recognized format
  198. $importRecordNumbersNotRecognizedFormatArray = array(); // same for all records that shall be imported BUT which had an UNrecognized format
  199. // Verify the record format:
  200. for ($i=0; $i<$recordsCount; $i++) // for each record...
  201. {
  202. $singleRecord = $recordArray[$i];
  203. if (($importRecordsRadio == "only") AND (!in_array(($i+1), $importRecordNumbersArray))) // if we're NOT supposed to import this record... ('$i' starts with 0 so we have to add 1 to point to the correct record number)
  204. {
  205. continue; // process next record (if any)
  206. }
  207. else // ...validate the format of the current record:
  208. {
  209. // We assume a single record as valid if the '$singleRecord' variable is not empty and if it does contain
  210. // at least the following three field identifiers: "TI: Title", "SO: Source", "AU: Author" (only exception: for book monographs we accept "ED: Editor" instead of "AU: Author")
  211. // In addition, each of these field identifiers must be followed by a return and/or newline and four spaces!
  212. if (!empty($singleRecord) AND preg_match("/^TI: Title *[\r\n]+ {4,4}/m", $singleRecord) AND preg_match("/^SO: Source *[\r\n]+ {4,4}/m", $singleRecord) AND (preg_match("/^AU: Author *[\r\n]+ {4,4}/m", $singleRecord) OR (preg_match("/^ED: Editor *[\r\n]+ {4,4}/m", $singleRecord) AND preg_match("/^(PT: Publication Type\s+Book Monograph|DT: Document Type\s+B)/m", $singleRecord))))
  213. {
  214. $importRecordNumbersRecognizedFormatArray[] = $i + 1; // append this record number to the list of numbers whose record format IS recognized ('$i' starts with 0 so we have to add 1 to point to the correct record number)
  215. }
  216. else // unrecognized record format
  217. {
  218. $importRecordNumbersNotRecognizedFormatArray[] = $i + 1; // append this record number to the list of numbers whose record format is NOT recognized
  219. // prepare an appropriate error message:
  220. $errorMessage = "Record " . ($i + 1) . ": Unrecognized data format!";
  221. $emptyFieldsArray = array();
  222. // check for required fields:
  223. if (!preg_match("/^TI: Title *[\r\n]+ {4,4}/m", $singleRecord)) // required field empty: 'title'
  224. $emptyFieldsArray[] = "title";
  225. if (!preg_match("/^AU: Author *[\r\n]+ {4,4}/m", $singleRecord) AND !preg_match("/^(PT: Publication Type\s+Book Monograph|DT: Document Type\s+B)/m", $singleRecord)) // non-books: required field empty: 'author'
  226. $emptyFieldsArray[] = "author";
  227. elseif (!preg_match("/^AU: Author *[\r\n]+ {4,4}/m", $singleRecord) AND !preg_match("/^ED: Editor *[\r\n]+ {4,4}/m", $singleRecord) AND preg_match("/^(PT: Publication Type\s+Book Monograph|DT: Document Type\s+B)/m", $singleRecord)) // books: required fields empty: 'author' AND 'editor' (for books, either 'author' or 'editor' must be given)
  228. $emptyFieldsArray[] = "author (or editor)";
  229. if (!preg_match("/^SO: Source *[\r\n]+ {4,4}/m", $singleRecord)) // required field empty: 'source'
  230. $emptyFieldsArray[] = "source";
  231. if (!empty($emptyFieldsArray)) // some required fields were missing
  232. {
  233. if (count($emptyFieldsArray) == 1) // one field missing
  234. $errorMessage .= " Required field missing: " . $emptyFieldsArray[0];
  235. else // several fields missing
  236. $errorMessage .= " Required fields missing: " . implode(', ', $emptyFieldsArray);
  237. }
  238. if (!isset($errors["sourceText"]))
  239. $errors["sourceText"] = $errorMessage;
  240. else
  241. $errors["sourceText"] = $errors["sourceText"] . "<br>" . $errorMessage;
  242. }
  243. }
  244. }
  245. if (empty($importRecordNumbersRecognizedFormatArray)) // if none of the records to import had a recognized format
  246. {
  247. // we'll file an additional error element here, which will indicate whether the 'Skip records with unrecognized data format' checkbox shall be displayed or not
  248. $errors["badRecords"] = "all";
  249. if (count($importRecordNumbersNotRecognizedFormatArray) > 1) // if the user attempted to import more than one record
  250. $errors["skipBadRecords"] = "Sorry, but all of the specified records were of unrecognized data format!";
  251. else // user tried to import one single record (will be also triggered if '$importRecords' is empty)
  252. $errors["skipBadRecords"] = ""; // we insert an empty 'skipBadRecords' element so that 'import_csa.php' does the right thing
  253. }
  254. elseif (!empty($importRecordNumbersNotRecognizedFormatArray)) // some records had a recognized format but some were NOT recognized
  255. {
  256. $errors["badRecords"] = "some"; // see note above
  257. $errors["skipBadRecords"] = "Skip records with unrecognized data format";
  258. }
  259. }
  260. // --------------------------------------------------------------------
  261. // Check if there were any validation errors:
  262. if (count($errors) > 0)
  263. {
  264. // we ignore errors regarding records with unrecognized format if:
  265. // - at least some of the specified records had a valid data format and
  266. // - the user did mark the 'Skip records with unrecognized data format' checkbox
  267. if (!(($errors["badRecords"] == "some") AND ($skipBadRecords == "1")))
  268. {
  269. // ...otherwise we'll redirect back to the CSA import form and present the error message(s):
  270. // Write back session variables:
  271. saveSessionVariable("errors", $errors); // function 'saveSessionVariable()' is defined in 'include.inc.php'
  272. saveSessionVariable("formVars", $formVars);
  273. // Redirect the browser back to the CSA import form:
  274. header("Location: " . $referer);
  275. exit; // >>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>> !EXIT! <<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<
  276. }
  277. }
  278. // --------------------------------------------------------------------
  279. // If we made it here, then the data is considered valid!
  280. // (3) PROCESS SOURCE DATA:
  281. $parsedRecordsArray = array(); // initialize array variable which will hold parsed data of all records that shall be imported
  282. // LOOP OVER EACH RECORD:
  283. for ($i=0; $i<$recordsCount; $i++) // for each record...
  284. {
  285. // if we're NOT supposed to import this record (because it was either not selected by the user -OR- because it did contain an unrecognized data format)
  286. if (!in_array(($i+1), $importRecordNumbersRecognizedFormatArray)) // '$i' starts with 0 so we have to add 1 to point to the correct record number
  287. {
  288. continue; // process next record (if any)
  289. }
  290. else // ...import the current record:
  291. {
  292. $singleRecord = $recordArray[$i];
  293. // if the "AU: Author" field is missing BUT the "ED: Editor" is present (which is allowed for book monographs, see above):
  294. // we replace the "ED: Editor" field identifier with "AU: Author" (this will keep any " (ed)" and " (eds)" tags in place which, in turn, will cause the "is Editor" checkbox in 'record.php' to get marked)
  295. if (!preg_match("/^AU: Author *[\r\n]+ {4,4}/m", $singleRecord) AND preg_match("/^ED: Editor *[\r\n]+ {4,4}/m", $singleRecord) AND preg_match("/^(PT: Publication Type\s+Book Monograph|DT: Document Type\s+B)/m", $singleRecord))
  296. $singleRecord = preg_replace("/^ED: Editor(?= *[\r\n]+ {4,4})/m", "AU: Author", $singleRecord);
  297. $fieldArray = preg_split("/[\r\n]+(?=\w\w: )/", $singleRecord); // split each record into its fields
  298. // initialize some variables:
  299. $fieldParametersArray = array(); // setup an empty array (it will hold the parameters that get passed to 'record.php')
  300. $additionalDocumentTypeInfo = ""; // will be used with the "PT: Publication Type" field
  301. $environmentalRegime = ""; // will be used with the "ER: Environmental Regime" field
  302. // GENERATE EXTRA FIELDS:
  303. // check if the fields "MT: Monograph Title", "JN: Journal Name", "JV: Journal Volume", "JI: Journal Issue" and "JP: Journal Pages" are present,
  304. // if not, we attempt to generate them from the "SO: Source" field:
  305. $sourceField = preg_replace("/.*SO: Source *[\r\n]+ {4,4}(.+?)(?=([\r\n]+\w\w: |\s*\z)).*/ms", "\\1", $singleRecord); // first, we need to extract the "SO: Source" field data from the record text
  306. $sourceField = preg_replace("/\s{2,}/", " ", $sourceField); // remove any hard returns and extra spaces within the source field data string
  307. // if the current record is of type "Book Monograph" but the field "MT: Monograph Title" is missing:
  308. if (preg_match("/^(PT: Publication Type\s+Book Monograph|DT: Document Type\s+B)/m", $singleRecord) AND !preg_match("/^MT: Monograph Title *[\r\n]+ {4,4}/m", $singleRecord))
  309. {
  310. $extractedSourceFieldData = preg_replace("/^([^.[]+).*/", "\\1", $sourceField); // attempt to extract the full monograph title from the source field
  311. if (preg_match("/^[[:upper:]\W\d]+$/", $extractedSourceFieldData)) // if all of the words within the monograph title are uppercase, we attempt to convert the string to something more readable:
  312. $extractedSourceFieldData = ucwords(strtolower($extractedSourceFieldData)); // perform case transformation (e.g. convert "BIOLOGY AND ECOLOGY OF GLACIAL RELICT CRUSTACEA" into "Biology And Ecology Of Glacial Relict Crustacea")
  313. $fieldArray[] = "MT: Monograph Title\r\n " . $extractedSourceFieldData; // add field "MT: Monograph Title" to the array of fields
  314. }
  315. // else if the current record is of type "Journal Article", "Report", etc (or wasn't specified) but the field "JN: Journal Name" is missing:
  316. elseif (!preg_match("/^JN: Journal Name *[\r\n]+ {4,4}/m", $singleRecord)) // preg_match("/^(PT: Publication Type\s+(Journal Article|Report)|DT: Document Type\s+(J|R))/m", $singleRecord)
  317. {
  318. if (preg_match("/\[/", $sourceField)) // if the source field data contain a square bracket we assume a format like: "Journal of Phycology [J. Phycol.]. Vol. 37, no. s3, pp. 18-18. Jun 2001."
  319. $extractedSourceFieldData = preg_replace("/^([^.[]+).*/", "\\1", $sourceField); // attempt to extract the full journal name from the source field
  320. else // source field format might be something like: "Phycologia, vol. 34, no. 2, pp. 135-144, 1995"
  321. $extractedSourceFieldData = preg_replace("/^([^.,]+).*/", "\\1", $sourceField); // attempt to extract the full journal name from the source field
  322. if (preg_match("/^[[:upper:]\W\d]+$/", $extractedSourceFieldData)) // if all of the words within the journal name are uppercase, we attempt to convert the string to something more readable:
  323. $extractedSourceFieldData = ucwords(strtolower($extractedSourceFieldData)); // perform case transformation (e.g. convert "POLAR BIOLOGY" into "Polar Biology")
  324. $fieldArray[] = "JN: Journal Name\r\n " . $extractedSourceFieldData; // add field "JN: Journal Name" to the array of fields
  325. }
  326. // if the "JV: Journal Volume" is missing BUT the "SO: Source" field contains a volume specification:
  327. if (!preg_match("/^JV: Journal Volume *[\r\n]+ {4,4}/m", $singleRecord) AND preg_match("/(?<=\W)vol[. ]+[\w\/-]+/i", $sourceField))
  328. {
  329. $extractedSourceFieldData = preg_replace("/.*(?<=\W)vol[. ]+([\w\/-]+).*/i", "\\1", $sourceField); // attempt to extract the journal volume from the source field
  330. $fieldArray[] = "JV: Journal Volume\r\n " . $extractedSourceFieldData; // add field "JV: Journal Volume" to the array of fields
  331. }
  332. // if the "JI: Journal Issue" is missing BUT the "SO: Source" field contains an issue specification:
  333. if (!preg_match("/^JI: Journal Issue *[\r\n]+ {4,4}/m", $singleRecord) AND preg_match("/(?<=\W)no[. ]+[\w\/-]+/i", $sourceField))
  334. {
  335. $extractedSourceFieldData = preg_replace("/.*(?<=\W)no[. ]+([\w\/-]+).*/i", "\\1", $sourceField); // attempt to extract the journal issue from the source field
  336. $fieldArray[] = "JI: Journal Issue\r\n " . $extractedSourceFieldData; // add field "JI: Journal Issue" to the array of fields
  337. }
  338. // if the "JP: Journal Pages" is missing BUT the "SO: Source" field contains a pages specification:
  339. if (!preg_match("/^JP: Journal Pages *[\r\n]+ {4,4}/m", $singleRecord) AND preg_match("/((?<=\W)pp?[. ]+[\w\/,-]+|[\d,]+ *pp\b)/i", $sourceField))
  340. {
  341. if (preg_match("/(?<=\W)pp?[. ]+[\w\/,-]+/i", $sourceField)) // e.g. "pp. 212-217" or "p. 216" etc
  342. $extractedSourceFieldData = preg_replace("/.*(?<=\W)pp?[. ]+([\w\/,-]+).*/i", "\\1", $sourceField); // attempt to extract the journal pages from the source field
  343. elseif (preg_match("/[\d,]+ *pp\b/", $sourceField)) // e.g. "452 pp"
  344. $extractedSourceFieldData = preg_replace("/.*?([\d,]+ *pp)\b.*/i", "\\1", $sourceField); // attempt to extract the journal pages from the source field
  345. $extractedSourceFieldData = preg_replace("/,/", "", $extractedSourceFieldData); // remove any thousands separators from journal pages
  346. $fieldArray[] = "JP: Journal Pages\r\n " . $extractedSourceFieldData; // add field "JP: Journal Pages" to the array of fields
  347. }
  348. // Additionally, we extract the abbreviated journal name from the "SO: Source" field (if available):
  349. if (preg_match("/\[/", $sourceField)) // if the source field data contain a square bracket we assume a format like: "Journal of Phycology [J. Phycol.]. Vol. 37, no. s3, pp. 18-18. Jun 2001."
  350. {
  351. $extractedSourceFieldData = preg_replace("/.*\[(.+?)\].*/", "\\1", $sourceField); // attempt to extract the abbreviated journal name from the source field
  352. $extractedSourceFieldData = preg_replace("/\./", "", $extractedSourceFieldData); // remove any dots from the abbreviated journal name
  353. if (preg_match("/^[[:upper:]\W\d]+$/", $extractedSourceFieldData)) // if all of the words within the abbreviated journal name are uppercase, we attempt to convert the string to something more readable:
  354. $extractedSourceFieldData = ucwords(strtolower($extractedSourceFieldData)); // perform case transformation (e.g. convert "BALT SEA ENVIRON PROC" into "Balt Sea Environ Proc")
  355. $fieldArray[] = "JA: Abbrev Journal Name\r\n " . $extractedSourceFieldData; // add field "JA: Abbrev Journal Name" to the array of fields (note that this field normally does NOT occur within the CSA full record format!)
  356. }
  357. // (END GENERATE EXTRA FIELDS)
  358. // LOOP OVER EACH FIELD:
  359. foreach ($fieldArray as $singleField) // for each field within the current record...
  360. {
  361. $singleField = preg_replace("/^(\w\w: [^\r\n]+)[\r\n]+ {4,4}/", "\\1___LabelDataSplitter___", $singleField); // insert a unique text string between the field identifier and the field data
  362. $fieldLabelPlusDataArray = preg_split("/___LabelDataSplitter___/", $singleField); // split each field into a 2-element array containing [0] the field identifier and [1] the field data
  363. $fieldLabelPlusDataArray[1] = preg_replace("/\s{2,}/", " ", $fieldLabelPlusDataArray[1]); // remove any hard returns and extra spaces within the data string
  364. $fieldLabelPlusDataArray[1] = trim($fieldLabelPlusDataArray[1]); // remove any preceeding and trailing whitespace from the field data
  365. if (preg_match("/AU: Author/", $fieldLabelPlusDataArray[0]))
  366. $fieldLabelPlusDataArray[1] = preg_replace("/\*/", "", $fieldLabelPlusDataArray[1]); // remove any asterisk ("*")
  367. elseif (preg_match("/ED: Editor/", $fieldLabelPlusDataArray[0]))
  368. $fieldLabelPlusDataArray[1] = preg_replace("/ \(eds?\)(?= *$| *;)/", "", $fieldLabelPlusDataArray[1]); // remove " (ed)" and/or " (eds)"
  369. elseif (preg_match("/TI: Title|AB: Abstract/", $fieldLabelPlusDataArray[0]))
  370. {
  371. if (preg_match("/TI: Title/", $fieldLabelPlusDataArray[0]))
  372. {
  373. $fieldLabelPlusDataArray[1] = preg_replace("/--/", "-", $fieldLabelPlusDataArray[1]); // remove en-dash markup
  374. $fieldLabelPlusDataArray[1] = preg_replace("/ *\. *$/", "", $fieldLabelPlusDataArray[1]); // remove any dot from end of title
  375. }
  376. if (preg_match("/ su(b|per)\(.+?\)/", $fieldLabelPlusDataArray[1]))
  377. $fieldLabelPlusDataArray[1] = preg_replace("/ (su(?:b|per))\((.+?)\)/", "[\\1:\\2]", $fieldLabelPlusDataArray[1]); // transform " sub(...)" & " super(...)" markup into "[sub:...]" & "[super:...]" markup
  378. if (preg_match("/(?<= )mu /", $fieldLabelPlusDataArray[1]))
  379. $fieldLabelPlusDataArray[1] = preg_replace("/(?<= )mu /", "", $fieldLabelPlusDataArray[1]); // transform "mu " markup into "�" markup
  380. }
  381. // BUILD FIELD PARAMETERS:
  382. // build an array of key/value pairs:
  383. // "AU: Author":
  384. if (preg_match("/AU: Author/", $fieldLabelPlusDataArray[0]))
  385. $fieldParametersArray['author'] = $fieldLabelPlusDataArray[1];
  386. // "TI: Title":
  387. elseif (preg_match("/TI: Title/", $fieldLabelPlusDataArray[0]))
  388. $fieldParametersArray['title'] = $fieldLabelPlusDataArray[1];
  389. // "PT: Publication Type":
  390. elseif (preg_match("/PT: Publication Type/", $fieldLabelPlusDataArray[0])) // could also check for "DT: Document Type" (but DT was added only recently)
  391. {
  392. if (preg_match("/[;:,.]/", $fieldLabelPlusDataArray[1])) // if the "PT: Publication Type" field contains a delimiter (e.g. like: "Journal Article; Conference")
  393. {
  394. $correctDocumentType = preg_replace("/(.+?)\s*[;:,.]\s*.*/", "\\1", $fieldLabelPlusDataArray[1]); // extract everything before this delimiter
  395. $additionalDocumentTypeInfo = preg_replace("/.*?\s*[;:,.]\s*(.+)/", "\\1", $fieldLabelPlusDataArray[1]); // extract everything after this delimiter
  396. $additionalDocumentTypeInfo = $additionalDocumentTypeInfo; // this info will be appended to any notes field data (see below)
  397. }
  398. else // we take the "PT: Publication Type" field contents as they are
  399. $correctDocumentType = $fieldLabelPlusDataArray[1];
  400. // Note that for books the "PT: Publication Type" field will always start with "Book Monograph", no matter whether the referenced
  401. // publication is a whole book or just a book chapter within that book! This is a design flaw within the CSA full record format.
  402. // So we can only apply some "good guessing" whether the current record actually references a complete book or just a book chapter:
  403. if (preg_match("/^(PT: Publication Type\s+Book Monograph|DT: Document Type\s+B)/m", $singleRecord)) // if the current record is of type "Book Monograph"
  404. {
  405. // and if the source field contains some page specification like "213 pp." (AND NOT something like "pp. 76-82" or "p. 216")...
  406. if (preg_match("/[\d,]+ *pp\b/i", $sourceField) AND !preg_match("/(?<=\W)pp?[. ]+[\w\/,-]+/i", $sourceField))
  407. $correctDocumentType = "Book Whole"; // ...we assume its a whole book
  408. else
  409. $correctDocumentType = "Book Chapter"; // ...otherwise we assume its a book chapter (which may NOT always be correct!)
  410. }
  411. $fieldParametersArray['type'] = $correctDocumentType;
  412. }
  413. // "PY: Publication Year":
  414. elseif (preg_match("/PY: Publication Year/", $fieldLabelPlusDataArray[0]))
  415. $fieldParametersArray['year'] = $fieldLabelPlusDataArray[1];
  416. // "JN: Journal Name":
  417. elseif (preg_match("/JN: Journal Name/", $fieldLabelPlusDataArray[0]))
  418. {
  419. // if the current record is of type "Book Monograph" AND the field "JN: Journal Name" was given within the *original* record data (i.e., before adding stuff to it):
  420. if (preg_match("/^(PT: Publication Type\s+Book Monograph|DT: Document Type\s+B)/m", $singleRecord) AND preg_match("/^JN: Journal Name *[\r\n]+ {4,4}/m", $singleRecord))
  421. // for book monographs the publication title is given in "MT: Monograph Title"; if a "JN: Journal Name" was originally provided as well, we assume, it's the series title:
  422. $fieldParametersArray['series_title'] = $fieldLabelPlusDataArray[1];
  423. else
  424. $fieldParametersArray['publication'] = $fieldLabelPlusDataArray[1];
  425. }
  426. // "JA: Abbrev Journal Name":
  427. elseif (preg_match("/JA: Abbrev Journal Name/", $fieldLabelPlusDataArray[0]))
  428. {
  429. if (preg_match("/^(PT: Publication Type\s+Book Monograph|DT: Document Type\s+B)/m", $singleRecord)) // if the current record is of type "Book Monograph"
  430. // for book monographs the publication title is given in "MT: Monograph Title"; if a "JA: Abbrev Journal Name" is provided as well, we assume, it's the abbreviated series title:
  431. $fieldParametersArray['abbrev_series_title'] = $fieldLabelPlusDataArray[1];
  432. else
  433. $fieldParametersArray['abbrev_journal'] = $fieldLabelPlusDataArray[1];
  434. }
  435. // "MT: Monograph Title":
  436. elseif (preg_match("/MT: Monograph Title/", $fieldLabelPlusDataArray[0]))
  437. {
  438. // if the source field contains some page specification like "213 pp." (AND NOT something like "pp. 76-82" or "p. 216")...
  439. if (preg_match("/[\d,]+ *pp\b/i", $sourceField) AND !preg_match("/(?<=\W)pp?[. ]+[\w\/,-]+/i", $sourceField))
  440. // ...we assume its a whole book (see above comment), in which case we assign the monograph title to the series title field:
  441. $fieldParametersArray['series_title'] = $fieldLabelPlusDataArray[1];
  442. else
  443. $fieldParametersArray['publication'] = $fieldLabelPlusDataArray[1];
  444. }
  445. // "JV: Journal Volume":
  446. elseif (preg_match("/JV: Journal Volume/", $fieldLabelPlusDataArray[0]))
  447. {
  448. if (preg_match("/^(PT: Publication Type\s+Book Monograph|DT: Document Type\s+B)/m", $singleRecord)) // if the current record is of type "Book Monograph"
  449. // for book monographs, if there's a volume given, we assume, it's the series volume:
  450. $fieldParametersArray['series_volume'] = $fieldLabelPlusDataArray[1];
  451. else
  452. $fieldParametersArray['volume'] = $fieldLabelPlusDataArray[1];
  453. }
  454. // "JI: Journal Issue":
  455. elseif (preg_match("/JI: Journal Issue/", $fieldLabelPlusDataArray[0]))
  456. {
  457. if (preg_match("/^(PT: Publication Type\s+Book Monograph|DT: Document Type\s+B)/m", $singleRecord)) // if the current record is of type "Book Monograph"
  458. // for book monographs, if there's an issue given, we assume, it's the series issue:
  459. $fieldParametersArray['series_issue'] = $fieldLabelPlusDataArray[1];
  460. else
  461. $fieldParametersArray['issue'] = $fieldLabelPlusDataArray[1];
  462. }
  463. // "JP: Journal Pages":
  464. elseif (preg_match("/JP: Journal Pages/", $fieldLabelPlusDataArray[0]))
  465. $fieldParametersArray['pages'] = $fieldLabelPlusDataArray[1];
  466. // "AF: Affiliation" & "AF: Author Affilition":
  467. elseif (preg_match("/AF: (Author )?Affilia?tion/", $fieldLabelPlusDataArray[0]))
  468. $fieldParametersArray['address'] = $fieldLabelPlusDataArray[1];
  469. // "CA: Corporate Author":
  470. elseif (preg_match("/CA: Corporate Author/", $fieldLabelPlusDataArray[0]))
  471. $fieldParametersArray['corporate_author'] = $fieldLabelPlusDataArray[1];
  472. // "DE: Descriptors":
  473. elseif (preg_match("/DE: Descriptors/", $fieldLabelPlusDataArray[0])) // currently, the fields "KW: Keywords" and "ID: Identifiers" are ignored!
  474. $fieldParametersArray['keywords'] = $fieldLabelPlusDataArray[1];
  475. // "AB: Abstract":
  476. elseif (preg_match("/AB: Abstract/", $fieldLabelPlusDataArray[0]))
  477. $fieldParametersArray['abstract'] = $fieldLabelPlusDataArray[1];
  478. // "PB: Publisher":
  479. elseif (preg_match("/PB: Publisher/", $fieldLabelPlusDataArray[0]))
  480. $fieldParametersArray['publisher'] = $fieldLabelPlusDataArray[1];
  481. // "ED: Editor":
  482. elseif (preg_match("/ED: Editor/", $fieldLabelPlusDataArray[0]))
  483. $fieldParametersArray['editor'] = $fieldLabelPlusDataArray[1];
  484. // "LA: Language":
  485. elseif (preg_match("/LA: Language/", $fieldLabelPlusDataArray[0]))
  486. $fieldParametersArray['language'] = $fieldLabelPlusDataArray[1];
  487. // "SL: Summary Language":
  488. elseif (preg_match("/SL: Summary Language/", $fieldLabelPlusDataArray[0]))
  489. $fieldParametersArray['summary_language'] = $fieldLabelPlusDataArray[1];
  490. // "OT: Original Title":
  491. elseif (preg_match("/OT: Original Title/", $fieldLabelPlusDataArray[0]))
  492. $fieldParametersArray['orig_title'] = $fieldLabelPlusDataArray[1];
  493. // "IS: ISSN":
  494. elseif (preg_match("/IS: ISSN/", $fieldLabelPlusDataArray[0]))
  495. $fieldParametersArray['issn'] = $fieldLabelPlusDataArray[1];
  496. // "IB: ISBN":
  497. elseif (preg_match("/IB: ISBN/", $fieldLabelPlusDataArray[0]))
  498. $fieldParametersArray['isbn'] = $fieldLabelPlusDataArray[1];
  499. // "ER: Environmental Regime":
  500. elseif (preg_match("/ER: Environmental Regime/", $fieldLabelPlusDataArray[0]))
  501. $environmentalRegime = $fieldLabelPlusDataArray[1]; // this info will be appended to any notes field data (see below)
  502. // "CF: Conference":
  503. elseif (preg_match("/CF: Conference/", $fieldLabelPlusDataArray[0]))
  504. $fieldParametersArray['conference'] = $fieldLabelPlusDataArray[1];
  505. // "NT: Notes":
  506. elseif (preg_match("/NT: Notes/", $fieldLabelPlusDataArray[0]))
  507. $fieldParametersArray['notes'] = $fieldLabelPlusDataArray[1];
  508. // "DO: DOI":
  509. elseif (preg_match("/DO: DOI/", $fieldLabelPlusDataArray[0]))
  510. $fieldParametersArray['doi'] = $fieldLabelPlusDataArray[1];
  511. }
  512. // (END LOOP OVER EACH FIELD)
  513. if (!empty($showSource)) // if we're supposed to display the original source data
  514. // append original source field data (they will be presented within the header message of 'record.php' for easy comparison with the extracted data):
  515. $fieldParametersArray['source'] = $sourceField;
  516. // we'll hack the "notes" element in order to append additional info:
  517. // (this cannot be done earlier above since we don't know about the presence & order of fields within the source text!)
  518. if (!empty($additionalDocumentTypeInfo)) // if the "PT: Publication Type" field contains some additional info
  519. {
  520. if (isset($fieldParametersArray['notes'])) // and if the notes element is present
  521. $fieldParametersArray['notes'] = $fieldParametersArray['notes'] . "; " . $additionalDocumentTypeInfo; // append additional info from "PT: Publication Type" field
  522. else // the notes parameter wasn't specified yet
  523. $fieldParametersArray['notes'] = $additionalDocumentTypeInfo; // add notes element with additional info from "PT: Publication Type" field
  524. }
  525. if (!empty($environmentalRegime)) // if the "ER: Environmental Regime" field contains some data
  526. {
  527. if (isset($fieldParametersArray['notes'])) // and if the notes element is present
  528. $fieldParametersArray['notes'] = $fieldParametersArray['notes'] . "; " . $environmentalRegime; // append "ER: Environmental Regime" field data
  529. else // the notes parameter wasn't specified yet
  530. $fieldParametersArray['notes'] = $environmentalRegime; // add notes element with "ER: Environmental Regime" field data
  531. }
  532. // Append the array of extracted field data to the main data array which holds all records to import:
  533. $parsedRecordsArray[] = $fieldParametersArray;
  534. }
  535. }
  536. // (END LOOP OVER EACH RECORD)
  537. // --------------------------------------------------------------------
  538. // IMPORT RECORDS:
  539. if (count($importRecordNumbersRecognizedFormatArray) == 1) // if this is the only record we'll need to import:
  540. {
  541. // we can use '$fieldParametersArray' directly here, since it still holds the data of the *one* record that we're supposed to import
  542. foreach ($fieldParametersArray as $fieldParameterKey => $fieldParameterValue)
  543. $fieldParametersArray[$fieldParameterKey] = $fieldParameterKey . "=" . rawurlencode($fieldParameterValue); // copy parameter name and equals sign in front of parameter value
  544. $fieldParameters = implode("&", $fieldParametersArray); // merge list of parameters
  545. // RELOCATE TO IMPORT PAGE:
  546. // call 'record.php' and load the form fields with the data of the current record
  547. header("Location: record.php?recordAction=add&mode=import&importSource=csa&" . $fieldParameters);
  548. exit; // >>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>> !EXIT! <<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<
  549. }
  550. else // import record(s) directly:
  551. {
  552. // Build an array structure suitable for passing to the 'addRecords()' function:
  553. $importDataArray = array(); // for an explanation of the structure of '$importDataArray', see the comments above the 'addRecords()' function (in 'include.inc.php')
  554. $importDataArray['type'] = "refbase"; // we use the "refbase" array format
  555. $importDataArray['version'] = "1.0"; // we use version "1.0" of the array structure
  556. $importDataArray['creator'] = "http://refbase.net"; // calling script/importer is "refbase"
  557. $importDataArray['author'] = "Matthias Steffens"; // author/contact name of the person who's responsible for this script/importer
  558. $importDataArray['contact'] = "refbase@extracts.de"; // author's email/contact address
  559. $importDataArray['options'] = array('prefix_call_number' => "true"); // if "true", any 'call_number' string will be prefixed with the correct call number prefix of the currently logged-in user (e.g. 'IP� @ msteffens @ ')
  560. $importDataArray['records'] = $parsedRecordsArray; // this array will hold the record(s) (with each record being a sub-array of fields)
  561. // NOTES: - due to the nature of the CSA format, this importer doesn't provide input for the following fields:
  562. // 'place', 'series_editor', 'edition', 'medium', 'area', 'expedition', 'call_number', 'approved', 'file', 'thesis', 'url', 'contribution_id', 'online_publication', 'online_citation', 'orig_record'
  563. // - the 'addRecords()' function will take care of the calculation fields ('first_author', 'author_count', 'first_page', 'volume_numeric' and 'series_volume_numeric')
  564. // - similarly, the *date/*time/*by fields ('created_date', 'created_time', 'created_by', 'modified_date', 'modified_time', 'modified_by') will be filled automatically
  565. // if no custom values (in correct date ['YYYY-MM-DD'] and time ['HH:MM:SS'] format) are given in the '$importDataArray'
  566. // - we could pass any custom info for the 'location' field with the '$importDataArray', omitting it here
  567. // causes the 'addRecords()' function to insert name & email address of the currently logged-in user (e.g. 'Matthias Steffens (refbase@extracts.de)')
  568. // - the serial number(s) will be assigned automatically (and returned by the 'addRecords()' function in form of an array)
  569. // - we don't add anything to the 'user_data' table since the import data don't contain any user-specific data
  570. // Add all records to the database (i.e., for each record, add a row entry to MySQL table 'refs'):
  571. // ('$importedRecordsArray' will hold the serial numbers of all newly imported records)
  572. $importedRecordsArray = addRecords($importDataArray); // function 'addRecords()' is defined in 'include.inc.php'
  573. }
  574. // --------------------------------------------------------------------
  575. // (4) DISPLAY RESULTS
  576. if (!empty($importedRecordsArray)) // if some records were successfully imported
  577. {
  578. $recordSerialsQueryString = implode(",", $importedRecordsArray);
  579. $importedRecordsCount = count($importedRecordsArray);
  580. // Send EMAIL announcement:
  581. if ($sendEmailAnnouncements == "yes")
  582. {
  583. // variables '$sendEmailAnnouncements', '$mailingListEmail', '$officialDatabaseName' and '$databaseBaseURL' are specified in 'ini.inc.php';
  584. // '$loginFirstName' and '$loginLastName' are provided as session variables by the 'start_session()' function in 'include.inc.php'
  585. // send a notification email to the mailing list email address given in '$mailingListEmail':
  586. $emailRecipient = "Literature Database Announcement List <" . $mailingListEmail . ">";
  587. if ($importedRecordsCount == 1)
  588. {
  589. $emailSubject = "New record added to the " . $officialDatabaseName;
  590. $emailBodyIntro = "One record has been added to the " . $officialDatabaseName . ":";
  591. $detailsURL = $databaseBaseURL . "show.php?record=" . $importedRecordsArray[0];
  592. }
  593. else // $importedRecordsCount > 1
  594. {
  595. $emailSubject = "New records added to the " . $officialDatabaseName;
  596. $emailBodyIntro = $importedRecordsCount . " records have been added to the " . $officialDatabaseName . ":";
  597. $detailsURL = $databaseBaseURL . "show.php?records=" . $recordSerialsQueryString;
  598. }
  599. $emailBody = $emailBodyIntro
  600. . "\n\n added by: " . $loginFirstName . " " . $loginLastName
  601. . "\n details: " . $detailsURL
  602. . "\n";
  603. sendEmail($emailRecipient, $emailSubject, $emailBody); // function 'sendEmail()' is defined in 'include.inc.php'
  604. }
  605. if ($importedRecordsCount == 1)
  606. $headerMessage = $importedRecordsCount . " " . $loc["RecordSuccessfullyImported"] . ":";
  607. else // $importedRecordsCount > 1
  608. $headerMessage = $importedRecordsCount . " " . $loc["RecordsSuccessfullyImported"] . ":";
  609. // DISPLAY all newly added records:
  610. header("Location: show.php?records=" . $recordSerialsQueryString . "&headerMsg=" . rawurlencode($headerMessage));
  611. }
  612. else // nothing imported
  613. {
  614. // we'll file again this additional error element here so that the 'errors' session variable isn't empty causing 'import_csa.php' to re-load the form data that were submitted by the user
  615. $errors["badRecords"] = "all";
  616. // return an appropriate error message:
  617. $HeaderString = returnMsg($loc["NoRecordsImported"] . "!", "warning", "strong", "HeaderString"); // function 'returnMsg()' is defined in 'include.inc.php'
  618. // Write back session variables:
  619. saveSessionVariable("errors", $errors); // function 'saveSessionVariable()' is defined in 'include.inc.php'
  620. saveSessionVariable("formVars", $formVars);
  621. header("Location: " . $referer); // redirect to the calling page (normally, 'import_csa.php')
  622. }
  623. // --------------------------------------------------------------------
  624. ?>