[TASK] Remove unneeded parenthesis on array-access
[Packages/TYPO3.CMS.git] / typo3 / sysext / lowlevel / Classes / DoubleFilesCommand.php
1 <?php
2 namespace TYPO3\CMS\Lowlevel;
3
4 /*
5 * This file is part of the TYPO3 CMS project.
6 *
7 * It is free software; you can redistribute it and/or modify it under
8 * the terms of the GNU General Public License, either version 2
9 * of the License, or any later version.
10 *
11 * For the full copyright and license information, please read the
12 * LICENSE.txt file that was distributed with this source code.
13 *
14 * The TYPO3 project - inspiring people to share!
15 */
16
17 /**
18 * Looking for double files
19 */
20 class DoubleFilesCommand extends CleanerCommand
21 {
22 /**
23 * @var bool
24 */
25 public $checkRefIndex = true;
26
27 /**
28 * Constructor
29 */
30 public function __construct()
31 {
32 parent::__construct();
33 // Setting up help:
34 $this->cli_help['name'] = 'double_files -- Looking for files from TYPO3 managed records which are referenced more than one time (only one time allowed)';
35 $this->cli_help['description'] = trim('
36 Assumptions:
37 - a perfect integrity of the reference index table (always update the reference index table before using this tool!)
38 - files found in deleted records are included (otherwise you would see a false list of lost files)
39
40 Files attached to records in TYPO3 using a "group" type configuration in TCA or FlexForm DataStructure are managed exclusively by the system and there must always exist a 1-1 reference between the file and the reference in the record.
41 This tool will expose when such files are referenced from multiple locations which is considered an integrity error.
42 If a multi-reference is found it was typically created because the record was copied or modified outside of TCEmain which will otherwise maintain the relations correctly.
43 Multi-references should be resolved to 1-1 references as soon as possible. The danger of keeping multi-references is that if the file is removed from one of the refering records it will actually be deleted in the file system, leaving missing files for the remaining referers!
44
45 Automatic Repair of Errors:
46 - The multi-referenced file is copied under a new name and references updated.
47
48 Manual repair suggestions:
49 - None that can not be handled by the automatic repair.
50 ');
51 $this->cli_help['examples'] = '/.../cli_dispatch.phpsh lowlevel_cleaner double_files -s -r
52 This will check the system for double files relations.';
53 }
54
55 /**
56 * Find managed files which are referred to more than one time
57 * Fix methods: API in \TYPO3\CMS\Core\Database\ReferenceIndex that allows to
58 * change the value of a reference (we could copy the file) or remove reference
59 *
60 * @return array
61 */
62 public function main()
63 {
64 // Initialize result array:
65 $resultArray = array(
66 'message' => $this->cli_help['name'] . LF . LF . $this->cli_help['description'],
67 'headers' => array(
68 'multipleReferencesList_count' => array('Number of multi-reference files', '(See below)', 0),
69 'singleReferencesList_count' => array('Number of files correctly referenced', 'The amount of correct 1-1 references', 0),
70 'multipleReferencesList' => array('Entries with files having multiple references', 'These are serious problems that should be resolved ASAP to prevent data loss! ' . $this->label_infoString, 3),
71 'dirname_registry' => array('Registry of directories in which files are found.', 'Registry includes which table/field pairs store files in them plus how many files their store.', 0),
72 'missingFiles' => array('Tracking missing files', '(Extra feature, not related to tracking of double references. Further, the list may include more files than found in the missing_files()-test because this list includes missing files from deleted records.)', 0),
73 'warnings' => array('Warnings picked up', '', 2)
74 ),
75 'multipleReferencesList_count' => array('count' => 0),
76 'singleReferencesList_count' => array('count' => 0),
77 'multipleReferencesList' => array(),
78 'dirname_registry' => array(),
79 'missingFiles' => array(),
80 'warnings' => array()
81 );
82 // Select all files in the reference table not found by a soft reference parser (thus TCA configured)
83 $recs = $GLOBALS['TYPO3_DB']->exec_SELECTgetRows('*', 'sys_refindex', 'ref_table=' . $GLOBALS['TYPO3_DB']->fullQuoteStr('_FILE', 'sys_refindex') . ' AND softref_key=' . $GLOBALS['TYPO3_DB']->fullQuoteStr('', 'sys_refindex'), '', 'sorting DESC');
84 // Traverse the files and put into a large table:
85 $tempCount = array();
86 if (is_array($recs)) {
87 foreach ($recs as $rec) {
88 // Compile info string for location of reference:
89 $infoString = $this->infoStr($rec);
90 // Registering occurencies in directories:
91 $resultArray['dirname_registry'][dirname($rec['ref_string'])][$rec['tablename'] . ':' . $rec['field']]++;
92 // Handle missing file:
93 if (!@is_file((PATH_site . $rec['ref_string']))) {
94 $resultArray['missingFiles'][$rec['ref_string']][$rec['hash']] = $infoString;
95 ksort($resultArray['missingFiles'][$rec['ref_string']]);
96 }
97 // Add entry if file has multiple references pointing to it:
98 if (isset($tempCount[$rec['ref_string']])) {
99 if (!is_array($resultArray['multipleReferencesList'][$rec['ref_string']])) {
100 $resultArray['multipleReferencesList'][$rec['ref_string']] = array();
101 $resultArray['multipleReferencesList'][$rec['ref_string']][$tempCount[$rec['ref_string']][1]] = $tempCount[$rec['ref_string']][0];
102 }
103 $resultArray['multipleReferencesList'][$rec['ref_string']][$rec['hash']] = $infoString;
104 ksort($resultArray['multipleReferencesList'][$rec['ref_string']]);
105 } else {
106 $tempCount[$rec['ref_string']] = array($infoString, $rec['hash']);
107 }
108 }
109 }
110 ksort($resultArray['missingFiles']);
111 ksort($resultArray['multipleReferencesList']);
112 // Add count for multi-references:
113 $resultArray['multipleReferencesList_count']['count'] = count($resultArray['multipleReferencesList']);
114 $resultArray['singleReferencesList_count']['count'] = count($tempCount) - $resultArray['multipleReferencesList_count']['count'];
115 // Sort dirname registry and add warnings for directories outside uploads/
116 ksort($resultArray['dirname_registry']);
117 foreach ($resultArray['dirname_registry'] as $dir => $temp) {
118 ksort($resultArray['dirname_registry'][$dir]);
119 if (!\TYPO3\CMS\Core\Utility\GeneralUtility::isFirstPartOfStr($dir, 'uploads/')) {
120 $resultArray['warnings'][\TYPO3\CMS\Core\Utility\GeneralUtility::shortmd5($dir)] = 'Directory "' . $dir . '" was outside uploads/ which is unusual practice in TYPO3 although not forbidden. Directory used by the following table:field pairs: ' . implode(',', array_keys($temp));
121 }
122 }
123 return $resultArray;
124 }
125
126 /**
127 * Mandatory autofix function
128 * Will run auto-fix on the result array. Echos status during processing.
129 *
130 * @param array $resultArray Result array from main() function
131 * @return void
132 */
133 public function main_autoFix($resultArray)
134 {
135 foreach ($resultArray['multipleReferencesList'] as $key => $value) {
136 $absFileName = \TYPO3\CMS\Core\Utility\GeneralUtility::getFileAbsFileName($key);
137 if ($absFileName && @is_file($absFileName)) {
138 echo 'Processing file: ' . $key . LF;
139 $c = 0;
140 foreach ($value as $hash => $recReference) {
141 if ($c == 0) {
142 echo ' Keeping ' . $key . ' for record "' . $recReference . '"' . LF;
143 } else {
144 // Create unique name for file:
145 $fileFunc = \TYPO3\CMS\Core\Utility\GeneralUtility::makeInstance(\TYPO3\CMS\Core\Utility\File\BasicFileUtility::class);
146 $newName = $fileFunc->getUniqueName(basename($key), dirname($absFileName));
147 echo ' Copying ' . $key . ' to ' . \TYPO3\CMS\Core\Utility\PathUtility::stripPathSitePrefix($newName) . ' for record "' . $recReference . '": ';
148 if ($bypass = $this->cli_noExecutionCheck($recReference)) {
149 echo $bypass;
150 } else {
151 \TYPO3\CMS\Core\Utility\GeneralUtility::upload_copy_move($absFileName, $newName);
152 clearstatcache();
153 if (@is_file($newName)) {
154 $sysRefObj = \TYPO3\CMS\Core\Utility\GeneralUtility::makeInstance(\TYPO3\CMS\Core\Database\ReferenceIndex::class);
155 $error = $sysRefObj->setReferenceValue($hash, basename($newName));
156 if ($error) {
157 echo ' ERROR: TYPO3\\CMS\\Core\\Database\\ReferenceIndex::setReferenceValue(): ' . $error . LF;
158 die;
159 } else {
160 echo 'DONE';
161 }
162 } else {
163 echo ' ERROR: File "' . $newName . '" was not created!';
164 }
165 }
166 echo LF;
167 }
168 $c++;
169 }
170 } else {
171 echo ' ERROR: File "' . $absFileName . '" was not found!';
172 }
173 }
174 }
175 }