29require
'../../main.inc.php';
30require_once DOL_DOCUMENT_ROOT.
'/core/lib/files.lib.php';
31require_once DOL_DOCUMENT_ROOT.
'/core/lib/geturl.lib.php';
48llxHeader(
'',
'',
'',
'', 0, 0,
'',
'',
'',
'mod-admin page-system_filecheck');
50print
load_fiche_titre($langs->trans(
"FileCheckDolibarr"),
'',
'title_setup');
52print
'<span class="opacitymedium">'.$langs->trans(
"FileCheckDesc").
'</span><br><br>';
55print
'<div class="div-table-responsive-no-min">';
56print
'<table class="noborder centpercent">';
57print
'<tr class="liste_titre"><td>'.$langs->trans(
"Version").
'</td><td></td></tr>'.
"\n";
58print
'<tr class="oddeven"><td width="300">'.$langs->trans(
"VersionLastInstall").
'</td><td>'.
getDolGlobalString(
'MAIN_VERSION_LAST_INSTALL').
'</td></tr>'.
"\n";
59print
'<tr class="oddeven"><td width="300">'.$langs->trans(
"VersionLastUpgrade").
'</td><td>'.
getDolGlobalString(
'MAIN_VERSION_LAST_UPGRADE').
'</td></tr>'.
"\n";
60print
'<tr class="oddeven"><td width="300">'.$langs->trans(
"VersionProgram").
'</td><td>'.DOL_VERSION;
65 print
' '.img_warning($langs->trans(
"RunningUpdateProcessMayBeRequired", DOL_VERSION,
getDolGlobalString(
'MAIN_VERSION_LAST_INSTALL')));
69 if (DOL_VERSION != $conf->global->MAIN_VERSION_LAST_UPGRADE) {
70 print
' '.img_warning($langs->trans(
"RunningUpdateProcessMayBeRequired", DOL_VERSION,
getDolGlobalString(
'MAIN_VERSION_LAST_UPGRADE')));
73print
'</td></tr>'.
"\n";
80$file_list = array(
'missing' => array(),
'updated' => array());
85$xmlfile = DOL_DOCUMENT_ROOT.
'/install/'.$xmlshortfile;
86if (!preg_match(
'/\.zip$/i', $xmlfile) &&
dol_is_file($xmlfile.
'.zip')) {
91$xmlremote =
GETPOST(
'xmlremote',
'alphanohtml');
95$param =
'MAIN_FILECHECK_URL_'.DOL_VERSION;
99if (empty($xmlremote)) {
100 $xmlremote =
'https://www.dolibarr.org/files/stable/signatures/filelist-'.DOL_VERSION.
'.xml';
102if ($xmlremote && !preg_match(
'/^https?:\/\//', $xmlremote)) {
103 $langs->load(
"errors");
104 setEventMessages($langs->trans(
"ErrorURLMustStartWithHttp", $xmlremote),
null,
'errors');
106} elseif ($xmlremote && !preg_match(
'/\.xml$/', $xmlremote)) {
107 $langs->load(
"errors");
108 setEventMessages($langs->trans(
"ErrorURLMustEndWith", $xmlremote,
'.xml'),
null,
'errors');
113$enableremotecheck =
true;
114if (preg_match(
'/beta|alpha|rc/i', DOL_VERSION) ||
getDolGlobalString(
'MAIN_ALLOW_INTEGRITY_CHECK_ON_UNSTABLE')) {
115 $enableremotecheck =
false;
117$enableremotecheck =
true;
119print
'<form name="check" action="'.$_SERVER[
"PHP_SELF"].
'">';
120print
'<input type="hidden" name="token" value="'.newToken().
'">';
121print $langs->trans(
"MakeIntegrityAnalysisFrom").
':<br>';
123print
'<div class="divsection">';
124print
'<!-- for a local check target=local&xmlshortfile=... -->'.
"\n";
126 print
'<input type="radio" name="target" id="checkboxlocal" value="local"'.((!
GETPOST(
'target') ||
GETPOST(
'target') ==
'local') ?
'checked="checked"' :
'').
'"> <label for="checkboxlocal">'.$langs->trans(
"LocalSignature").
'</label> = ';
127 print
'<input name="xmlshortfile" class="flat minwidth400" value="'.dol_escape_htmltag($xmlshortfile).
'">';
130 print
'<input type="radio" name="target" id="checkboxlocal" value="local"> <label for="checkboxlocal">'.$langs->trans(
"LocalSignature").
' = ';
131 print
'<input name="xmlshortfile" class="flat minwidth400" value="'.dol_escape_htmltag($xmlshortfile).
'">';
132 print
' <span class="warning">('.$langs->trans(
"AvailableOnlyOnPackagedVersions").
')</span></label>';
135print
'<!-- for a remote target=remote&xmlremote=... -->'.
"\n";
136if ($enableremotecheck) {
137 print
'<input type="radio" name="target" id="checkboxremote" value="remote"'.(GETPOST(
'target') ==
'remote' ?
'checked="checked"' :
'').
'> <label for="checkboxremote">'.$langs->trans(
"RemoteSignature").
'</label> = ';
138 print
'<input name="xmlremote" class="flat minwidth500" value="'.dol_escape_htmltag($xmlremote).
'"><br>';
140 print
'<input type="radio" name="target" id="checkboxremote" value="remote" disabled="disabled"> '.$langs->trans(
"RemoteSignature").
' = '.
dol_escape_htmltag($xmlremote);
142 print
' <span class="warning">('.$langs->trans(
"FeatureAvailableOnlyOnStable").
')</span>';
148print
'<div class="center"><input type="submit" name="check" class="button" value="'.$langs->trans(
"Check").
'"></div>';
153if (
GETPOST(
'target') ==
'local') {
156 if (preg_match(
'/\.zip$/i', $xmlfile)) {
158 $xmlfilenew = preg_replace(
'/\.zip$/i',
'', $xmlfile);
160 if (empty($result[
'error'])) {
161 $xmlfile = $conf->admin->dir_temp.
'/'.basename($xmlfilenew);
163 print $langs->trans(
'FailedToUncompressFile').
': '.$xmlfile;
167 $xml = simplexml_load_file($xmlfile);
168 if ($xml ===
false) {
169 print
'<div class="warning">'.$langs->trans(
'XmlCorrupted').
': '.$xmlfile.
'</span>';
173 print
'<div class="warning">'.$langs->trans(
'XmlNotFound').
': '.$xmlfile.
'</span>';
177if (
GETPOST(
'target') ==
'remote') {
178 $xmlarray =
getURLContent($xmlremote,
'GET',
'', 1, array(), array(
'http',
'https'), 0);
181 if (!$xmlarray[
'curl_error_no'] && $xmlarray[
'http_code'] !=
'400' && $xmlarray[
'http_code'] !=
'404') {
182 $xmlfile = $xmlarray[
'content'];
184 if (LIBXML_VERSION < 20900) {
188 libxml_disable_entity_loader(
true);
191 $xml = simplexml_load_string($xmlfile,
'SimpleXMLElement', LIBXML_NOCDATA | LIBXML_NONET);
193 $errormsg = $langs->trans(
'XmlNotFound').
': '.$xmlremote.
' - '.$xmlarray[
'http_code'].(($xmlarray[
'http_code'] == 400 && $xmlarray[
'content']) ?
' '.$xmlarray[
'content'] :
'').
' '.$xmlarray[
'curl_error_no'].
' '.$xmlarray[
'curl_error_msg'];
200if (empty($error) && !empty($xml)) {
201 $checksumconcat = array();
202 $file_list = array();
206 if (is_object($xml->dolibarr_constants[0])) {
209 $out .=
'<div class="div-table-responsive-no-min">';
210 $out .=
'<table class="noborder">';
211 $out .=
'<tr class="liste_titre">';
212 $out .=
'<td>#</td>';
213 $out .=
'<td>'.$langs->trans(
"Constant").
'</td>';
214 $out .=
'<td class="center">'.$langs->trans(
"ExpectedValue").
'</td>';
215 $out .=
'<td class="center">'.$langs->trans(
"Value").
'</td>';
216 $out .=
'</tr>'.
"\n";
219 foreach ($xml->dolibarr_constants[0]->constant as $constant) {
220 $constname = $constant[
'name'];
221 $constvalue = (string) $constant;
222 $constvalue = (empty($constvalue) ?
'0' : $constvalue);
228 $valueforchecksum = (empty($value) ?
'0' : $value);
230 $checksumconcat[] = $valueforchecksum;
233 $out .=
'<tr class="oddeven">';
234 $out .=
'<td>'.$i.
'</td>'.
"\n";
235 $out .=
'<td>'.dol_escape_htmltag($constname).
'</td>'.
"\n";
236 $out .=
'<td class="center">'.dol_escape_htmltag($constvalue).
'</td>'.
"\n";
237 $out .=
'<td class="center">'.dol_escape_htmltag($valueforchecksum).
'</td>'.
"\n";
242 $out .=
'<tr class="oddeven"><td colspan="4"><span class="opacitymedium">'.$langs->trans(
"None").
'</span></td></tr>';
251 if (is_object($xml->dolibarr_htdocs_dir[0])) {
253 $includecustom = (empty($xml->dolibarr_htdocs_dir[0][
'includecustom']) ? 0 : $xml->dolibarr_htdocs_dir[0][
'includecustom']);
256 $regextoinclude =
'\.(php|php3|php4|php5|phtml|phps|phar|inc|css|scss|html|xml|js|json|tpl|jpg|jpeg|png|gif|ico|sql|lang|txt|yml|bak|md|mp3|mp4|wav|mkv|z|gz|zip|rar|tar|less|svg|eot|woff|woff2|ttf|manifest)$';
257 $regextoexclude =
'('.($includecustom ?
'' :
'custom|').
'documents|conf|install|dejavu-fonts-ttf-.*|public\/test|sabre\/sabre\/.*\/tests|Shared\/PCLZip|nusoap\/lib\/Mail|php\/example|php\/test|geoip\/sample.*\.php|ckeditor\/samples|ckeditor\/adapters)$';
258 $scanfiles =
dol_dir_list(DOL_DOCUMENT_ROOT,
'files', 1, $regextoinclude, $regextoexclude);
261 $ret =
getFilesUpdated($file_list, $xml->dolibarr_htdocs_dir[0],
'', DOL_DOCUMENT_ROOT, $checksumconcat);
263 foreach ($scanfiles as $keyfile => $valfile) {
264 $tmprelativefilename = preg_replace(
'/^'.preg_quote(DOL_DOCUMENT_ROOT,
'/').
'/',
'', $valfile[
'fullname']);
265 if (!in_array($tmprelativefilename, $file_list[
'insignature'])) {
266 $md5newfile = @md5_file($valfile[
'fullname']);
267 $file_list[
'added'][] = array(
'filename' => $tmprelativefilename,
'md5' => $md5newfile);
274 $out .=
'<div class="div-table-responsive-no-min">';
275 $out .=
'<table class="noborder">';
276 $out .=
'<tr class="liste_titre">';
277 $out .=
'<td>#</td>';
278 $out .=
'<td>'.$langs->trans(
"Filename").
'</td>';
279 $out .=
'<td class="right">'.$langs->trans(
"ExpectedSize").
'</td>';
280 $out .=
'<td class="center">'.$langs->trans(
"ExpectedChecksum").
'</td>';
281 $out .=
'</tr>'.
"\n";
283 if (is_array($tmpfilelist) && count($tmpfilelist)) {
285 foreach ($tmpfilelist as $file) {
287 $out .=
'<tr class="oddeven">';
288 $out .=
'<td>'.$i.
'</td>'.
"\n";
289 $out .=
'<td>'.dol_escape_htmltag($file[
'filename']).
'</td>'.
"\n";
290 $out .=
'<td class="right">';
291 if (!empty($file[
'expectedsize'])) {
294 $out .=
'</td>'.
"\n";
295 $out .=
'<td class="center">'.dol_escape_htmltag($file[
'expectedmd5']).
'</td>'.
"\n";
299 $out .=
'<tr class="oddeven"><td colspan="4"><span class="opacitymedium">'.$langs->trans(
"None").
'</span></td></tr>';
310 $out .=
'<div class="div-table-responsive-no-min">';
311 $out .=
'<table class="noborder">';
312 $out .=
'<tr class="liste_titre">';
313 $out .=
'<td>#</td>';
314 $out .=
'<td>'.$langs->trans(
"Filename").
'</td>';
315 $out .=
'<td class="center">'.$langs->trans(
"ExpectedChecksum").
'</td>';
316 $out .=
'<td class="center">'.$langs->trans(
"CurrentChecksum").
'</td>';
317 $out .=
'<td class="right">'.$langs->trans(
"ExpectedSize").
'</td>';
318 $out .=
'<td class="right">'.$langs->trans(
"CurrentSize").
'</td>';
319 $out .=
'<td class="right">'.$langs->trans(
"DateModification").
'</td>';
320 $out .=
'</tr>'.
"\n";
321 $tmpfilelist2 =
dol_sort_array($file_list[
'updated'],
'filename');
322 if (is_array($tmpfilelist2) && count($tmpfilelist2)) {
324 foreach ($tmpfilelist2 as $file) {
326 $out .=
'<tr class="oddeven">';
327 $out .=
'<td>'.$i.
'</td>'.
"\n";
328 $out .=
'<td>'.dol_escape_htmltag($file[
'filename']).
'</td>'.
"\n";
329 $out .=
'<td class="center">'.dol_escape_htmltag($file[
'expectedmd5']).
'</td>'.
"\n";
330 $out .=
'<td class="center">'.dol_escape_htmltag($file[
'md5']).
'</td>'.
"\n";
331 $out .=
'<td class="right">';
332 if ($file[
'expectedsize']) {
335 $out .=
'</td>'.
"\n";
336 $size =
dol_filesize(DOL_DOCUMENT_ROOT.
'/'.$file[
'filename']);
338 $out .=
'<td class="right">'.dol_print_size($size).
'</td>'.
"\n";
339 $out .=
'<td class="right">'.dol_print_date(
dol_filemtime(DOL_DOCUMENT_ROOT.
'/'.$file[
'filename']),
'dayhour').
'</td>'.
"\n";
342 $out .=
'<tr class="liste_total">';
343 $out .=
'<td></td>'.
"\n";
344 $out .=
'<td>'.$langs->trans(
"Total").
'</td>'.
"\n";
345 $out .=
'<td class="center"></td>'.
"\n";
346 $out .=
'<td class="center"></td>'.
"\n";
347 $out .=
'<td class="center"></td>'.
"\n";
348 $out .=
'<td class="right">'.dol_print_size($totalsize).
'</td>'.
"\n";
349 $out .=
'<td class="right"></td>'.
"\n";
352 $out .=
'<tr class="oddeven"><td colspan="7"><span class="opacitymedium">'.$langs->trans(
"None").
'</span></td></tr>';
363 $out .=
'<div class="div-table-responsive-no-min">';
364 $out .=
'<table class="noborder">';
365 $out .=
'<tr class="liste_titre">';
366 $out .=
'<td>#</td>';
367 $out .=
'<td>'.$langs->trans(
"Filename").
'</td>';
368 $out .=
'<td class="center">'.$langs->trans(
"ExpectedChecksum").
'</td>';
369 $out .=
'<td class="center">'.$langs->trans(
"CurrentChecksum").
'</td>';
370 $out .=
'<td class="right">'.$langs->trans(
"Size").
'</td>';
371 $out .=
'<td class="right">'.$langs->trans(
"DateModification").
'</td>';
372 $out .=
'</tr>'.
"\n";
374 if (is_array($tmpfilelist3) && count($tmpfilelist3)) {
376 foreach ($tmpfilelist3 as $file) {
378 $out .=
'<tr class="oddeven">';
379 $out .=
'<td>'.$i.
'</td>'.
"\n";
380 $out .=
'<td>'.dol_escape_htmltag($file[
'filename']);
381 if (!preg_match(
'/^win/i', PHP_OS)) {
382 $htmltext = $langs->trans(
"YouCanDeleteFileOnServerWith",
'rm '.DOL_DOCUMENT_ROOT.$file[
'filename']);
383 $out .=
' '.$form->textwithpicto(
'', $htmltext, 1,
'help',
'', 0, 2,
'helprm'.$i);
385 $out .=
'</td>'.
"\n";
386 $out .=
'<td class="center">'.dol_escape_htmltag($file[
'expectedmd5']).
'</td>'.
"\n";
387 $out .=
'<td class="center">'.dol_escape_htmltag($file[
'md5']).
'</td>'.
"\n";
388 $size =
dol_filesize(DOL_DOCUMENT_ROOT.
'/'.$file[
'filename']);
390 $out .=
'<td class="right">'.dol_print_size($size).
'</td>'.
"\n";
391 $out .=
'<td class="right nowraponall">'.dol_print_date(
dol_filemtime(DOL_DOCUMENT_ROOT.
'/'.$file[
'filename']),
'dayhour').
'</td>'.
"\n";
394 $out .=
'<tr class="liste_total">';
395 $out .=
'<td></td>'.
"\n";
396 $out .=
'<td>'.$langs->trans(
"Total").
'</td>'.
"\n";
397 $out .=
'<td class="center"></td>'.
"\n";
398 $out .=
'<td class="center"></td>'.
"\n";
399 $out .=
'<td class="right">'.dol_print_size($totalsize).
'</td>'.
"\n";
400 $out .=
'<td class="right"></td>'.
"\n";
403 $out .=
'<tr class="oddeven"><td colspan="6"><span class="opacitymedium">'.$langs->trans(
"None").
'</span></td></tr>';
408 print
'<div class="error">';
409 print
'Error: Failed to found <b>dolibarr_htdocs_dir</b> into content of XML file:<br>'.dol_escape_htmltag(
dol_trunc($xmlfile, 500));
424 asort($checksumconcat);
426 $checksumget = md5(implode(
',', $checksumconcat));
427 $checksumtoget = trim((
string) $xml->dolibarr_htdocs_dir_checksum);
436 $outexpectedchecksum = ($checksumtoget ? $checksumtoget : $langs->trans(
"Unknown"));
437 if ($checksumget == $checksumtoget) {
438 if (is_array($file_list[
'added']) && count($file_list[
'added'])) {
439 $resultcode =
'warning';
440 $resultcomment =
'FileIntegrityIsOkButFilesWereAdded';
441 $outcurrentchecksum = $checksumget.
' - <span class="'.$resultcode.
'">'.$langs->trans($resultcomment).
'</span>';
444 $resultcomment =
'Success';
445 $outcurrentchecksum =
'<span class="'.$resultcode.
'">'.$checksumget.
'</span>';
448 $resultcode =
'error';
449 $resultcomment =
'Error';
450 $outcurrentchecksum =
'<span class="'.$resultcode.
'">'.$checksumget.
'</span>';
454 if (empty($tmpfilelist) && empty($tmpfilelist2) && empty($tmpfilelist3) && $resultcode ==
'ok') {
455 setEventMessages($langs->trans(
"FileIntegrityIsStrictlyConformedWithReference"),
null,
'mesgs');
457 if ($resultcode ==
'warning') {
460 setEventMessages($langs->trans(
"FileIntegritySomeFilesWereRemovedOrModified"),
null,
'errors');
465 print $langs->trans(
"ExpectedChecksum").
' = '.$outexpectedchecksum.
'<br>';
466 print $langs->trans(
"CurrentChecksum").
' = '.$outcurrentchecksum;
if(!defined('NOREQUIRESOC')) if(!defined( 'NOREQUIRETRAN')) if(!defined('NOTOKENRENEWAL')) if(!defined( 'NOREQUIREMENU')) if(!defined('NOREQUIREHTML')) if(!defined( 'NOREQUIREAJAX')) llxHeader($head='', $title='', $help_url='', $target='', $disablejs=0, $disablehead=0, $arrayofjs='', $arrayofcss='', $morequerystring='', $morecssonbody='', $replacemainareaby='', $disablenofollow=0, $disablenoindex=0)
Empty header.
getFilesUpdated(&$file_list, SimpleXMLElement $dir, $path='', $pathref='', &$checksumconcat=array())
Function to get list of updated or modified files.
dol_filemtime($pathoffile)
Return time of a file.
dol_filesize($pathoffile)
Return size of a file.
dol_uncompress($inputfile, $outputdir)
Uncompress a file.
dol_is_file($pathoffile)
Return if path is a file.
dol_dir_list($utf8_path, $types="all", $recursive=0, $filter="", $excludefilter=null, $sortcriteria="name", $sortorder=SORT_ASC, $mode=0, $nohook=0, $relativename="", $donotfollowsymlinks=0, $nbsecondsold=0)
Scan a directory and return a list of files/directories.
dol_print_size($size, $shortvalue=0, $shortunit=0)
Return string with formatted size.
load_fiche_titre($title, $morehtmlright='', $picto='generic', $pictoisfullpath=0, $id='', $morecssontable='', $morehtmlcenter='')
Load a title with picto.
setEventMessages($mesg, $mesgs, $style='mesgs', $messagekey='', $noduplicate=0, $attop=0)
Set event messages in dol_events session object.
dol_sort_array(&$array, $index, $order='asc', $natsort=0, $case_sensitive=0, $keepindex=0)
Advanced sort array by the value of a given key, which produces ascending (default) or descending out...
GETPOST($paramname, $check='alphanohtml', $method=0, $filter=null, $options=null, $noreplace=0)
Return value of a param into GET or POST supervariable.
dol_sanitizeFileName($str, $newstr='_', $unaccent=1)
Clean a string to use it as a file name.
dol_trunc($string, $size=40, $trunc='right', $stringencoding='UTF-8', $nodot=0, $display=0)
Truncate a string to a particular length adding '…' if string larger than length.
getDolGlobalString($key, $default='')
Return a Dolibarr global constant string value.
dol_mkdir($dir, $dataroot='', $newmask='')
Creation of a directory (this can create recursive subdir)
dol_escape_htmltag($stringtoescape, $keepb=0, $keepn=0, $noescapetags='', $escapeonlyhtmltags=0, $cleanalsojavascript=0)
Returns text escaped for inclusion in HTML alt or title or value tags, or into values of HTML input f...
getURLContent($url, $postorget='GET', $param='', $followlocation=1, $addheaders=array(), $allowedschemes=array('http', 'https'), $localurl=0, $ssl_verifypeer=-1)
Function to get a content from an URL (use proxy if proxy defined).
accessforbidden($message='', $printheader=1, $printfooter=1, $showonlymessage=0, $params=null)
Show a message to say access is forbidden and stop program.