51 'shortname'=>array(
'shortname',
'Short Name'),
52 'licensetype'=>array(
'licensetype',
'License Type'),
53 'fullname'=>array(
'fullname',
'Long Name'),
54 'spdx_id'=>array(
'spdx_id',
'SPDX ID'),
55 'text'=>array(
'text',
'Full Text'),
56 'parent_shortname'=>array(
'parent_shortname',
'Decider Short Name'),
57 'report_shortname'=>array(
'report_shortname',
'Regular License Text Short Name'),
58 'url'=>array(
'url',
'URL'),
59 'notes'=>array(
'notes'),
60 'source'=>array(
'source',
'Foreign ID'),
61 'risk'=>array(
'risk',
'risk_level'),
62 'group'=>array(
'group',
'License group'),
63 'obligations'=>array(
'obligations',
'License obligations'),
64 'external_id'=>array(
'id',
'LicenseDB Id'),
108 if (!is_file($filename) || ($handle = fopen($filename,
'r')) ===
false) {
109 return _(
'Internal error');
114 if ($fileExtension ==
'csv') {
115 while (($row = fgetcsv($handle,0,$this->delimiter,$this->enclosure)) !==
false) {
122 $msg .= _(
'Read csv').(
": $cnt ")._(
'licenses');
124 $jsonContent = fread($handle, filesize($filename));
125 $data = json_decode($jsonContent,
true);
126 if ($data ===
null && json_last_error() !== JSON_ERROR_NONE) {
127 $msg .=
"Error decoding JSON: " . json_last_error_msg() .
"\n";
130 $msg .= _(
'Read json').(
":". count($data) .
" ")._(
'licenses');
134 return $msg .= _(
'Error while parsing file').
': '.$e->getMessage();
148 $defaultValues = array(
149 'parent_shortname' =>
null,
150 'report_shortname' =>
null,
159 foreach ($row as $key => $value) {
161 foreach ($this->alias as $aliasKey => $aliasValues) {
162 if (in_array($key, $aliasValues)) {
167 $newArray[$newKey] = $value;
169 foreach ($defaultValues as $key => $defaultValue) {
170 if (!array_key_exists($key, $newArray)) {
171 $newArray[$key] = $defaultValue;
185 if ($this->headrow ===
null) {
191 foreach (array(
'shortname',
'fullname',
'text') as $needle) {
192 $mRow[$needle] = $row[$this->headrow[$needle]];
194 foreach (array(
'parent_shortname' =>
null,
'report_shortname' =>
null,
195 'url' =>
'',
'notes' =>
'',
'source' =>
'',
'risk' => 0,
196 'group' =>
null,
'spdx_id' =>
null,
'licensetype' =>
"Permisssive"
197 ) as $optNeedle=>$defaultValue) {
198 $mRow[$optNeedle] = $defaultValue;
199 if ($this->headrow[$optNeedle]!==
false && array_key_exists($this->headrow[$optNeedle], $row)) {
200 $mRow[$optNeedle] = $row[$this->headrow[$optNeedle]];
216 foreach (array(
'parent_shortname' =>
null,
'report_shortname' =>
null,
217 'url' =>
'',
'notes' =>
'',
'source' =>
'',
'risk' => 0,
218 'group' =>
null,
'spdx_id' =>
null,
'licensetype' =>
'Permisssive',
219 'obligation_ids' => array()
220 ) as $optNeedle => $defaultValue) {
221 if (!array_key_exists($optNeedle, $mRow)) {
222 $mRow[$optNeedle] = $defaultValue;
238 $row[0] =
trim($row[0],
"\xEF\xBB\xBF");
239 foreach (array(
'shortname',
'fullname',
'text') as $needle) {
240 $col = ArrayOperation::multiSearch($this->alias[$needle], $row);
241 if (
false === $col) {
242 throw new \Exception(
"Undetermined position of $needle");
246 foreach (array(
'parent_shortname',
'report_shortname',
'url',
'notes',
247 'source',
'risk',
'group',
'spdx_id',
'licensetype') as $optNeedle) {
248 $headrow[$optNeedle] = ArrayOperation::multiSearch($this->alias[$optNeedle], $row);
261 $stmt = __METHOD__ .
'.getOldLicense';
262 $oldLicense = $this->
dbManager->getSingleRow(
'SELECT ' .
263 'rf_shortname, rf_fullname, rf_spdx_id, rf_text, rf_url, rf_notes, rf_source, rf_risk, rf_licensetype ' .
264 'FROM license_ref WHERE rf_pk = $1', array($rfPk), $stmt);
266 $log =
"License '$row[shortname]' already exists in DB (id = $rfPk)";
267 $stmt = __METHOD__ .
'.updateLicense';
268 $sql =
"UPDATE license_ref SET ";
269 if (! empty($row[
'group'])) {
270 $sql =
"UPDATE license_candidate SET ";
272 $extraParams = array();
273 $param = array($rfPk);
274 if (isset($row[
'fullname']) && $row[
'fullname'] != $oldLicense[
'rf_fullname']) {
275 $param[] = $row[
'fullname'];
277 $extraParams[] =
"rf_fullname=$" . count($param);
278 $log .=
", updated fullname";
281 if (isset($row[
'external_id']) && isset($row[
'shortname']) && $row[
'shortname'] !== $oldLicense[
'rf_shortname']) {
282 $param[] = $row[
'shortname'];
284 $extraParams[] =
"rf_shortname=$" . count($param);
285 $log .=
", updated shortname";
287 if (isset($row[
'spdx_id']) && $row[
'spdx_id'] != $oldLicense[
'rf_spdx_id']) {
288 $param[] = $row[
'spdx_id'];
290 $extraParams[] =
"rf_spdx_id=$" . count($param);
291 $log .=
", updated SPDX ID";
293 if (isset($row[
'text']) && $row[
'text'] != $oldLicense[
'rf_text'] && $row[
'text'] != LicenseMap::TEXT_MAX_CHAR_LIMIT) {
294 $param[] = $row[
'text'];
296 $extraParams[] =
"rf_text=$" . count($param) .
",rf_md5=md5($" .
298 $log .=
", updated text";
300 if (isset($row[
'url']) && $row[
'url'] != $oldLicense[
'rf_url']) {
301 $param[] = $row[
'url'];
303 $extraParams[] =
"rf_url=$" . count($param);
304 $log .=
", updated URL";
306 if (isset($row[
'notes']) && $row[
'notes'] != $oldLicense[
'rf_notes']) {
307 $param[] = $row[
'notes'];
309 $extraParams[] =
"rf_notes=$" . count($param);
310 $log .=
", updated notes";
312 if (isset($row[
'source']) && $row[
'source'] != $oldLicense[
'rf_source']) {
313 $param[] = $row[
'source'];
314 $stmt .=
'.updSource';
315 $extraParams[] =
"rf_source=$".count($param);
316 $log .=
', updated the source';
318 if (isset($row[
'risk']) && $row[
'risk'] != $oldLicense[
'rf_risk']) {
319 $param[] = $row[
'risk'];
321 $extraParams[] =
"rf_risk=$".count($param);
322 $log .=
', updated the risk level';
324 if (isset($row[
'licensetype']) && $row[
'licensetype'] != $oldLicense[
'rf_licensetype']) {
325 $param[] = $row[
'licensetype'];
327 $extraParams[] =
"rf_licensetype=$".count($param);
328 $log .=
', updated the licensetype';
330 if (count($param) > 1) {
331 $sql .= join(
",", $extraParams);
332 $sql .=
" WHERE rf_pk=$1;";
333 $this->
dbManager->getSingleRow($sql, $param, $stmt);
334 if (!isset($row[
'external_id'])) {
335 $this->mdkMap[md5($row[
'text'])] = $rfPk;
341 if ($row[
'external_id'] ===
null) {
342 $stmt = __METHOD__ .
'.getOldMapping';
343 $sql =
'SELECT rf_parent FROM license_map WHERE rf_fk = $1 AND usage = $2;';
345 $oldParentRow = $this->
dbManager->getSingleRow($sql, array($rfPk,
346 LicenseMap::CONCLUSION), $stmt);
347 if (!empty($oldParentRow)) {
348 $oldParent = $oldParentRow[
'rf_parent'];
351 $oldReportRow = $this->
dbManager->getSingleRow($sql, array($rfPk,
352 LicenseMap::REPORT), $stmt);
353 if (!empty($oldReportRow)) {
354 $oldReport = $oldReportRow[
'rf_parent'];
358 $newParent = ($row[
'parent_shortname'] ==
null) ?
null :
362 $newReport = ($row[
'report_shortname'] ==
null) ?
null :
365 if (($oldParent != $newParent) && $this->
setMap($newParent, $rfPk, LicenseMap::CONCLUSION)) {
366 $log .=
" with conclusion '$row[parent_shortname]'";
368 if (($oldReport != $newReport) && $this->
setMap($newReport, $rfPk, LicenseMap::REPORT)) {
369 $log .=
" reporting '$row[report_shortname]'";
385 if (empty($row[
'risk'])) {
388 if (empty($row[
'external_id'])) {
389 $row[
'external_id'] =
null;
396 if ($rfPk !==
false) {
397 if (! empty($row[
'group']) || ($md5Match == $rfPk || $md5Match ===
false)) {
400 return "Error: MD5 checksum of '" . $row[
'shortname'] .
401 "' collides with license id=$md5Match";
404 if ($md5Match !==
false && empty($row[
'group'])) {
405 return "Error: MD5 checksum of '" . $row[
'shortname'] .
406 "' collides with license id=$md5Match";
410 if (!empty($row[
'group'])) {
415 return $return[
'log'];
432 if (empty($row[
"external_id"])) {
433 return "Error: external_id cannot be empty";
435 $stmt = __METHOD__ .
".getExistingLicense";
436 $rfPk = $this->
dbManager->getSingleRow(
"SELECT rf_pk FROM " .
437 "license_ref WHERE rf_external_id=$1", array($row[
"external_id"]), $stmt);
445 $stmt = __METHOD__ .
"getNewLicenseObligations";
447 foreach ($row[
'obligation_ids'] as $obExternalId) {
448 $obPk = $this->
dbManager->getSingleRow(
"SELECT ob_pk FROM obligation_ref WHERE ob_external_id = $1;",
449 array($obExternalId), $stmt);
451 $newObIds[] = $obPk[
'ob_pk'];
453 $log .= \sprintf(
'obligation with ob_external_id %s not found', $obExternalId);
458 $stmt = __METHOD__ .
"getOldObligationAssociations";
459 $oldObIdsDB = $this->
dbManager->getRows(
"SELECT ob_fk FROM obligation_map WHERE rf_fk = $1", array($rfPk[
'rf_pk']), $stmt);
461 foreach ($oldObIdsDB as $obId) {
462 $oldObIds[] = $obId[
'ob_fk'];
470 foreach ($diff[
'remove'] as $obid) {
471 $this->obligationMap->unassociateLicenseFromObligation($obid, $rfPk[
'rf_pk']);
475 foreach ($diff[
'add'] as $obid) {
476 $this->obligationMap->associateLicenseWithObligation($obid, $rfPk[
'rf_pk']);
481 $log .= $licRetVal[
'log'];
485 $stmt = __METHOD__ .
"getNewLicenseObligationsForInsert";
486 foreach ($row[
'obligation_ids'] as $obExternalId) {
487 $obPk = $this->
dbManager->getSingleRow(
"SELECT ob_pk FROM obligation_ref WHERE ob_external_id = $1;",
488 array($obExternalId), $stmt);
490 $newObIds[] = $obPk[
'ob_pk'];
492 $log .= \sprintf(
'obligation with ob_external_id %s not found', $obExternalId);
497 foreach ($newObIds as $obid) {
498 $this->obligationMap->associateLicenseWithObligation($obid, $licRetVal[
'pkey']);
519 $isNontrivial = ($fromName!==
null && $fromName!=$toName && $this->
getKeyFromShortname($fromName)!==
false);
521 $this->
dbManager->insertTableRow(
'license_map',
526 return $isNontrivial;
536 $keyName = $shortname;
537 $tableName =
"license_ref";
538 $addCondition =
"AND rf_external_id IS NULL";
539 $statement = __METHOD__ .
".getId";
540 $params = array($shortname);
542 if ($groupFk !=
null) {
543 $keyName .= $groupFk;
544 $tableName =
"license_candidate";
545 $addCondition =
"AND group_fk = $2";
546 $statement .=
".candidate";
547 $params[] = $this->userDao->getGroupIdByName($groupFk);
549 $sql =
"SELECT rf_pk FROM ONLY $tableName WHERE rf_shortname = $1 $addCondition;";
551 if (array_key_exists($keyName, $this->nkMap)) {
552 return $this->nkMap[$keyName];
554 $row = $this->
dbManager->getSingleRow($sql, $params, $statement);
555 $this->nkMap[$keyName] = ($row===
false) ?
false : $row[
'rf_pk'];
556 return $this->nkMap[$keyName];
566 $md5 = md5($licenseText);
567 if (array_key_exists($md5, $this->mdkMap)) {
568 return $this->mdkMap[$md5];
570 $row = $this->
dbManager->getSingleRow(
"SELECT rf_pk " .
571 "FROM ONLY license_ref WHERE rf_md5=md5($1) AND rf_external_id IS NULL",
572 array($licenseText));
573 $this->mdkMap[$md5] = (empty($row)) ?
false : $row[
'rf_pk'];
574 return $this->mdkMap[$md5];
587 private function setMap($from, $to, $usage)
591 $sql =
"SELECT license_map_pk, rf_parent FROM license_map WHERE rf_fk = $1 AND usage = $2;";
592 $statement = __METHOD__ .
".getCurrentMapping";
593 $row = $this->
dbManager->getSingleRow($sql, array($to, $usage), $statement);
594 if (!empty($row) && $row[
'rf_parent'] != $from) {
595 $this->
dbManager->updateTableRow(
"license_map", array(
597 'rf_parent' => $from,
599 ),
'license_map_pk', $row[
'license_map_pk']);
601 } elseif (empty($row)) {
602 $this->
dbManager->insertTableRow(
'license_map', array(
604 'rf_parent' => $from,
624 $stmtInsert = __METHOD__ .
'.insert.' . $tableName;
626 "rf_shortname" => $row[
'shortname'],
627 "rf_licensetype" => $row[
'licensetype'],
628 "rf_fullname" => $row[
'fullname'],
629 "rf_spdx_id" => $row[
'spdx_id'],
630 "rf_text" => $row[
'text'],
631 "rf_md5" => md5($row[
'text']),
632 "rf_detector_type" => 1,
633 "rf_url" => $row[
'url'],
634 "rf_notes" => $row[
'notes'],
635 "rf_source" => $row[
'source'],
636 "rf_risk" => $row[
'risk'],
637 "rf_external_id" => $row[
'external_id']
641 if ($tableName ==
"license_candidate") {
642 $groupId = $this->userDao->getGroupIdByName($row[
'group']);
643 if (empty($groupId)) {
644 return "Error: Unable to insert candidate license " . $row[
'shortname'] .
645 " as group " . $row[
'group'] .
" does not exist";
647 $columns[
"group_fk"] = $groupId;
648 $columns[
"marydone"] = $this->
dbManager->booleanToDb(
true);
649 $as =
" as candidate license under group " . $row[
"group"];
652 $newPk = $this->
dbManager->insertTableRow($tableName, $columns, $stmtInsert,
'rf_pk');
655 $log =
"Inserted '$row[shortname]' in DB" . $as;
656 if ($row[
'external_id'] ===
null) {
657 if ($tableName ==
"license_candidate") {
658 $this->nkMap[$row[
'shortname'].$row[
'group']] = $newPk;
660 $this->nkMap[$row[
'shortname']] = $newPk;
662 $this->mdkMap[md5($row[
'text'])] = $newPk;
664 if ($this->
insertMapIfNontrivial($row[
'parent_shortname'], $row[
'shortname'], LicenseMap::CONCLUSION)) {
665 $log .=
" with conclusion '$row[parent_shortname]'";
668 $log .=
" reporting '$row[report_shortname]'";
673 $return[
'log'] = $log;
674 $return[
'pkey'] = $newPk;
685 foreach ($data as $row) {
Import licenses from CSV.
insertMapIfNontrivial($fromName, $toName, $usage)
Insert in license_map table if the license conclusion is non-trivial.
handleFile($filename, $fileExtension)
Read the CSV line by line and import it.
importJsonData($data, string $msg)
handleCsvLicense($row)
Handle a single row from CSV.
updateLicense($row, $rfPk)
Update the license info in the DB.
handleLicenseDBLicenseImport($row)
Handle a single row from csv import form LicenseDB.
setEnclosure($enclosure='"')
Update the enclosure.
getKeyFromShortname($shortname, $groupFk=null)
Get the license id using license shortname from DB or nkMap.
__construct(DbManager $dbManager, UserDao $userDao, $obligationMap=null)
handleHeadCsv($row)
Handle a row as head row.
insertNewLicense($row, $tableName="license_ref")
Insert a new license in DB.
setDelimiter($delimiter=',')
Update the delimiter.
setMap($from, $to, $usage)
Update license mappings.
getKeyFromMd5($licenseText)
Wrapper class for license map.
Wrapper class for obligation map.
static getArrayDiffs(array $oldArray, array $newArray)
Get list of additions/removals to make $oldArray equal to $newArray.
char * trim(char *ptext)
Trimming whitespace.
fo_dbManager * dbManager
fo_dbManager object
Utility functions for specific applications.