dolibarr 19.0.3
doc_generic_asset_odt.modules.php
Go to the documentation of this file.
1<?php
2/* Copyright (C) 2010-2012 Laurent Destailleur <eldy@users.sourceforge.net>
3 * Copyright (C) 2012 Juanjo Menent <jmenent@2byte.es>
4 * Copyright (C) 2014 Marcos García <marcosgdf@gmail.com>
5 * Copyright (C) 2016 Charlie Benke <charlie@patas-monkey.com>
6 * Copyright (C) 2018-2021 Philippe Grand <philippe.grand@atoo-net.com>
7 * Copyright (C) 2018 Frédéric France <frederic.france@netlogic.fr>
8 *
9 * This program is free software; you can redistribute it and/or modify
10 * it under the terms of the GNU General Public License as published by
11 * the Free Software Foundation; either version 3 of the License, or
12 * (at your option) any later version.
13 *
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 * GNU General Public License for more details.
18 *
19 * You should have received a copy of the GNU General Public License
20 * along with this program. If not, see <https://www.gnu.org/licenses/>.
21 * or see https://www.gnu.org/
22 */
23
30dol_include_once('/asset/core/modules/asset/modules_asset.php');
31require_once DOL_DOCUMENT_ROOT.'/product/class/product.class.php';
32require_once DOL_DOCUMENT_ROOT.'/core/lib/company.lib.php';
33require_once DOL_DOCUMENT_ROOT.'/core/lib/functions2.lib.php';
34require_once DOL_DOCUMENT_ROOT.'/core/lib/files.lib.php';
35require_once DOL_DOCUMENT_ROOT.'/core/lib/doc.lib.php';
36
37
42{
46 public $version = 'dolibarr';
47
53 public function __construct($db)
54 {
55 global $conf, $langs, $mysoc;
56
57 // Load translation files required by the page
58 $langs->loadLangs(array("main", "companies"));
59
60 $this->db = $db;
61 $this->name = "ODT templates";
62 $this->description = $langs->trans("DocumentModelOdt");
63 $this->scandir = 'ASSET_ASSET_ADDON_PDF_ODT_PATH'; // Name of constant that is used to save list of directories to scan
64
65 // Page size for A4 format
66 $this->type = 'odt';
67 $this->page_largeur = 0;
68 $this->page_hauteur = 0;
69 $this->format = array($this->page_largeur, $this->page_hauteur);
70 $this->marge_gauche = 0;
71 $this->marge_droite = 0;
72 $this->marge_haute = 0;
73 $this->marge_basse = 0;
74
75 $this->option_logo = 1; // Display logo
76 $this->option_tva = 0; // Manage the vat option FACTURE_TVAOPTION
77 $this->option_modereg = 0; // Display payment mode
78 $this->option_condreg = 0; // Display payment terms
79 $this->option_multilang = 1; // Available in several languages
80 $this->option_escompte = 0; // Displays if there has been a discount
81 $this->option_credit_note = 0; // Support credit notes
82 $this->option_freetext = 1; // Support add of a personalised text
83 $this->option_draft_watermark = 0; // Support add of a watermark on drafts
84
85 // Get source company
86 $this->emetteur = $mysoc;
87 if (!$this->emetteur->country_code) {
88 $this->emetteur->country_code = substr($langs->defaultlang, -2); // By default if not defined
89 }
90 }
91
92
99 public function info($langs)
100 {
101 global $conf, $langs;
102
103 // Load translation files required by the page
104 $langs->loadLangs(array("errors", "companies"));
105
106 $form = new Form($this->db);
107
108 $texte = $this->description.".<br>\n";
109 $texte .= '<form action="'.$_SERVER["PHP_SELF"].'" method="POST">';
110 $texte .= '<input type="hidden" name="token" value="'.newToken().'">';
111 $texte .= '<input type="hidden" name="action" value="setModuleOptions">';
112 $texte .= '<input type="hidden" name="param1" value="ASSET_ASSET_ADDON_PDF_ODT_PATH">';
113 $texte .= '<table class="nobordernopadding" width="100%">';
114
115 // List of directories area
116 $texte .= '<tr><td>';
117 $texttitle = $langs->trans("ListOfDirectories");
118 $listofdir = explode(',', preg_replace('/[\r\n]+/', ',', trim($conf->global->ASSET_ASSET_ADDON_PDF_ODT_PATH)));
119 $listoffiles = array();
120 foreach ($listofdir as $key => $tmpdir) {
121 $tmpdir = trim($tmpdir);
122 $tmpdir = preg_replace('/DOL_DATA_ROOT/', DOL_DATA_ROOT, $tmpdir);
123 if (!$tmpdir) {
124 unset($listofdir[$key]);
125 continue;
126 }
127 if (!is_dir($tmpdir)) {
128 $texttitle .= img_warning($langs->trans("ErrorDirNotFound", $tmpdir), 0);
129 } else {
130 $tmpfiles = dol_dir_list($tmpdir, 'files', 0, '\.(ods|odt)');
131 if (count($tmpfiles)) {
132 $listoffiles = array_merge($listoffiles, $tmpfiles);
133 }
134 }
135 }
136 $texthelp = $langs->trans("ListOfDirectoriesForModelGenODT");
137 $texthelp .= '<br><br><span class="opacitymedium">'.$langs->trans("ExampleOfDirectoriesForModelGen").'</span>';
138 // Add list of substitution keys
139 $texthelp .= '<br>'.$langs->trans("FollowingSubstitutionKeysCanBeUsed").'<br>';
140 $texthelp .= $langs->transnoentitiesnoconv("FullListOnOnlineDocumentation"); // This contains an url, we don't modify it
141
142 $texte .= $form->textwithpicto($texttitle, $texthelp, 1, 'help', '', 1, 3, $this->name);
143 $texte .= '<div><div style="display: inline-block; min-width: 100px; vertical-align: middle;">';
144 $texte .= '<textarea class="flat" cols="60" name="value1">';
145 $texte .= getDolGlobalString('ASSET_ASSET_ADDON_PDF_ODT_PATH');
146 $texte .= '</textarea>';
147 $texte .= '</div><div style="display: inline-block; vertical-align: middle;">';
148 $texte .= '<input type="submit" class="button button-edit reposition smallpaddingimp" name="Button"value="'.$langs->trans("Modify").'">';
149 $texte .= '<br></div></div>';
150
151 // Scan directories
152 $nbofiles = count($listoffiles);
153 if (getDolGlobalString('ASSET_ASSET_ADDON_PDF_ODT_PATH')) {
154 $texte .= $langs->trans("NumberOfModelFilesFound").': <b>';
155 //$texte.=$nbofiles?'<a id="a_'.get_class($this).'" href="#">':'';
156 $texte .= count($listoffiles);
157 //$texte.=$nbofiles?'</a>':'';
158 $texte .= '</b>';
159 }
160
161 if ($nbofiles) {
162 $texte .= '<div id="div_'.get_class($this).'" class="hidden">';
163 foreach ($listoffiles as $file) {
164 $texte .= '- '.$file['name'].' <a href="'.DOL_URL_ROOT.'/document.php?modulepart=doctemplates&file=asset_asset/'.urlencode(basename($file['name'])).'">'.img_picto('', 'listlight').'</a>';
165 $texte .= ' &nbsp; <a class="reposition" href="'.$_SERVER["PHP_SELF"].'?modulepart=doctemplates&keyforuploaddir=ASSET_ASSET_ADDON_PDF_ODT_PATH&action=deletefile&token='.newToken().'&file='.urlencode(basename($file['name'])).'">'.img_picto('', 'delete').'</a>';
166 $texte .= '<br>';
167 }
168 $texte .= '</div>';
169 }
170
171 $texte .= '</td>';
172
173 $texte .= '</tr>';
174
175 $texte .= '</table>';
176 $texte .= '</form>';
177
178 return $texte;
179 }
180
181 // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
193 public function write_file($object, $outputlangs, $srctemplatepath, $hidedetails = 0, $hidedesc = 0, $hideref = 0)
194 {
195 // phpcs:enable
196 global $user, $langs, $conf, $mysoc, $hookmanager;
197
198 if (empty($srctemplatepath)) {
199 dol_syslog("doc_generic_odt::write_file parameter srctemplatepath empty", LOG_WARNING);
200 return -1;
201 }
202
203 // Add odtgeneration hook
204 if (!is_object($hookmanager)) {
205 include_once DOL_DOCUMENT_ROOT.'/core/class/hookmanager.class.php';
206 $hookmanager = new HookManager($this->db);
207 }
208 $hookmanager->initHooks(array('odtgeneration'));
209 global $action;
210
211 if (!is_object($outputlangs)) {
212 $outputlangs = $langs;
213 }
214 $sav_charset_output = $outputlangs->charset_output;
215 $outputlangs->charset_output = 'UTF-8';
216
217 $outputlangs->loadLangs(array("main", "dict", "companies", "bills"));
218
219 if ($conf->asset->dir_output) {
220 // If $object is id instead of object
221 if (!is_object($object)) {
222 $id = $object;
223 $object = new Asset($this->db);
224 $result = $object->fetch($id);
225 if ($result < 0) {
226 dol_print_error($this->db, $object->error);
227 return -1;
228 }
229 }
230
231 $object->fetch_thirdparty();
232
233 $dir = $conf->asset->multidir_output[isset($object->entity) ? $object->entity : 1];
234 $objectref = dol_sanitizeFileName($object->ref);
235 if (!preg_match('/specimen/i', $objectref)) {
236 $dir .= "/".$objectref;
237 }
238 $file = $dir."/".$objectref.".odt";
239
240 if (!file_exists($dir)) {
241 if (dol_mkdir($dir) < 0) {
242 $this->error = $langs->transnoentities("ErrorCanNotCreateDir", $dir);
243 return -1;
244 }
245 }
246
247 if (file_exists($dir)) {
248 //print "srctemplatepath=".$srctemplatepath; // Src filename
249 $newfile = basename($srctemplatepath);
250 $newfiletmp = preg_replace('/\.od[ts]/i', '', $newfile);
251 $newfiletmp = preg_replace('/template_/i', '', $newfiletmp);
252 $newfiletmp = preg_replace('/modele_/i', '', $newfiletmp);
253 $newfiletmp = $objectref.'_'.$newfiletmp;
254 //$file=$dir.'/'.$newfiletmp.'.'.dol_print_date(dol_now(),'%Y%m%d%H%M%S').'.odt';
255 // Get extension (ods or odt)
256 $newfileformat = substr($newfile, strrpos($newfile, '.') + 1);
257 if (getDolGlobalString('MAIN_DOC_USE_TIMING')) {
258 $format = getDolGlobalString('MAIN_DOC_USE_TIMING');
259 if ($format == '1') {
260 $format = '%Y%m%d%H%M%S';
261 }
262 $filename = $newfiletmp.'-'.dol_print_date(dol_now(), $format).'.'.$newfileformat;
263 } else {
264 $filename = $newfiletmp.'.'.$newfileformat;
265 }
266 $file = $dir.'/'.$filename;
267 //print "newdir=".$dir;
268 //print "newfile=".$newfile;
269 //print "file=".$file;
270 //print "conf->societe->dir_temp=".$conf->societe->dir_temp;
271
272 dol_mkdir($conf->asset->dir_temp);
273 if (!is_writable($conf->asset->dir_temp)) {
274 $this->error = $langs->transnoentities("ErrorFailedToWriteInTempDirectory", $conf->asset->dir_temp);
275 dol_syslog('Error in write_file: ' . $this->error, LOG_ERR);
276 return -1;
277 }
278
279 // If CUSTOMER contact defined on order, we use it
280 $usecontact = false;
281 $arrayidcontact = $object->getIdContact('external', 'CUSTOMER');
282 if (count($arrayidcontact) > 0) {
283 $usecontact = true;
284 $result = $object->fetch_contact($arrayidcontact[0]);
285 }
286
287 // Recipient name
288 $contactobject = null;
289 if (!empty($usecontact)) {
290 // We can use the company of contact instead of thirdparty company
291 if ($object->contact->socid != $object->thirdparty->id && (!isset($conf->global->MAIN_USE_COMPANY_NAME_OF_CONTACT) || getDolGlobalString('MAIN_USE_COMPANY_NAME_OF_CONTACT'))) {
292 $object->contact->fetch_thirdparty();
293 $socobject = $object->contact->thirdparty;
294 $contactobject = $object->contact;
295 } else {
296 $socobject = $object->thirdparty;
297 // if we have a CUSTOMER contact and we dont use it as thirdparty recipient we store the contact object for later use
298 $contactobject = $object->contact;
299 }
300 } else {
301 $socobject = $object->thirdparty;
302 }
303
304 // Make substitution
305 $substitutionarray = array(
306 '__FROM_NAME__' => $this->emetteur->name,
307 '__FROM_EMAIL__' => $this->emetteur->email,
308 '__TOTAL_TTC__' => $object->total_ttc,
309 '__TOTAL_HT__' => $object->total_ht,
310 '__TOTAL_VAT__' => $object->total_tva
311 );
312 complete_substitutions_array($substitutionarray, $langs, $object);
313 // Call the ODTSubstitution hook
314 $parameters = array('file'=>$file, 'object'=>$object, 'outputlangs'=>$outputlangs, 'substitutionarray'=>&$substitutionarray);
315 $reshook = $hookmanager->executeHooks('ODTSubstitution', $parameters, $this, $action); // Note that $action and $object may have been modified by some hooks
316
317 // Line of free text
318 $newfreetext = '';
319 $paramfreetext = 'ORDER_FREE_TEXT';
320 if (!empty($conf->global->$paramfreetext)) {
321 $newfreetext = make_substitutions(getDolGlobalString($paramfreetext), $substitutionarray);
322 }
323
324 // Open and load template
325 require_once ODTPHP_PATH.'odf.php';
326 try {
327 $odfHandler = new Odf(
328 $srctemplatepath,
329 array(
330 'PATH_TO_TMP' => $conf->asset->dir_temp,
331 'ZIP_PROXY' => 'PclZipProxy', // PhpZipProxy or PclZipProxy. Got "bad compression method" error when using PhpZipProxy.
332 'DELIMITER_LEFT' => '{',
333 'DELIMITER_RIGHT' => '}'
334 )
335 );
336 } catch (Exception $e) {
337 $this->error = $e->getMessage();
338 dol_syslog($e->getMessage(), LOG_INFO);
339 return -1;
340 }
341 // After construction $odfHandler->contentXml contains content and
342 // [!-- BEGIN row.lines --]*[!-- END row.lines --] has been replaced by
343 // [!-- BEGIN lines --]*[!-- END lines --]
344 //print html_entity_decode($odfHandler->__toString());
345 //print exit;
346
347
348 // Make substitutions into odt of freetext
349 try {
350 $odfHandler->setVars('free_text', $newfreetext, true, 'UTF-8');
351 } catch (OdfException $e) {
352 dol_syslog($e->getMessage(), LOG_INFO);
353 }
354
355 // Define substitution array
356 $substitutionarray = getCommonSubstitutionArray($outputlangs, 0, null, $object);
357 $array_object_from_properties = $this->get_substitutionarray_each_var_object($object, $outputlangs);
358 $array_objet = $this->get_substitutionarray_object($object, $outputlangs);
359 $array_user = $this->get_substitutionarray_user($user, $outputlangs);
360 $array_soc = $this->get_substitutionarray_mysoc($mysoc, $outputlangs);
361 $array_thirdparty = $this->get_substitutionarray_thirdparty($socobject, $outputlangs);
362 $array_other = $this->get_substitutionarray_other($outputlangs);
363 // retrieve contact information for use in object as contact_xxx tags
364 $array_thirdparty_contact = array();
365 if ($usecontact && is_object($contactobject)) {
366 $array_thirdparty_contact = $this->get_substitutionarray_contact($contactobject, $outputlangs, 'contact');
367 }
368
369 $tmparray = array_merge($substitutionarray, $array_object_from_properties, $array_user, $array_soc, $array_thirdparty, $array_objet, $array_other, $array_thirdparty_contact);
370 complete_substitutions_array($tmparray, $outputlangs, $object);
371
372 // Call the ODTSubstitution hook
373 $parameters = array('odfHandler'=>&$odfHandler, 'file'=>$file, 'object'=>$object, 'outputlangs'=>$outputlangs, 'substitutionarray'=>&$tmparray);
374 $reshook = $hookmanager->executeHooks('ODTSubstitution', $parameters, $this, $action); // Note that $action and $object may have been modified by some hooks
375
376 foreach ($tmparray as $key => $value) {
377 try {
378 if (preg_match('/logo$/', $key)) {
379 // Image
380 if (file_exists($value)) {
381 $odfHandler->setImage($key, $value);
382 } else {
383 $odfHandler->setVars($key, 'ErrorFileNotFound', true, 'UTF-8');
384 }
385 } else {
386 // Text
387 $odfHandler->setVars($key, $value, true, 'UTF-8');
388 }
389 } catch (OdfException $e) {
390 dol_syslog($e->getMessage(), LOG_INFO);
391 }
392 }
393 // Replace tags of lines
394 try {
395 $foundtagforlines = 1;
396 try {
397 $listlines = $odfHandler->setSegment('lines');
398 } catch (OdfExceptionSegmentNotFound $e) {
399 // We may arrive here if tags for lines not present into template
400 $foundtagforlines = 0;
401 dol_syslog($e->getMessage(), LOG_INFO);
402 } catch (OdfException $e) {
403 $foundtagforlines = 0;
404 dol_syslog($e->getMessage(), LOG_INFO);
405 }
406 if ($foundtagforlines) {
407 $linenumber = 0;
408 foreach ($object->lines as $line) {
409 $linenumber++;
410 $tmparray = $this->get_substitutionarray_lines($line, $outputlangs, $linenumber);
411 complete_substitutions_array($tmparray, $outputlangs, $object, $line, "completesubstitutionarray_lines");
412 // Call the ODTSubstitutionLine hook
413 $parameters = array('odfHandler'=>&$odfHandler, 'file'=>$file, 'object'=>$object, 'outputlangs'=>$outputlangs, 'substitutionarray'=>&$tmparray, 'line'=>$line);
414 $reshook = $hookmanager->executeHooks('ODTSubstitutionLine', $parameters, $this, $action); // Note that $action and $object may have been modified by some hooks
415 foreach ($tmparray as $key => $val) {
416 try {
417 $listlines->setVars($key, $val, true, 'UTF-8');
418 } catch (OdfException $e) {
419 dol_syslog($e->getMessage(), LOG_INFO);
420 } catch (SegmentException $e) {
421 dol_syslog($e->getMessage(), LOG_INFO);
422 }
423 }
424 $listlines->merge();
425 }
426 $odfHandler->mergeSegment($listlines);
427 }
428 } catch (OdfException $e) {
429 $this->error = $e->getMessage();
430 dol_syslog($this->error, LOG_WARNING);
431 return -1;
432 }
433
434 // Replace labels translated
435 $tmparray = $outputlangs->get_translations_for_substitutions();
436 foreach ($tmparray as $key => $value) {
437 try {
438 $odfHandler->setVars($key, $value, true, 'UTF-8');
439 } catch (OdfException $e) {
440 dol_syslog($e->getMessage(), LOG_INFO);
441 }
442 }
443
444 // Call the beforeODTSave hook
445
446 $parameters = array('odfHandler'=>&$odfHandler, 'file'=>$file, 'object'=>$object, 'outputlangs'=>$outputlangs, 'substitutionarray'=>&$tmparray);
447 $reshook = $hookmanager->executeHooks('beforeODTSave', $parameters, $this, $action); // Note that $action and $object may have been modified by some hooks
448
449 // Write new file
450 if (getDolGlobalString('MAIN_ODT_AS_PDF')) {
451 try {
452 $odfHandler->exportAsAttachedPDF($file);
453 } catch (Exception $e) {
454 $this->error = $e->getMessage();
455 dol_syslog($e->getMessage(), LOG_INFO);
456 return -1;
457 }
458 } else {
459 try {
460 $odfHandler->saveToDisk($file);
461 } catch (Exception $e) {
462 $this->error = $e->getMessage();
463 dol_syslog($e->getMessage(), LOG_INFO);
464 return -1;
465 }
466 }
467
468 $parameters = array('odfHandler'=>&$odfHandler, 'file'=>$file, 'object'=>$object, 'outputlangs'=>$outputlangs, 'substitutionarray'=>&$tmparray);
469 $reshook = $hookmanager->executeHooks('afterODTCreation', $parameters, $this, $action); // Note that $action and $object may have been modified by some hooks
470
471 dolChmod($file);
472
473 $odfHandler = null; // Destroy object
474
475 $this->result = array('fullpath'=>$file);
476
477 return 1; // Success
478 } else {
479 $this->error = $langs->transnoentities("ErrorCanNotCreateDir", $dir);
480 return -1;
481 }
482 }
483
484 return -1;
485 }
486}
Class for Asset.
get_substitutionarray_each_var_object(&$object, $outputlangs, $recursive=1)
Define array with couple substitution key => substitution value.
get_substitutionarray_object($object, $outputlangs, $array_key='object')
Define array with couple substitution key => substitution value Note that vars into substitutions arr...
get_substitutionarray_mysoc($mysoc, $outputlangs)
Define array with couple substitution key => substitution value.
get_substitutionarray_contact($object, $outputlangs, $array_key='object')
Define array with couple substitution key => substitution value.
get_substitutionarray_other($outputlangs)
Define array with couple substitution key => substitution value.
get_substitutionarray_lines($line, $outputlangs, $linenumber=0)
Define array with couple substitution key => substitution value Note that vars into substitutions arr...
get_substitutionarray_thirdparty($object, $outputlangs, $array_key='company')
Define array with couple substitution key => substitution value For example {company_name}...
get_substitutionarray_user($user, $outputlangs)
Define array with couple substitution key => substitution value.
Class to manage generation of HTML components Only common components must be here.
Class to manage hooks.
Parent class for documents models.
Class to build documents using ODF templates generator.
write_file($object, $outputlangs, $srctemplatepath, $hidedetails=0, $hidedesc=0, $hideref=0)
Function to build a document on disk using the generic odt module.
info($langs)
Return description of a module.
print $script_file $mode $langs defaultlang(is_numeric($duration_value) ? " delay=". $duration_value :"").(is_numeric($duration_value2) ? " after cd cd cd description as description
Only used if Module[ID]Desc translation string is not found.
dol_dir_list($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.
Definition files.lib.php:62
img_warning($titlealt='default', $moreatt='', $morecss='pictowarning')
Show warning logo.
dol_print_error($db='', $error='', $errors=null)
Displays error message system with all the information to facilitate the diagnosis and the escalation...
dol_print_date($time, $format='', $tzoutput='auto', $outputlangs='', $encodetooutput=false)
Output date in a string format according to outputlangs (or langs if not defined).
dolChmod($filepath, $newmask='')
Change mod of a file.
dol_now($mode='auto')
Return date for now.
img_picto($titlealt, $picto, $moreatt='', $pictoisfullpath=false, $srconly=0, $notitle=0, $alt='', $morecss='', $marginleftonlyshort=2)
Show picto whatever it's its name (generic function)
if(!function_exists( 'dol_getprefix')) dol_include_once($relpath, $classname='')
Make an include_once using default root and alternate root if it fails.
complete_substitutions_array(&$substitutionarray, $outputlangs, $object=null, $parameters=null, $callfunc="completesubstitutionarray")
Complete the $substitutionarray with more entries coming from external module that had set the "subst...
make_substitutions($text, $substitutionarray, $outputlangs=null, $converttextinhtmlifnecessary=0)
Make substitution into a text string, replacing keys with vals from $substitutionarray (oldval=>newva...
dol_sanitizeFileName($str, $newstr='_', $unaccent=1)
Clean a string to use it as a file name.
getCommonSubstitutionArray($outputlangs, $onlykey=0, $exclude=null, $object=null, $include=null)
Return array of possible common substitutions.
getDolGlobalString($key, $default='')
Return dolibarr global constant string value.
dol_syslog($message, $level=LOG_INFO, $ident=0, $suffixinfilename='', $restricttologhandler='', $logcontext=null)
Write log message into outputs.
dol_mkdir($dir, $dataroot='', $newmask='')
Creation of a directory (this can create recursive subdir)
if(preg_match('/crypted:/i', $dolibarr_main_db_pass)||!empty($dolibarr_main_db_encrypted_pass)) $conf db type
Definition repair.php:121
$conf db name
Only used if Module[ID]Name translation string is not found.
Definition repair.php:124