FOSSology  4.4.0
Open Source License Compliance by Open Source Software
AjaxSHDetailsBrowser.php
1 <?php
2 /*
3  SPDX-FileCopyrightText: © 2019 Sandip Kumar Bhuyan <sandipbhuyan@gmail.com>
4  Author: Sandip Kumar Bhuyan<sandipbhyan@gmail.com>
5 
6  SPDX-License-Identifier: GPL-2.0-only
7 */
18 use Symfony\Component\HttpFoundation\JsonResponse;
19 use \Symfony\Component\HttpFoundation\Request;
20 use Symfony\Component\HttpFoundation\Response;
21 
23 {
24  const NAME = "ajax_sh_browser";
25 
26  private $uploadtree_tablename = "";
28  private $uploadDao;
30  private $licenseDao;
32  private $agentDao;
37  private $shDao;
42  private $configuration;
43 
44  protected $agentNames = array('softwareHeritage' => 'SH');
45 
46  public function __construct()
47  {
48  parent::__construct(self::NAME, array(
49  self::TITLE => _("Ajax: File Browser"),
50  self::DEPENDENCIES => array("fileBrowse"),
51  self::PERMISSION => Auth::PERM_READ,
52  self::REQUIRES_LOGIN => false
53  ));
54 
55  $this->uploadDao = $this->getObject('dao.upload');
56  $this->licenseDao = $this->getObject('dao.license');
57  $this->agentDao = $this->getObject('dao.agent');
58  $this->shDao = $this->container->get('dao.softwareHeritage');
59  $sysconfig = $GLOBALS['SysConf']['SYSCONFIG'];
60  $this->configuration = [
61  'url' => trim($sysconfig['SwhURL']),
62  'uri' => trim($sysconfig['SwhBaseURL']),
63  'content' => trim($sysconfig['SwhContent']),
64  'maxtime' => intval($sysconfig['SwhSleep']),
65  'token' => trim($sysconfig['SwhToken'])
66  ];
67  }
68 
73  public function handle(Request $request)
74  {
75  $upload = intval($request->get("upload"));
76  $groupId = Auth::getGroupId();
77  if (!$this->uploadDao->isAccessible($upload, $groupId)) {
78  throw new \Exception("Permission Denied");
79  }
80 
81  $item = intval($request->get("item"));
82  $this->uploadtree_tablename = $this->uploadDao->getUploadtreeTableName($upload);
83  $itemTreeBounds = $this->uploadDao->getItemTreeBounds($item, $this->uploadtree_tablename);
84  $left = $itemTreeBounds->getLeft();
85  if (empty($left)) {
86  throw new \Exception("Job unpack/adj2nest hasn't completed.");
87  }
88 
89  $scannerAgents = array_keys($this->agentNames);
90  $scanJobProxy = new ScanJobProxy($this->agentDao, $upload);
91  $scanJobProxy->createAgentStatus($scannerAgents);
92  $selectedAgentId = intval($request->get('agentId'));
93  $tag_pk = intval($request->get('tag'));
94 
95  $UniqueTagArray = array();
96  $this->licenseProjector = new LicenseMap($this->getObject('db.manager'),$groupId,LicenseMap::CONCLUSION,true);
97  $vars = $this->createFileListing($tag_pk, $itemTreeBounds, $UniqueTagArray, $selectedAgentId, $groupId, $scanJobProxy, $request);
98 
99  return new JsonResponse(array(
100  'sEcho' => intval($request->get('sEcho')),
101  'aaData' => $vars['fileData'],
102  'iTotalRecords' => $vars['iTotalDisplayRecords'],
103  'iTotalDisplayRecords' => $vars['iTotalDisplayRecords']
104  ));
105  }
106 
116  private function createFileListing($tagId, ItemTreeBounds $itemTreeBounds, &$UniqueTagArray, $selectedAgentId, $groupId, $scanJobProxy, $request)
117  {
118  if (!empty($selectedAgentId)) {
119  $agentName = $this->agentDao->getAgentName($selectedAgentId);
120  $selectedScanners = array($agentName=>$selectedAgentId);
121  } else {
122  $selectedScanners = $scanJobProxy->getLatestSuccessfulAgentIds();
123  }
124 
126  $uploadId = $itemTreeBounds->getUploadId();
127  $isFlat = isset($_GET['flatten']);
128 
129  if ($isFlat) {
130  $options = array(UploadTreeProxy::OPT_RANGE => $itemTreeBounds);
131  } else {
132  $options = array(UploadTreeProxy::OPT_REALPARENT => $itemTreeBounds->getItemId());
133  }
134 
135  $searchMap = array();
136  foreach (explode(' ',$request->get('sSearch')) as $pair) {
137  $a = explode(':',$pair);
138  if (count($a) == 1) {
139  $searchMap['head'] = $pair;
140  } else {
141  $searchMap[$a[0]] = $a[1];
142  }
143  }
144 
145  if (array_key_exists('ext', $searchMap) && strlen($searchMap['ext'])>=1) {
146  $options[UploadTreeProxy::OPT_EXT] = $searchMap['ext'];
147  }
148  if (array_key_exists('head', $searchMap) && strlen($searchMap['head'])>=1) {
149  $options[UploadTreeProxy::OPT_HEAD] = $searchMap['head'];
150  }
151 
152  $descendantView = new UploadTreeProxy($uploadId, $options, $itemTreeBounds->getUploadTreeTableName(), 'uberItems');
153 
154  $vars['iTotalDisplayRecords'] = $descendantView->count();
155 
156  $columnNamesInDatabase = array($isFlat?'ufile_name':'lft');
157  $defaultOrder = array(array(0, "asc"));
158  $orderString = $this->getObject('utils.data_tables_utility')->getSortingString($_GET, $columnNamesInDatabase, $defaultOrder);
159 
160  $offset = GetParm('iDisplayStart', PARM_INTEGER);
161  $limit = GetParm('iDisplayLength', PARM_INTEGER);
162  if ($offset) {
163  $orderString .= " OFFSET $offset";
164  }
165  if ($limit) {
166  $orderString .= " LIMIT $limit";
167  }
168 
169  /* Get ALL the items under this Uploadtree_pk */
170  $sql = $descendantView->getDbViewQuery()." $orderString";
171  $dbManager = $this->getObject('db.manager');
172 
173  $dbManager->prepare($stmt=__METHOD__.$orderString,$sql);
174  $res = $dbManager->execute($stmt,$descendantView->getParams());
175  $descendants = $dbManager->fetchAll($res);
176  $dbManager->freeResult($res);
177 
178  /* Filter out Children that don't have tag */
179  if (!empty($tagId)) {
180  TagFilter($descendants, $tagId, $itemTreeBounds->getUploadTreeTableName());
181  }
182  if (empty($descendants)) {
183  $vars['fileData'] = array();
184  return $vars;
185  }
186 
187  if ($isFlat) {
188  $firstChild = reset($descendants);
189  $lastChild = end($descendants);
190  $nameRange = array($firstChild['ufile_name'],$lastChild['ufile_name']);
191  } else {
192  $nameRange = array();
193  }
194 
195  /******* File Listing ************/
196  $pfileLicenses = array();
197  foreach ($selectedScanners as $agentName=>$agentId) {
198  $licensePerPfile = $this->licenseDao->getLicenseIdPerPfileForAgentId($itemTreeBounds, $agentId, $isFlat, $nameRange);
199  foreach ($licensePerPfile as $pfile => $licenseRow) {
200  foreach ($licenseRow as $licId => $row) {
201  $lic = $this->licenseProjector->getProjectedShortname($licId);
202  $pfileLicenses[$pfile][$lic][$agentName] = $row;
203  }
204  }
205  }
206 
207  $baseUri = Traceback_uri().'?mod=sh-agent'.Traceback_parm_keep(array('upload','folder','show'));
208 
209  $tableData = array();
210  $latestSuccessfulAgentIds = $scanJobProxy->getLatestSuccessfulAgentIds();
211  foreach ($descendants as $child) {
212  if (empty($child)) {
213  continue;
214  }
215  $tableData[] = $this->createFileDataRow($child, $uploadId, $selectedAgentId,
216  $baseUri, $UniqueTagArray, $isFlat);
217  }
218 
219  $vars['fileData'] = $tableData;
220  return $vars;
221  }
222 
232  private function createFileDataRow($child, $uploadId, $selectedAgentId, $uri, &$UniqueTagArray, $isFlat)
233  {
234  $fileId = $child['pfile_fk'];
235  $childUploadTreeId = $child['uploadtree_pk'];
236  $linkUri = '';
237  if (!empty($fileId)) {
238  $linkUri = Traceback_uri();
239  $linkUri .= "?mod=view-license&upload=$uploadId&item=$childUploadTreeId";
240  if ($selectedAgentId) {
241  $linkUri .= "&agentId=$selectedAgentId";
242  }
243  }
244 
245  /* Determine link for containers */
246  $isContainer = Iscontainer($child['ufile_mode']);
247  if ($isContainer && !$isFlat) {
248  $uploadtree_pk = $child['uploadtree_pk'];
249  $linkUri = "$uri&item=" . $uploadtree_pk;
250  if ($selectedAgentId) {
251  $linkUri .= "&agentId=$selectedAgentId";
252  }
253  } else if ($isContainer) {
254  $uploadtree_pk = Isartifact($child['ufile_mode']) ? DirGetNonArtifact($childUploadTreeId, $this->uploadtree_tablename) : $childUploadTreeId;
255  $linkUri = "$uri&item=" . $uploadtree_pk;
256  if ($selectedAgentId) {
257  $linkUri .= "&agentId=$selectedAgentId";
258  }
259  }
260 
261  /* Populate the output ($VF) - file list */
262  /* id of each element is its uploadtree_pk */
263  $fileName = htmlspecialchars($child['ufile_name']);
264  if ($isContainer) {
265  $fileName = "<a href='$linkUri'><span style='color: darkblue'> <b>$fileName</b> </span></a>";
266  } else if (! empty($linkUri)) {
267  $fileName = "<a href='$linkUri'>$fileName</a>";
268  }
269 
270  $pfileHash = $this->uploadDao->getUploadHashesFromPfileId($fileId);
271  $shRecord = $this->shDao->getSoftwareHetiageRecord($fileId);
272  $fileListLinks = FileListLinks($uploadId, $childUploadTreeId, 0, $fileId, true, $UniqueTagArray, $this->uploadtree_tablename, !$isFlat);
273 
274  if (! $isContainer) {
275  $text = _("Software Heritage");
276  $shLink = $this->configuration['url'] .
277  $this->configuration['uri'] . strtolower($pfileHash["sha256"]) .
278  $this->configuration['content'];
279  $fileListLinks .= "[<a href='".$shLink."' target=\"_blank\">$text</a>]";
280  }
281  $img = "";
282  if (! $isContainer) {
283  $img = $shRecord["img"];
284  }
285 
286  return [$fileName, $pfileHash["sha256"], $shRecord["license"], $img, $fileListLinks];
287  }
288 }
289 
290 register_plugin(new AjaxSHDetailsBrowser());
char * uploadtree_tablename
upload.uploadtree_tablename
Definition: adj2nest.c:100
createFileDataRow($child, $uploadId, $selectedAgentId, $uri, &$UniqueTagArray, $isFlat)
handle(Request $request)
createFileListing($tagId, ItemTreeBounds $itemTreeBounds, &$UniqueTagArray, $selectedAgentId, $groupId, $scanJobProxy, $request)
Contains the constants and helpers for authentication of user.
Definition: Auth.php:24
Wrapper class for license map.
Definition: LicenseMap.php:19
Isartifact($mode)
Definition: common-dir.php:29
DirGetNonArtifact($UploadtreePk, $uploadtree_tablename='uploadtree')
Given an artifact directory (uploadtree_pk), return the first non-artifact directory (uploadtree_pk).
Definition: common-dir.php:158
Iscontainer($mode)
Definition: common-dir.php:38
FileListLinks($upload_fk, $uploadtree_pk, $napk, $pfile_pk, $Recurse=True, &$UniqueTagArray=array(), $uploadtree_tablename="uploadtree", $wantTags=true)
Get list of links: [View][Info][Download]
Traceback_uri()
Get the URI without query to this location.
Definition: common-parm.php:97
const PARM_INTEGER
Definition: common-parm.php:14
GetParm($parameterName, $parameterType)
This function will retrieve the variables and check data types.
Definition: common-parm.php:46
Traceback_parm_keep($List)
Create a new URI, keeping only these items.
TagFilter(&$UploadtreeRows, $tag_pk, $uploadtree_tablename)
Given a list of uploadtree recs, remove recs that do not have $tag_pk.
char * trim(char *ptext)
Trimming whitespace.
Definition: fossconfig.c:690
#define PERM_READ
Read-only permission.
Definition: libfossology.h:32