| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836283728382839284028412842284328442845284628472848284928502851285228532854285528562857285828592860286128622863286428652866286728682869287028712872287328742875287628772878287928802881288228832884288528862887288828892890289128922893289428952896289728982899290029012902290329042905290629072908290929102911291229132914291529162917291829192920292129222923292429252926292729282929293029312932293329342935293629372938293929402941294229432944294529462947294829492950295129522953295429552956295729582959296029612962296329642965296629672968296929702971297229732974297529762977297829792980298129822983298429852986298729882989299029912992299329942995299629972998299930003001300230033004300530063007300830093010301130123013301430153016301730183019302030213022302330243025302630273028302930303031303230333034303530363037303830393040304130423043304430453046304730483049305030513052305330543055305630573058305930603061306230633064306530663067306830693070307130723073307430753076307730783079308030813082308330843085308630873088308930903091309230933094309530963097309830993100310131023103310431053106310731083109311031113112311331143115311631173118311931203121312231233124312531263127312831293130313131323133313431353136313731383139314031413142314331443145314631473148314931503151315231533154315531563157315831593160316131623163316431653166316731683169317031713172317331743175317631773178317931803181318231833184318531863187318831893190319131923193319431953196319731983199320032013202320332043205320632073208320932103211321232133214321532163217321832193220322132223223322432253226322732283229323032313232323332343235323632373238323932403241324232433244324532463247324832493250325132523253325432553256325732583259326032613262326332643265326632673268326932703271327232733274327532763277327832793280328132823283328432853286328732883289 | <?php/** * @file * Tests for locale.module. * * The test file includes: *  - a functional test for the language configuration forms; *  - functional tests for the translation functionalities, including searching; *  - a functional test for the PO files import feature, including validation; *  - functional tests for translations and templates export feature; *  - functional tests for the uninstall process; *  - a functional test for the language switching feature; *  - a functional test for a user's ability to change their default language; *  - a functional test for configuring a different path alias per language; *  - a functional test for configuring a different path alias per language; *  - a functional test for multilingual support by content type and on nodes. *  - a functional test for multilingual fields. *  - a functional test for comment language. *  - a functional test fot language types/negotiation info. *//** * Functional tests for the language configuration forms. */class LocaleConfigurationTest extends DrupalWebTestCase {  public static function getInfo() {    return array(      'name' => 'Language configuration',      'description' => 'Adds a new locale and tests changing its status and the default language.',      'group' => 'Locale',    );  }  function setUp() {    parent::setUp('locale');  }  /**   * Functional tests for adding, editing and deleting languages.   */  function testLanguageConfiguration() {    global $base_url;    // User to add and remove language.    $admin_user = $this->drupalCreateUser(array('administer languages', 'access administration pages'));    $this->drupalLogin($admin_user);    // Add predefined language.    $edit = array(      'langcode' => 'fr',    );    $this->drupalPost('admin/config/regional/language/add', $edit, t('Add language'));    $this->assertText('fr', 'Language added successfully.');    $this->assertEqual($this->getUrl(), url('admin/config/regional/language', array('absolute' => TRUE)), 'Correct page redirection.');    // Add custom language.    // Code for the language.    $langcode = 'xx';    // The English name for the language.    $name = $this->randomName(16);    // The native name for the language.    $native = $this->randomName(16);    // The domain prefix.    $prefix = $langcode;    $edit = array(      'langcode' => $langcode,      'name' => $name,      'native' => $native,      'prefix' => $prefix,      'direction' => '0',    );    $this->drupalPost('admin/config/regional/language/add', $edit, t('Add custom language'));    $this->assertEqual($this->getUrl(), url('admin/config/regional/language', array('absolute' => TRUE)), 'Correct page redirection.');    $this->assertText($langcode, 'Language code found.');    $this->assertText($name, 'Name found.');    $this->assertText($native, 'Native found.');    $this->assertText($native, 'Test language added.');    // Check if we can change the default language.    $path = 'admin/config/regional/language';    $this->drupalGet($path);    $this->assertFieldChecked('edit-site-default-en', 'English is the default language.');    // Change the default language.    $edit = array(      'site_default' => $langcode,    );    $this->drupalPost(NULL, $edit, t('Save configuration'));    $this->assertNoFieldChecked('edit-site-default-en', 'Default language updated.');    $this->assertEqual($this->getUrl(), url('admin/config/regional/language', array('absolute' => TRUE)), 'Correct page redirection.');    // Check if a valid language prefix is added after changing the default    // language.    $this->drupalGet('admin/config/regional/language/edit/en');    $this->assertFieldByXPath('//input[@name="prefix"]', 'en', 'A valid path prefix has been added to the previous default language.');    // Ensure we can't delete the default language.    $this->drupalGet('admin/config/regional/language/delete/' . $langcode);    $this->assertEqual($this->getUrl(), url('admin/config/regional/language', array('absolute' => TRUE)), 'Correct page redirection.');    $this->assertText(t('The default language cannot be deleted.'), 'Failed to delete the default language.');    // Check if we can disable a language.    $edit = array(      'enabled[en]' => FALSE,    );    $this->drupalPost($path, $edit, t('Save configuration'));    $this->assertNoFieldChecked('edit-enabled-en', 'Language disabled.');    // Set disabled language to be the default and ensure it is re-enabled.    $edit = array(      'site_default' => 'en',    );    $this->drupalPost(NULL, $edit, t('Save configuration'));    $this->assertFieldChecked('edit-enabled-en', 'Default language re-enabled.');    // Ensure 'edit' link works.    $this->clickLink(t('edit'));    $this->assertTitle(t('Edit language | Drupal'), 'Page title is "Edit language".');    // Edit a language.    $name = $this->randomName(16);    $edit = array(      'name' => $name,    );    $this->drupalPost('admin/config/regional/language/edit/' . $langcode, $edit, t('Save language'));    $this->assertRaw($name, 'The language has been updated.');    $this->assertEqual($this->getUrl(), url('admin/config/regional/language', array('absolute' => TRUE)), 'Correct page redirection.');    // Ensure 'delete' link works.    $this->drupalGet('admin/config/regional/language');    $this->clickLink(t('delete'));    $this->assertText(t('Are you sure you want to delete the language'), '"delete" link is correct.');    // Delete an enabled language.    $this->drupalGet('admin/config/regional/language/delete/' . $langcode);    // First test the 'cancel' link.    $this->clickLink(t('Cancel'));    $this->assertEqual($this->getUrl(), url('admin/config/regional/language', array('absolute' => TRUE)), 'Correct page redirection.');    $this->assertRaw($name, 'The language was not deleted.');    // Delete the language for real. This a confirm form, we do not need any    // fields changed.    $this->drupalPost('admin/config/regional/language/delete/' . $langcode, array(), t('Delete'));    // We need raw here because %locale will add HTML.    $this->assertRaw(t('The language %locale has been removed.', array('%locale' => $name)), 'The test language has been removed.');    $this->assertEqual($this->getUrl(), url('admin/config/regional/language', array('absolute' => TRUE)), 'Correct page redirection.');    // Verify that language is no longer found.    $this->drupalGet('admin/config/regional/language/delete/' . $langcode);    $this->assertResponse(404, 'Language no longer found.');    // Make sure the "language_count" variable has been updated correctly.    drupal_static_reset('language_list');    $enabled = language_list('enabled');    $this->assertEqual(variable_get('language_count', 1), count($enabled[1]), 'Language count is correct.');    // Delete a disabled language.    // Disable an enabled language.    $edit = array(      'enabled[fr]' => FALSE,    );    $this->drupalPost($path, $edit, t('Save configuration'));    $this->assertNoFieldChecked('edit-enabled-fr', 'French language disabled.');    // Get the count of enabled languages.    drupal_static_reset('language_list');    $enabled = language_list('enabled');    // Delete the disabled language.    $this->drupalPost('admin/config/regional/language/delete/fr', array(), t('Delete'));    // We need raw here because %locale will add HTML.    $this->assertRaw(t('The language %locale has been removed.', array('%locale' => 'French')), 'Disabled language has been removed.');    $this->assertEqual($this->getUrl(), url('admin/config/regional/language', array('absolute' => TRUE)), 'Correct page redirection.');    // Verify that language is no longer found.    $this->drupalGet('admin/config/regional/language/delete/fr');    $this->assertResponse(404, 'Language no longer found.');    // Make sure the "language_count" variable has not changed.    $this->assertEqual(variable_get('language_count', 1), count($enabled[1]), 'Language count is correct.');    // Ensure we can't delete the English language.    $this->drupalGet('admin/config/regional/language/delete/en');    $this->assertEqual($this->getUrl(), url('admin/config/regional/language', array('absolute' => TRUE)), 'Correct page redirection.');    $this->assertText(t('The English language cannot be deleted.'), 'Failed to delete English language.');  }}/** * Tests localization of the JavaScript libraries. * * Currently, only the jQuery datepicker is localized using Drupal translations. */class LocaleLibraryInfoAlterTest extends DrupalWebTestCase {  public static function getInfo() {    return array(      'name' => 'Javascript library localisation',      'description' => 'Tests the localisation of JavaScript libraries.',      'group' => 'Locale',    );  }  function setUp() {    parent::setUp('locale', 'locale_test');  }  /**   * Verifies that the datepicker can be localized.   *   * @see locale_library_info_alter()   */  public function testLibraryInfoAlter() {    drupal_add_library('system', 'ui.datepicker');    $scripts = drupal_get_js();    $this->assertTrue(strpos($scripts, 'locale.datepicker.js'), 'locale.datepicker.js added to scripts.');  }}/** * Functional tests for JavaScript parsing for translatable strings. */class LocaleJavascriptTranslationTest extends DrupalWebTestCase {  public static function getInfo() {    return array(      'name' => 'Javascript translation',      'description' => 'Tests parsing js files for translatable strings',      'group' => 'Locale',    );  }  function setUp() {    parent::setUp('locale', 'locale_test');  }  function testFileParsing() {    $filename = drupal_get_path('module', 'locale_test') . '/locale_test.js';    // Parse the file to look for source strings.    _locale_parse_js_file($filename);    // Get all of the source strings that were found.    $source_strings = db_select('locales_source', 's')      ->fields('s', array('source', 'context'))      ->condition('s.location', $filename)      ->execute()      ->fetchAllKeyed();    // List of all strings that should be in the file.    $test_strings = array(      "Standard Call t" => '',      "Whitespace Call t" => '',      "Single Quote t" => '',      "Single Quote \\'Escaped\\' t" => '',      "Single Quote Concat strings t" => '',      "Double Quote t" => '',      "Double Quote \\\"Escaped\\\" t" => '',      "Double Quote Concat strings t" => '',      "Context !key Args t" => "Context string",      "Context Unquoted t" => "Context string unquoted",      "Context Single Quoted t" => "Context string single quoted",      "Context Double Quoted t" => "Context string double quoted",      "Standard Call plural" => '',      "Standard Call @count plural" => '',      "Whitespace Call plural" => '',      "Whitespace Call @count plural" => '',      "Single Quote plural" => '',      "Single Quote @count plural" => '',      "Single Quote \\'Escaped\\' plural" => '',      "Single Quote \\'Escaped\\' @count plural" => '',      "Double Quote plural" => '',      "Double Quote @count plural" => '',      "Double Quote \\\"Escaped\\\" plural" => '',      "Double Quote \\\"Escaped\\\" @count plural" => '',      "Context !key Args plural" => "Context string",      "Context !key Args @count plural" => "Context string",      "Context Unquoted plural" => "Context string unquoted",      "Context Unquoted @count plural" => "Context string unquoted",      "Context Single Quoted plural" => "Context string single quoted",      "Context Single Quoted @count plural" => "Context string single quoted",      "Context Double Quoted plural" => "Context string double quoted",      "Context Double Quoted @count plural" => "Context string double quoted",    );    // Assert that all strings were found properly.    foreach ($test_strings as $str => $context) {      $args = array('%source' => $str, '%context' => $context);      // Make sure that the string was found in the file.      $this->assertTrue(isset($source_strings[$str]), format_string('Found source string: %source', $args));      // Make sure that the proper context was matched.      $this->assertTrue(isset($source_strings[$str]) && $source_strings[$str] === $context, strlen($context) > 0 ? format_string('Context for %source is %context', $args) : format_string('Context for %source is blank', $args));    }    $this->assertEqual(count($source_strings), count($test_strings), 'Found correct number of source strings.');  }}/** * Functional test for string translation and validation. */class LocaleTranslationFunctionalTest extends DrupalWebTestCase {  public static function getInfo() {    return array(      'name' => 'String translate, search and validate',      'description' => 'Adds a new locale and translates its name. Checks the validation of translation strings and search results.',      'group' => 'Locale',    );  }  function setUp() {    parent::setUp('locale');  }  /**   * Adds a language and tests string translation by users with the appropriate permissions.   */  function testStringTranslation() {    global $base_url;    // User to add and remove language.    $admin_user = $this->drupalCreateUser(array('administer languages', 'access administration pages'));    // User to translate and delete string.    $translate_user = $this->drupalCreateUser(array('translate interface', 'access administration pages'));    // Code for the language.    $langcode = 'xx';    // The English name for the language. This will be translated.    $name = $this->randomName(16);    // The native name for the language.    $native = $this->randomName(16);    // The domain prefix.    $prefix = $langcode;    // This is the language indicator on the translation search screen for    // untranslated strings. Copied straight from locale.inc.    $language_indicator = "<em class=\"locale-untranslated\">$langcode</em> ";    // This will be the translation of $name.    $translation = $this->randomName(16);    // Add custom language.    $this->drupalLogin($admin_user);    $edit = array(      'langcode' => $langcode,      'name' => $name,      'native' => $native,      'prefix' => $prefix,      'direction' => '0',    );    $this->drupalPost('admin/config/regional/language/add', $edit, t('Add custom language'));    // Add string.    t($name, array(), array('langcode' => $langcode));    // Reset locale cache.    locale_reset();    $this->assertText($langcode, 'Language code found.');    $this->assertText($name, 'Name found.');    $this->assertText($native, 'Native found.');    // No t() here, we do not want to add this string to the database and it's    // surely not translated yet.    $this->assertText($native, 'Test language added.');    $this->drupalLogout();    // Search for the name and translate it.    $this->drupalLogin($translate_user);    $search = array(      'string' => $name,      'language' => 'all',      'translation' => 'all',      'group' => 'all',    );    $this->drupalPost('admin/config/regional/translate/translate', $search, t('Filter'));    // assertText() seems to remove the input field where $name always could be    // found, so this is not a false assert. See how assertNoText succeeds    // later.    $this->assertText($name, 'Search found the name.');    $this->assertRaw($language_indicator, 'Name is untranslated.');    // Assume this is the only result, given the random name.    $this->clickLink(t('edit'));    // We save the lid from the path.    $matches = array();    preg_match('!admin/config/regional/translate/edit/(\d+)!', $this->getUrl(), $matches);    $lid = $matches[1];    // No t() here, it's surely not translated yet.    $this->assertText($name, 'name found on edit screen.');    $edit = array(      "translations[$langcode]" => $translation,    );    $this->drupalPost(NULL, $edit, t('Save translations'));    $this->assertText(t('The string has been saved.'), 'The string has been saved.');    $this->assertEqual($this->getUrl(), url('admin/config/regional/translate/translate', array('absolute' => TRUE)), 'Correct page redirection.');    $this->assertTrue($name != $translation && t($name, array(), array('langcode' => $langcode)) == $translation, 't() works.');    $this->drupalPost('admin/config/regional/translate/translate', $search, t('Filter'));    // The indicator should not be here.    $this->assertNoRaw($language_indicator, 'String is translated.');    // Verify that a translation set which has an empty target string can be    // updated without any database error.    db_update('locales_target')      ->fields(array('translation' => ''))      ->condition('language', $langcode, '=')      ->condition('lid', $lid, '=')      ->execute();    $this->drupalPost('admin/config/regional/translate/edit/' . $lid, $edit, t('Save translations'));    $this->assertText(t('The string has been saved.'), 'The string has been saved.');    // Try to edit a non-existent string and ensure we're redirected correctly.    // Assuming we don't have 999,999 strings already.    $random_lid = 999999;    $this->drupalGet('admin/config/regional/translate/edit/' . $random_lid);    $this->assertText(t('String not found'), 'String not found.');    $this->assertEqual($this->getUrl(), url('admin/config/regional/translate/translate', array('absolute' => TRUE)), 'Correct page redirection.');    $this->drupalLogout();    // Delete the language.    $this->drupalLogin($admin_user);    $path = 'admin/config/regional/language/delete/' . $langcode;    // This a confirm form, we do not need any fields changed.    $this->drupalPost($path, array(), t('Delete'));    // We need raw here because %locale will add HTML.    $this->assertRaw(t('The language %locale has been removed.', array('%locale' => $name)), 'The test language has been removed.');    // Reload to remove $name.    $this->drupalGet($path);    // Verify that language is no longer found.    $this->assertResponse(404, 'Language no longer found.');    $this->drupalLogout();    // Delete the string.    $this->drupalLogin($translate_user);    $search = array(      'string' => $name,      'language' => 'all',      'translation' => 'all',      'group' => 'all',    );    $this->drupalPost('admin/config/regional/translate/translate', $search, t('Filter'));    // Assume this is the only result, given the random name.    $this->clickLink(t('delete'));    $this->assertText(t('Are you sure you want to delete the string'), '"delete" link is correct.');    // Delete the string.    $path = 'admin/config/regional/translate/delete/' . $lid;    $this->drupalGet($path);    // First test the 'cancel' link.    $this->clickLink(t('Cancel'));    $this->assertEqual($this->getUrl(), url('admin/config/regional/translate/translate', array('absolute' => TRUE)), 'Correct page redirection.');    $this->assertRaw($name, 'The string was not deleted.');    // Delete the name string.    $this->drupalPost('admin/config/regional/translate/delete/' . $lid, array(), t('Delete'));    $this->assertText(t('The string has been removed.'), 'The string has been removed message.');    $this->assertEqual($this->getUrl(), url('admin/config/regional/translate/translate', array('absolute' => TRUE)), 'Correct page redirection.');    $this->drupalPost('admin/config/regional/translate/translate', $search, t('Filter'));    $this->assertNoText($name, 'Search now can not find the name.');  }  /*   * Adds a language and checks that the JavaScript translation files are   * properly created and rebuilt on deletion.   */  function testJavaScriptTranslation() {    $user = $this->drupalCreateUser(array('translate interface', 'administer languages', 'access administration pages'));    $this->drupalLogin($user);    $langcode = 'xx';    // The English name for the language. This will be translated.    $name = $this->randomName(16);    // The native name for the language.    $native = $this->randomName(16);    // The domain prefix.    $prefix = $langcode;    // Add custom language.    $edit = array(      'langcode' => $langcode,      'name' => $name,      'native' => $native,      'prefix' => $prefix,      'direction' => '0',    );    $this->drupalPost('admin/config/regional/language/add', $edit, t('Add custom language'));    drupal_static_reset('language_list');    // Build the JavaScript translation file.    $this->drupalGet('admin/config/regional/translate/translate');    // Retrieve the id of the first string available in the {locales_source}    // table and translate it.    $query = db_select('locales_source', 'l');    $query->addExpression('min(l.lid)', 'lid');    $result = $query->condition('l.location', '%.js%', 'LIKE')      ->condition('l.textgroup', 'default')      ->execute();    $url = 'admin/config/regional/translate/edit/' . $result->fetchObject()->lid;    $edit = array('translations['. $langcode .']' => $this->randomName());    $this->drupalPost($url, $edit, t('Save translations'));    // Trigger JavaScript translation parsing and building.    require_once DRUPAL_ROOT . '/includes/locale.inc';    _locale_rebuild_js($langcode);    // Retrieve the JavaScript translation hash code for the custom language to    // check that the translation file has been properly built.    $file = db_select('languages', 'l')      ->fields('l', array('javascript'))      ->condition('language', $langcode)      ->execute()      ->fetchObject();    $js_file = 'public://' . variable_get('locale_js_directory', 'languages') . '/' . $langcode . '_' . $file->javascript . '.js';    $this->assertTrue($result = file_exists($js_file), format_string('JavaScript file created: %file', array('%file' => $result ? $js_file : 'not found')));    // Test JavaScript translation rebuilding.    file_unmanaged_delete($js_file);    $this->assertTrue($result = !file_exists($js_file), format_string('JavaScript file deleted: %file', array('%file' => $result ? $js_file : 'found')));    cache_clear_all();    _locale_rebuild_js($langcode);    $this->assertTrue($result = file_exists($js_file), format_string('JavaScript file rebuilt: %file', array('%file' => $result ? $js_file : 'not found')));  }  /**   * Tests the validation of the translation input.   */  function testStringValidation() {    global $base_url;    // User to add language and strings.    $admin_user = $this->drupalCreateUser(array('administer languages', 'access administration pages', 'translate interface'));    $this->drupalLogin($admin_user);    $langcode = 'xx';    // The English name for the language. This will be translated.    $name = $this->randomName(16);    // The native name for the language.    $native = $this->randomName(16);    // The domain prefix.    $prefix = $langcode;    // This is the language indicator on the translation search screen for    // untranslated strings. Copied straight from locale.inc.    $language_indicator = "<em class=\"locale-untranslated\">$langcode</em> ";    // These will be the invalid translations of $name.    $key = $this->randomName(16);    $bad_translations[$key] = "<script>alert('xss');</script>" . $key;    $key = $this->randomName(16);    $bad_translations[$key] = '<img SRC="javascript:alert(\'xss\');">' . $key;    $key = $this->randomName(16);    $bad_translations[$key] = '<<SCRIPT>alert("xss");//<</SCRIPT>' . $key;    $key = $this->randomName(16);    $bad_translations[$key] ="<BODY ONLOAD=alert('xss')>" . $key;    // Add custom language.    $edit = array(      'langcode' => $langcode,      'name' => $name,      'native' => $native,      'prefix' => $prefix,      'direction' => '0',    );    $this->drupalPost('admin/config/regional/language/add', $edit, t('Add custom language'));    // Add string.    t($name, array(), array('langcode' => $langcode));    // Reset locale cache.    $search = array(      'string' => $name,      'language' => 'all',      'translation' => 'all',      'group' => 'all',    );    $this->drupalPost('admin/config/regional/translate/translate', $search, t('Filter'));    // Find the edit path.    $content = $this->drupalGetContent();    $this->assertTrue(preg_match('@(admin/config/regional/translate/edit/[0-9]+)@', $content, $matches), 'Found the edit path.');    $path = $matches[0];    foreach ($bad_translations as $key => $translation) {      $edit = array(        "translations[$langcode]" => $translation,      );      $this->drupalPost($path, $edit, t('Save translations'));      // Check for a form error on the textarea.      $form_class = $this->xpath('//form[@id="locale-translate-edit-form"]//textarea/@class');      $this->assertNotIdentical(FALSE, strpos($form_class[0], 'error'), 'The string was rejected as unsafe.');      $this->assertNoText(t('The string has been saved.'), 'The string was not saved.');    }  }  /**   * Tests translation search form.   */  function testStringSearch() {    global $base_url;    // User to add and remove language.    $admin_user = $this->drupalCreateUser(array('administer languages', 'access administration pages'));    // User to translate and delete string.    $translate_user = $this->drupalCreateUser(array('translate interface', 'access administration pages'));    // Code for the language.    $langcode = 'xx';    // The English name for the language. This will be translated.    $name = $this->randomName(16);    // The native name for the language.    $native = $this->randomName(16);    // The domain prefix.    $prefix = $langcode;    // This is the language indicator on the translation search screen for    // untranslated strings. Copied straight from locale.inc.    $language_indicator = "<em class=\"locale-untranslated\">$langcode</em> ";    // This will be the translation of $name.    $translation = $this->randomName(16);    // Add custom language.    $this->drupalLogin($admin_user);    $edit = array(      'langcode' => $langcode,      'name' => $name,      'native' => $native,      'prefix' => $prefix,      'direction' => '0',    );    $this->drupalPost('admin/config/regional/language/add', $edit, t('Add custom language'));    // Add string.    t($name, array(), array('langcode' => $langcode));    // Reset locale cache.    locale_reset();    $this->drupalLogout();    // Search for the name.    $this->drupalLogin($translate_user);    $search = array(      'string' => $name,      'language' => 'all',      'translation' => 'all',      'group' => 'all',    );    $this->drupalPost('admin/config/regional/translate/translate', $search, t('Filter'));    // assertText() seems to remove the input field where $name always could be    // found, so this is not a false assert. See how assertNoText succeeds    // later.    $this->assertText($name, 'Search found the string.');    // Ensure untranslated string doesn't appear if searching on 'only    // translated strings'.    $search = array(      'string' => $name,      'language' => 'all',      'translation' => 'translated',      'group' => 'all',    );    $this->drupalPost('admin/config/regional/translate/translate', $search, t('Filter'));    $this->assertText(t('No strings available.'), "Search didn't find the string.");    // Ensure untranslated string appears if searching on 'only untranslated    // strings' in "all" (hasn't been translated to any language).    $search = array(      'string' => $name,      'language' => 'all',      'translation' => 'untranslated',      'group' => 'all',    );    $this->drupalPost('admin/config/regional/translate/translate', $search, t('Filter'));    $this->assertNoText(t('No strings available.'), 'Search found the string.');    // Ensure untranslated string appears if searching on 'only untranslated    // strings' in the custom language (hasn't been translated to that specific language).    $search = array(      'string' => $name,      'language' => $langcode,      'translation' => 'untranslated',      'group' => 'all',    );    $this->drupalPost('admin/config/regional/translate/translate', $search, t('Filter'));    $this->assertNoText(t('No strings available.'), 'Search found the string.');    // Add translation.    // Assume this is the only result, given the random name.    $this->clickLink(t('edit'));    // We save the lid from the path.    $matches = array();    preg_match('!admin/config/regional/translate/edit/(\d)+!', $this->getUrl(), $matches);    $lid = $matches[1];    $edit = array(      "translations[$langcode]" => $translation,    );    $this->drupalPost(NULL, $edit, t('Save translations'));    // Ensure translated string does appear if searching on 'only    // translated strings'.    $search = array(      'string' => $translation,      'language' => 'all',      'translation' => 'translated',      'group' => 'all',    );    $this->drupalPost('admin/config/regional/translate/translate', $search, t('Filter'));    $this->assertNoText(t('No strings available.'), 'Search found the translation.');    // Ensure translated source string doesn't appear if searching on 'only    // untranslated strings'.    $search = array(      'string' => $name,      'language' => 'all',      'translation' => 'untranslated',      'group' => 'all',    );    $this->drupalPost('admin/config/regional/translate/translate', $search, t('Filter'));    $this->assertText(t('No strings available.'), "Search didn't find the source string.");    // Ensure translated string doesn't appear if searching on 'only    // untranslated strings'.    $search = array(      'string' => $translation,      'language' => 'all',      'translation' => 'untranslated',      'group' => 'all',    );    $this->drupalPost('admin/config/regional/translate/translate', $search, t('Filter'));    $this->assertText(t('No strings available.'), "Search didn't find the translation.");    // Ensure translated string does appear if searching on the custom language.    $search = array(      'string' => $translation,      'language' => $langcode,      'translation' => 'all',      'group' => 'all',    );    $this->drupalPost('admin/config/regional/translate/translate', $search, t('Filter'));    $this->assertNoText(t('No strings available.'), 'Search found the translation.');    // Ensure translated string doesn't appear if searching on English.    $search = array(      'string' => $translation,      'language' => 'en',      'translation' => 'all',      'group' => 'all',    );    $this->drupalPost('admin/config/regional/translate/translate', $search, t('Filter'));    $this->assertText(t('No strings available.'), "Search didn't find the translation.");    // Search for a string that isn't in the system.    $unavailable_string = $this->randomName(16);    $search = array(      'string' => $unavailable_string,      'language' => 'all',      'translation' => 'all',      'group' => 'all',    );    $this->drupalPost('admin/config/regional/translate/translate', $search, t('Filter'));    $this->assertText(t('No strings available.'), "Search didn't find the invalid string.");  }}/** * Tests plural index computation functionality. */class LocalePluralFormatTest extends DrupalWebTestCase {  public static function getInfo() {    return array(      'name' => 'Plural formula evaluation',      'description' => 'Tests plural formula evaluation for various languages.',      'group' => 'Locale',    );  }  function setUp() {    parent::setUp('locale', 'locale_test');    $admin_user = $this->drupalCreateUser(array('administer languages', 'translate interface', 'access administration pages'));    $this->drupalLogin($admin_user);    // Import some .po files with formulas to set up the environment.    // These will also add the languages to the system and enable them.    $this->importPoFile($this->getPoFileWithSimplePlural(), array(      'langcode' => 'fr',    ));    $this->importPoFile($this->getPoFileWithComplexPlural(), array(      'langcode' => 'hr',    ));  }  /**   * Tests locale_get_plural() functionality.   */  function testGetPluralFormat() {    $this->drupalGet('locale_test_plural_format_page');    $tests = _locale_test_plural_format_tests();    $result = array();    foreach ($tests as $test) {      $this->assertPluralFormat($test['count'], $test['language'], $test['expected-result']);    }  }  /**   * Helper assert to test locale_get_plural page.   *   * @param $count   *  Number for testing.   * @param $lang   *  Language for testing   * @param $expected_result   *   Expected result.   * @param $message   */  function assertPluralFormat($count, $lang, $expected_result) {    $message_param =  array(      '@lang' => $lang,      '@count' => $count,      '@expected_result' => $expected_result,    );    $message = t("Computed plural index for '@lang' with count @count is @expected_result.", $message_param);    $message_param = array(      '@lang' => $lang,      '@expected_result' => $expected_result,    );    $this->assertText(format_string('Language: @lang, locale_get_plural: @expected_result.', $message_param, $message));  }  /**   * Imports a standalone .po file in a given language.   *   * @param $contents   *   Contents of the .po file to import.   * @param $options   *   Additional options to pass to the translation import form.   */  function importPoFile($contents, array $options = array()) {    $name = tempnam('temporary://', "po_") . '.po';    file_put_contents($name, $contents);    $options['files[file]'] = $name;    $this->drupalPost('admin/config/regional/translate/import', $options, t('Import'));    drupal_unlink($name);  }  /**   * Returns a .po file with a simple plural formula.   */  function getPoFileWithSimplePlural() {    return <<< EOFmsgid ""msgstr """Project-Id-Version: Drupal 7\\n""MIME-Version: 1.0\\n""Content-Type: text/plain; charset=UTF-8\\n""Content-Transfer-Encoding: 8bit\\n""Plural-Forms: nplurals=2; plural=(n!=1);\\n"msgid "One sheep"msgid_plural "@count sheep"msgstr[0] "un mouton"msgstr[1] "@count moutons"msgid "Monday"msgstr "lundi"EOF;  }  /**   * Returns a .po file with a complex plural formula.   */  function getPoFileWithComplexPlural() {    return <<< EOFmsgid ""msgstr """Project-Id-Version: Drupal 7\\n""MIME-Version: 1.0\\n""Content-Type: text/plain; charset=UTF-8\\n""Content-Transfer-Encoding: 8bit\\n""Plural-Forms: nplurals=3; plural=n%10==1 && n%100!=11 ? 0 : n%10>=2 && n%10<=4 && (n%100<10 || n%100>=20) ? 1 : 2;\\n"msgid "1 hour"msgid_plural "@count hours"msgstr[0] "@count sat"msgstr[1] "@count sata"msgstr[2] "@count sati"msgid "Monday"msgstr "Ponedjeljak"EOF;  }}/** * Functional tests for the import of translation files. */class LocaleImportFunctionalTest extends DrupalWebTestCase {  public static function getInfo() {    return array(      'name' => 'Translation import',      'description' => 'Tests the import of locale files.',      'group' => 'Locale',    );  }  /**   * A user able to create languages and import translations.   */  protected $admin_user = NULL;  function setUp() {    parent::setUp('locale', 'locale_test');    $this->admin_user = $this->drupalCreateUser(array('administer languages', 'translate interface', 'access administration pages'));    $this->drupalLogin($this->admin_user);  }  /**   * Test import of standalone .po files.   */  function testStandalonePoFile() {    // Try importing a .po file.    $this->importPoFile($this->getPoFile(), array(      'langcode' => 'fr',    ));    // The import should automatically create the corresponding language.    $this->assertRaw(t('The language %language has been created.', array('%language' => 'French')), 'The language has been automatically created.');    // The import should have created 7 strings.    $this->assertRaw(t('The translation was successfully imported. There are %number newly created translated strings, %update strings were updated and %delete strings were removed.', array('%number' => 9, '%update' => 0, '%delete' => 0)), 'The translation file was successfully imported.');    // This import should have saved plural forms to have 2 variants.    $this->assert(db_query("SELECT plurals FROM {languages} WHERE language = 'fr'")->fetchField() == 2, 'Plural number initialized.');    // Ensure we were redirected correctly.    $this->assertEqual($this->getUrl(), url('admin/config/regional/translate', array('absolute' => TRUE)), 'Correct page redirection.');    // Try importing a .po file with invalid tags in the default text group.    $this->importPoFile($this->getBadPoFile(), array(      'langcode' => 'fr',    ));    // The import should have created 1 string and rejected 2.    $this->assertRaw(t('The translation was successfully imported. There are %number newly created translated strings, %update strings were updated and %delete strings were removed.', array('%number' => 1, '%update' => 0, '%delete' => 0)), 'The translation file was successfully imported.');    $skip_message = format_plural(2, 'One translation string was skipped because it contains disallowed HTML.', '@count translation strings were skipped because they contain disallowed HTML.');    $this->assertRaw($skip_message, 'Unsafe strings were skipped.');    // Try importing a .po file with invalid tags in a non default text group.    $this->importPoFile($this->getBadPoFile(), array(      'langcode' => 'fr',      'group' => 'custom',    ));    // The import should have created 3 strings.    $this->assertRaw(t('The translation was successfully imported. There are %number newly created translated strings, %update strings were updated and %delete strings were removed.', array('%number' => 3, '%update' => 0, '%delete' => 0)), 'The translation file was successfully imported.');    // Try importing a .po file which doesn't exist.    $name = $this->randomName(16);    $this->drupalPost('admin/config/regional/translate/import', array(      'langcode' => 'fr',      'files[file]' => $name,      'group' => 'custom',    ), t('Import'));    $this->assertEqual($this->getUrl(), url('admin/config/regional/translate/import', array('absolute' => TRUE)), 'Correct page redirection.');    $this->assertText(t('File to import not found.'), 'File to import not found message.');    // Try importing a .po file with overriding strings, and ensure existing    // strings are kept.    $this->importPoFile($this->getOverwritePoFile(), array(      'langcode' => 'fr',      'mode' => 1, // Existing strings are kept, only new strings are added.    ));    // The import should have created 1 string.    $this->assertRaw(t('The translation was successfully imported. There are %number newly created translated strings, %update strings were updated and %delete strings were removed.', array('%number' => 1, '%update' => 0, '%delete' => 0)), 'The translation file was successfully imported.');    // Ensure string wasn't overwritten.    $search = array(      'string' => 'Montag',      'language' => 'fr',      'translation' => 'translated',      'group' => 'all',    );    $this->drupalPost('admin/config/regional/translate/translate', $search, t('Filter'));    $this->assertText(t('No strings available.'), 'String not overwritten by imported string.');    // This import should not have changed number of plural forms.    $this->assert(db_query("SELECT plurals FROM {languages} WHERE language = 'fr'")->fetchField() == 2, 'Plural numbers untouched.');    $this->importPoFile($this->getPoFileWithBrokenPlural(), array(      'langcode' => 'fr',      'mode' => 1, // Existing strings are kept, only new strings are added.    ));    // Attempt to import broken .po file as well to prove that this    // will not overwrite the proper plural formula imported above.    $this->assert(db_query("SELECT plurals FROM {languages} WHERE language = 'fr'")->fetchField() == 2, 'Broken plurals: plural numbers untouched.');    $this->importPoFile($this->getPoFileWithMissingPlural(), array(      'langcode' => 'fr',      'mode' => 1, // Existing strings are kept, only new strings are added.    ));    // Attempt to import .po file which has no plurals and prove that this    // will not overwrite the proper plural formula imported above.    $this->assert(db_query("SELECT plurals FROM {languages} WHERE language = 'fr'")->fetchField() == 2, 'No plurals: plural numbers untouched.');    // Try importing a .po file with overriding strings, and ensure existing    // strings are overwritten.    $this->importPoFile($this->getOverwritePoFile(), array(      'langcode' => 'fr',      'mode' => 0, // Strings in the uploaded file replace existing ones, new ones are added.    ));    // The import should have updated 2 strings.    $this->assertRaw(t('The translation was successfully imported. There are %number newly created translated strings, %update strings were updated and %delete strings were removed.', array('%number' => 0, '%update' => 2, '%delete' => 0)), 'The translation file was successfully imported.');    // Ensure string was overwritten.    $search = array(      'string' => 'Montag',      'language' => 'fr',      'translation' => 'translated',      'group' => 'all',    );    $this->drupalPost('admin/config/regional/translate/translate', $search, t('Filter'));    $this->assertNoText(t('No strings available.'), 'String overwritten by imported string.');    // This import should have changed number of plural forms.    $this->assert(db_query("SELECT plurals FROM {languages} WHERE language = 'fr'")->fetchField() == 3, 'Plural numbers changed.');  }  /**   * Test automatic import of a module's translation files when a language is   * enabled.   */  function testAutomaticModuleTranslationImportLanguageEnable() {    // Code for the language - manually set to match the test translation file.    $langcode = 'xx';    // The English name for the language.    $name = $this->randomName(16);    // The native name for the language.    $native = $this->randomName(16);    // The domain prefix.    $prefix = $langcode;    // Create a custom language.    $edit = array(      'langcode' => $langcode,      'name' => $name,      'native' => $native,      'prefix' => $prefix,      'direction' => '0',    );    $this->drupalPost('admin/config/regional/language/add', $edit, t('Add custom language'));    // Ensure the translation file was automatically imported when language was    // added.    $this->assertText(t('One translation file imported for the enabled modules.'), 'Language file automatically imported.');    // Ensure strings were successfully imported.    $search = array(      'string' => 'lundi',      'language' => $langcode,      'translation' => 'translated',      'group' => 'all',    );    $this->drupalPost('admin/config/regional/translate/translate', $search, t('Filter'));    $this->assertNoText(t('No strings available.'), 'String successfully imported.');  }  /**   * Test msgctxt context support.   */  function testLanguageContext() {    // Try importing a .po file.    $this->importPoFile($this->getPoFileWithContext(), array(      'langcode' => 'hr',    ));    $this->assertIdentical(t('May', array(), array('langcode' => 'hr', 'context' => 'Long month name')), 'Svibanj', 'Long month name context is working.');    $this->assertIdentical(t('May', array(), array('langcode' => 'hr')), 'Svi.', 'Default context is working.');  }  /**   * Test empty msgstr at end of .po file see #611786.   */  function testEmptyMsgstr() {    $langcode = 'hu';    // Try importing a .po file.    $this->importPoFile($this->getPoFileWithMsgstr(), array(      'langcode' => $langcode,    ));    $this->assertRaw(t('The translation was successfully imported. There are %number newly created translated strings, %update strings were updated and %delete strings were removed.', array('%number' => 1, '%update' => 0, '%delete' => 0)), 'The translation file was successfully imported.');    $this->assertIdentical(t('Operations', array(), array('langcode' => $langcode)), 'Műveletek', 'String imported and translated.');    // Try importing a .po file.    $this->importPoFile($this->getPoFileWithEmptyMsgstr(), array(      'langcode' => $langcode,      'mode' => 0,    ));    $this->assertRaw(t('The translation was successfully imported. There are %number newly created translated strings, %update strings were updated and %delete strings were removed.', array('%number' => 0, '%update' => 0, '%delete' => 1)), 'The translation file was successfully imported.');    // This is the language indicator on the translation search screen for    // untranslated strings. Copied straight from locale.inc.    $language_indicator = "<em class=\"locale-untranslated\">$langcode</em> ";    $str = "Operations";    $search = array(      'string' => $str,      'language' => 'all',      'translation' => 'all',      'group' => 'all',    );    $this->drupalPost('admin/config/regional/translate/translate', $search, t('Filter'));    // assertText() seems to remove the input field where $str always could be    // found, so this is not a false assert.    $this->assertText($str, 'Search found the string.');    $this->assertRaw($language_indicator, 'String is untranslated again.');  }  /**   * Helper function: import a standalone .po file in a given language.   *   * @param $contents   *   Contents of the .po file to import.   * @param $options   *   Additional options to pass to the translation import form.   */  function importPoFile($contents, array $options = array()) {    $name = tempnam('temporary://', "po_") . '.po';    file_put_contents($name, $contents);    $options['files[file]'] = $name;    $this->drupalPost('admin/config/regional/translate/import', $options, t('Import'));    drupal_unlink($name);  }  /**   * Helper function that returns a proper .po file.   */  function getPoFile() {    return <<< EOFmsgid ""msgstr """Project-Id-Version: Drupal 7\\n""MIME-Version: 1.0\\n""Content-Type: text/plain; charset=UTF-8\\n""Content-Transfer-Encoding: 8bit\\n""Plural-Forms: nplurals=2; plural=(n > 1);\\n"msgid "One sheep"msgid_plural "@count sheep"msgstr[0] "un mouton"msgstr[1] "@count moutons"msgid "Monday"msgstr "lundi"msgid "Tuesday"msgstr "mardi"msgid "Wednesday"msgstr "mercredi"msgid "Thursday"msgstr "jeudi"msgid "Friday"msgstr "vendredi"msgid "Saturday"msgstr "samedi"msgid "Sunday"msgstr "dimanche"EOF;  }  /**   * Helper function that returns a bad .po file.   */  function getBadPoFile() {    return <<< EOFmsgid ""msgstr """Project-Id-Version: Drupal 7\\n""MIME-Version: 1.0\\n""Content-Type: text/plain; charset=UTF-8\\n""Content-Transfer-Encoding: 8bit\\n""Plural-Forms: nplurals=2; plural=(n > 1);\\n"msgid "Save configuration"msgstr "Enregistrer la configuration"msgid "edit"msgstr "modifier<img SRC="javascript:alert(\'xss\');">"msgid "delete"msgstr "supprimer<script>alert('xss');</script>"EOF;  }  /**   * Helper function that returns a proper .po file, for testing overwriting   * existing translations.   */  function getOverwritePoFile() {    return <<< EOFmsgid ""msgstr """Project-Id-Version: Drupal 7\\n""MIME-Version: 1.0\\n""Content-Type: text/plain; charset=UTF-8\\n""Content-Transfer-Encoding: 8bit\\n""Plural-Forms: nplurals=3; plural=n%10==1 && n%100!=11 ? 0 : n%10>=2 && n%10<=4 && (n%100<10 || n%100>=20) ? 1 : 2;\\n"msgid "Monday"msgstr "Montag"msgid "Day"msgstr "Jour"EOF;  }  /**   * Helper function that returns a .po file with context.   */  function getPoFileWithContext() {    // Croatian (code hr) is one of the languages that have a different    // form for the full name and the abbreviated name for the month May.    return <<< EOFmsgid ""msgstr """Project-Id-Version: Drupal 7\\n""MIME-Version: 1.0\\n""Content-Type: text/plain; charset=UTF-8\\n""Content-Transfer-Encoding: 8bit\\n""Plural-Forms: nplurals=3; plural=n%10==1 && n%100!=11 ? 0 : n%10>=2 && n%10<=4 && (n%100<10 || n%100>=20) ? 1 : 2;\\n"msgctxt "Long month name"msgid "May"msgstr "Svibanj"msgid "May"msgstr "Svi."EOF;  }  /**   * Helper function that returns a .po file with an empty last item.   */  function getPoFileWithEmptyMsgstr() {    return <<< EOFmsgid ""msgstr """Project-Id-Version: Drupal 7\\n""MIME-Version: 1.0\\n""Content-Type: text/plain; charset=UTF-8\\n""Content-Transfer-Encoding: 8bit\\n""Plural-Forms: nplurals=2; plural=(n > 1);\\n"msgid "Operations"msgstr ""EOF;  }  /**   * Helper function that returns a .po file with an empty last item.   */  function getPoFileWithMsgstr() {    return <<< EOFmsgid ""msgstr """Project-Id-Version: Drupal 7\\n""MIME-Version: 1.0\\n""Content-Type: text/plain; charset=UTF-8\\n""Content-Transfer-Encoding: 8bit\\n""Plural-Forms: nplurals=2; plural=(n > 1);\\n"msgid "Operations"msgstr "Műveletek"msgid "Will not appear in Drupal core, so we can ensure the test passes"msgstr ""EOF;  }  /**   * Returns a .po file with a missing plural formula.   */  function getPoFileWithMissingPlural() {    return <<< EOFmsgid ""msgstr """Project-Id-Version: Drupal 7\\n""MIME-Version: 1.0\\n""Content-Type: text/plain; charset=UTF-8\\n""Content-Transfer-Encoding: 8bit\\n"msgid "Monday"msgstr "Ponedjeljak"EOF;  }  /**   * Returns a .po file with a broken plural formula.   */  function getPoFileWithBrokenPlural() {    return <<< EOFmsgid ""msgstr """Project-Id-Version: Drupal 7\\n""MIME-Version: 1.0\\n""Content-Type: text/plain; charset=UTF-8\\n""Content-Transfer-Encoding: 8bit\\n""Plural-Forms: broken, will not parse\\n"msgid "Monday"msgstr "lundi"EOF;  }}/** * Functional tests for the export of translation files. */class LocaleExportFunctionalTest extends DrupalWebTestCase {  public static function getInfo() {    return array(      'name' => 'Translation export',      'description' => 'Tests the exportation of locale files.',      'group' => 'Locale',    );  }  /**   * A user able to create languages and export translations.   */  protected $admin_user = NULL;  function setUp() {    parent::setUp('locale', 'locale_test');    $this->admin_user = $this->drupalCreateUser(array('administer languages', 'translate interface', 'access administration pages'));    $this->drupalLogin($this->admin_user);  }  /**   * Test exportation of translations.   */  function testExportTranslation() {    // First import some known translations.    // This will also automatically enable the 'fr' language.    $name = tempnam('temporary://', "po_") . '.po';    file_put_contents($name, $this->getPoFile());    $this->drupalPost('admin/config/regional/translate/import', array(      'langcode' => 'fr',      'files[file]' => $name,    ), t('Import'));    drupal_unlink($name);    // Get the French translations.    $this->drupalPost('admin/config/regional/translate/export', array(      'langcode' => 'fr',    ), t('Export'));    // Ensure we have a translation file.    $this->assertRaw('# French translation of Drupal', 'Exported French translation file.');    // Ensure our imported translations exist in the file.    $this->assertRaw('msgstr "lundi"', 'French translations present in exported file.');  }  /**   * Test exportation of translation template file.   */  function testExportTranslationTemplateFile() {    // Get the translation template file.    // There are two 'Export' buttons on this page, but it somehow works.  It'd    // be better if we could use the submit button id like documented but that    // doesn't work.    $this->drupalPost('admin/config/regional/translate/export', array(), t('Export'));    // Ensure we have a translation file.    $this->assertRaw('# LANGUAGE translation of PROJECT', 'Exported translation template file.');  }  /**   * Helper function that returns a proper .po file.   */  function getPoFile() {    return <<< EOFmsgid ""msgstr """Project-Id-Version: Drupal 6\\n""MIME-Version: 1.0\\n""Content-Type: text/plain; charset=UTF-8\\n""Content-Transfer-Encoding: 8bit\\n""Plural-Forms: nplurals=2; plural=(n > 1);\\n"msgid "Monday"msgstr "lundi"EOF;  }}/** * Tests for the st() function. */class LocaleInstallTest extends DrupalWebTestCase {  public static function getInfo() {    return array(      'name' => 'String translation using st()',      'description' => 'Tests that st() works like t().',      'group' => 'Locale',    );  }  function setUp() {    parent::setUp('locale');    // st() lives in install.inc, so ensure that it is loaded for all tests.    require_once DRUPAL_ROOT . '/includes/install.inc';  }  /**   * Verify that function signatures of t() and st() are equal.   */  function testFunctionSignatures() {    $reflector_t = new ReflectionFunction('t');    $reflector_st = new ReflectionFunction('st');    $this->assertEqual($reflector_t->getParameters(), $reflector_st->getParameters(), 'Function signatures of t() and st() are equal.');  }}/** * Locale uninstall with English UI functional test. */class LocaleUninstallFunctionalTest extends DrupalWebTestCase {  public static function getInfo() {    return array(      'name' => 'Locale uninstall (EN)',      'description' => 'Tests the uninstall process using the built-in UI language.',      'group' => 'Locale',    );  }  /**   * The default language set for the UI before uninstall.   */  protected $language;  function setUp() {    parent::setUp('locale');    $this->language = 'en';  }  /**   * Check if the values of the Locale variables are correct after uninstall.   */  function testUninstallProcess() {    $locale_module = array('locale');    // Add a new language and optionally set it as default.    require_once DRUPAL_ROOT . '/includes/locale.inc';    locale_add_language('fr', 'French', 'Français', LANGUAGE_LTR, '', '', TRUE, $this->language == 'fr');    // Check the UI language.    drupal_language_initialize();    global $language;    $this->assertEqual($language->language, $this->language, format_string('Current language: %lang', array('%lang' => $language->language)));    // Enable multilingual workflow option for articles.    variable_set('language_content_type_article', 1);    // Change JavaScript translations directory.    variable_set('locale_js_directory', 'js_translations');    // Build the JavaScript translation file for French.    $user = $this->drupalCreateUser(array('translate interface', 'access administration pages'));    $this->drupalLogin($user);    $this->drupalGet('admin/config/regional/translate/translate');    $string = db_query('SELECT min(lid) AS lid FROM {locales_source} WHERE location LIKE :location AND textgroup = :textgroup', array(      ':location' => '%.js%',      ':textgroup' => 'default',    ))->fetchObject();    $edit = array('translations[fr]' => 'french translation');    $this->drupalPost('admin/config/regional/translate/edit/' . $string->lid, $edit, t('Save translations'));    _locale_rebuild_js('fr');    $file = db_query('SELECT javascript FROM {languages} WHERE language = :language', array(':language' => 'fr'))->fetchObject();    $js_file = 'public://' . variable_get('locale_js_directory', 'languages') . '/fr_' . $file->javascript . '.js';    $this->assertTrue($result = file_exists($js_file), format_string('JavaScript file created: %file', array('%file' => $result ? $js_file : 'none')));    // Disable string caching.    variable_set('locale_cache_strings', 0);    // Change language negotiation options.    drupal_load('module', 'locale');    variable_set('language_types', drupal_language_types() + array('language_custom' => TRUE));    variable_set('language_negotiation_' . LANGUAGE_TYPE_INTERFACE, locale_language_negotiation_info());    variable_set('language_negotiation_' . LANGUAGE_TYPE_CONTENT, locale_language_negotiation_info());    variable_set('language_negotiation_' . LANGUAGE_TYPE_URL, locale_language_negotiation_info());    // Change language providers settings.    variable_set('locale_language_negotiation_url_part', LOCALE_LANGUAGE_NEGOTIATION_URL_PREFIX);    variable_set('locale_language_negotiation_session_param', TRUE);    // Uninstall Locale.    module_disable($locale_module);    drupal_uninstall_modules($locale_module);    // Visit the front page.    $this->drupalGet('');    // Check the init language logic.    drupal_language_initialize();    $this->assertEqual($language->language, 'en', format_string('Language after uninstall: %lang', array('%lang' => $language->language)));    // Check JavaScript files deletion.    $this->assertTrue($result = !file_exists($js_file), format_string('JavaScript file deleted: %file', array('%file' => $result ? $js_file : 'found')));    // Check language count.    $language_count = variable_get('language_count', 1);    $this->assertEqual($language_count, 1, format_string('Language count: %count', array('%count' => $language_count)));    // Check language negotiation.    require_once DRUPAL_ROOT . '/includes/language.inc';    $this->assertTrue(count(language_types()) == count(drupal_language_types()), 'Language types reset');    $language_negotiation = language_negotiation_get(LANGUAGE_TYPE_INTERFACE) == LANGUAGE_NEGOTIATION_DEFAULT;    $this->assertTrue($language_negotiation, format_string('Interface language negotiation: %setting', array('%setting' => $language_negotiation ? 'none' : 'set')));    $language_negotiation = language_negotiation_get(LANGUAGE_TYPE_CONTENT) == LANGUAGE_NEGOTIATION_DEFAULT;    $this->assertTrue($language_negotiation, format_string('Content language negotiation: %setting', array('%setting' => $language_negotiation ? 'none' : 'set')));    $language_negotiation = language_negotiation_get(LANGUAGE_TYPE_URL) == LANGUAGE_NEGOTIATION_DEFAULT;    $this->assertTrue($language_negotiation, format_string('URL language negotiation: %setting', array('%setting' => $language_negotiation ? 'none' : 'set')));    // Check language providers settings.    $this->assertFalse(variable_get('locale_language_negotiation_url_part', FALSE), 'URL language provider indicator settings cleared.');    $this->assertFalse(variable_get('locale_language_negotiation_session_param', FALSE), 'Visit language provider settings cleared.');    // Check JavaScript parsed.    $javascript_parsed_count = count(variable_get('javascript_parsed', array()));    $this->assertEqual($javascript_parsed_count, 0, format_string('JavaScript parsed count: %count', array('%count' => $javascript_parsed_count)));    // Check multilingual workflow option for articles.    $multilingual = variable_get('language_content_type_article', 0);    $this->assertEqual($multilingual, 0, format_string('Multilingual workflow option: %status', array('%status' => $multilingual ? 'enabled': 'disabled')));    // Check JavaScript translations directory.    $locale_js_directory = variable_get('locale_js_directory', 'languages');    $this->assertEqual($locale_js_directory, 'languages', format_string('JavaScript translations directory: %dir', array('%dir' => $locale_js_directory)));    // Check string caching.    $locale_cache_strings = variable_get('locale_cache_strings', 1);    $this->assertEqual($locale_cache_strings, 1, format_string('String caching: %status', array('%status' => $locale_cache_strings ? 'enabled': 'disabled')));  }}/** * Locale uninstall with French UI functional test. * * Because this class extends LocaleUninstallFunctionalTest, it doesn't require a new * test of its own. Rather, it switches the default UI language in setUp and then * runs the testUninstallProcess (which it inherits from LocaleUninstallFunctionalTest) * to test with this new language. */class LocaleUninstallFrenchFunctionalTest extends LocaleUninstallFunctionalTest {  public static function getInfo() {    return array(      'name' => 'Locale uninstall (FR)',      'description' => 'Tests the uninstall process using French as interface language.',      'group' => 'Locale',    );  }  function setUp() {    parent::setUp();    $this->language = 'fr';  }}/** * Functional tests for the language switching feature. */class LocaleLanguageSwitchingFunctionalTest extends DrupalWebTestCase {  public static function getInfo() {    return array(      'name' => 'Language switching',      'description' => 'Tests for the language switching feature.',      'group' => 'Locale',    );  }  function setUp() {    parent::setUp('locale');    // Create and login user.    $admin_user = $this->drupalCreateUser(array('administer blocks', 'administer languages', 'translate interface', 'access administration pages'));    $this->drupalLogin($admin_user);  }  /**   * Functional tests for the language switcher block.   */  function testLanguageBlock() {    // Enable the language switching block.    $language_type = LANGUAGE_TYPE_INTERFACE;    $edit = array(      "blocks[locale_{$language_type}][region]" => 'sidebar_first',    );    $this->drupalPost('admin/structure/block', $edit, t('Save blocks'));    // Add language.    $edit = array(      'langcode' => 'fr',    );    $this->drupalPost('admin/config/regional/language/add', $edit, t('Add language'));    // Enable URL language detection and selection.    $edit = array('language[enabled][locale-url]' => '1');    $this->drupalPost('admin/config/regional/language/configure', $edit, t('Save settings'));    // Assert that the language switching block is displayed on the frontpage.    $this->drupalGet('');    $this->assertText(t('Languages'), 'Language switcher block found.');    // Assert that only the current language is marked as active.    list($language_switcher) = $this->xpath('//div[@id=:id]/div[@class="content"]', array(':id' => 'block-locale-' . $language_type));    $links = array(      'active' => array(),      'inactive' => array(),    );    $anchors = array(      'active' => array(),      'inactive' => array(),    );    foreach ($language_switcher->ul->li as $link) {      $classes = explode(" ", (string) $link['class']);      list($language) = array_intersect($classes, array('en', 'fr'));      if (in_array('active', $classes)) {        $links['active'][] = $language;      }      else {        $links['inactive'][] = $language;      }      $anchor_classes = explode(" ", (string) $link->a['class']);      if (in_array('active', $anchor_classes)) {        $anchors['active'][] = $language;      }      else {        $anchors['inactive'][] = $language;      }    }    $this->assertIdentical($links, array('active' => array('en'), 'inactive' => array('fr')), 'Only the current language list item is marked as active on the language switcher block.');    $this->assertIdentical($anchors, array('active' => array('en'), 'inactive' => array('fr')), 'Only the current language anchor is marked as active on the language switcher block.');  }}/** * Test browser language detection. */class LocaleBrowserDetectionTest extends DrupalUnitTestCase {  public static function getInfo() {    return array(      'name' => 'Browser language detection',      'description' => 'Tests for the browser language detection.',      'group' => 'Locale',    );  }  /**   * Unit tests for the locale_language_from_browser() function.   */  function testLanguageFromBrowser() {    // Load the required functions.    require_once DRUPAL_ROOT . '/includes/locale.inc';    $languages = array(      // In our test case, 'en' has priority over 'en-US'.      'en' => (object) array(        'language' => 'en',      ),      'en-US' => (object) array(        'language' => 'en-US',      ),      // But 'fr-CA' has priority over 'fr'.      'fr-CA' => (object) array(        'language' => 'fr-CA',      ),      'fr' => (object) array(        'language' => 'fr',      ),      // 'es-MX' is alone.      'es-MX' => (object) array(        'language' => 'es-MX',      ),      // 'pt' is alone.      'pt' => (object) array(        'language' => 'pt',      ),      // Language codes with more then one dash are actually valid.      // eh-oh-laa-laa is the official language code of the Teletubbies.      'eh-oh-laa-laa' => (object) array(        'language' => 'eh-oh-laa-laa',      ),    );    $test_cases = array(      // Equal qvalue for each language, choose the site preferred one.      'en,en-US,fr-CA,fr,es-MX' => 'en',      'en-US,en,fr-CA,fr,es-MX' => 'en',      'fr,en' => 'en',      'en,fr' => 'en',      'en-US,fr' => 'en',      'fr,en-US' => 'en',      'fr,fr-CA' => 'fr-CA',      'fr-CA,fr' => 'fr-CA',      'fr' => 'fr-CA',      'fr;q=1' => 'fr-CA',      'fr,es-MX' => 'fr-CA',      'fr,es' => 'fr-CA',      'es,fr' => 'fr-CA',      'es-MX,de' => 'es-MX',      'de,es-MX' => 'es-MX',      // Different cases and whitespace.      'en' => 'en',      'En' => 'en',      'EN' => 'en',      ' en' => 'en',      'en ' => 'en',      'en, fr' => 'en',      // A less specific language from the browser matches a more specific one      // from the website, and the other way around for compatibility with      // some versions of Internet Explorer.      'es' => 'es-MX',      'es-MX' => 'es-MX',      'pt' => 'pt',      'pt-PT' => 'pt',      'pt-PT;q=0.5,pt-BR;q=1,en;q=0.7' => 'en',      'pt-PT;q=1,pt-BR;q=0.5,en;q=0.7' => 'en',      'pt-PT;q=0.4,pt-BR;q=0.1,en;q=0.7' => 'en',      'pt-PT;q=0.1,pt-BR;q=0.4,en;q=0.7' => 'en',      // Language code with several dashes are valid. The less specific language      // from the browser matches the more specific one from the website.      'eh-oh-laa-laa' => 'eh-oh-laa-laa',      'eh-oh-laa' => 'eh-oh-laa-laa',      'eh-oh' => 'eh-oh-laa-laa',      'eh' => 'eh-oh-laa-laa',      // Different qvalues.      'fr,en;q=0.5' => 'fr-CA',      'fr,en;q=0.5,fr-CA;q=0.25' => 'fr',      // Silly wildcards are also valid.      '*,fr-CA;q=0.5' => 'en',      '*,en;q=0.25' => 'fr-CA',      'en,en-US;q=0.5,fr;q=0.25' => 'en',      'en-US,en;q=0.5,fr;q=0.25' => 'en-US',      // Unresolvable cases.      '' => FALSE,      'de,pl' => FALSE,      'iecRswK4eh' => FALSE,      $this->randomName(10) => FALSE,    );    foreach ($test_cases as $accept_language => $expected_result) {      $_SERVER['HTTP_ACCEPT_LANGUAGE'] = $accept_language;      $result = locale_language_from_browser($languages);      $this->assertIdentical($result, $expected_result, format_string("Language selection '@accept-language' selects '@result', result = '@actual'", array('@accept-language' => $accept_language, '@result' => $expected_result, '@actual' => isset($result) ? $result : 'none')));    }  }}/** * Functional tests for a user's ability to change their default language. */class LocaleUserLanguageFunctionalTest extends DrupalWebTestCase {  public static function getInfo() {    return array(      'name' => 'User language settings',      'description' => "Tests user's ability to change their default language.",      'group' => 'Locale',    );  }  function setUp() {    parent::setUp('locale');  }  /**   * Test if user can change their default language.   */  function testUserLanguageConfiguration() {    global $base_url;    // User to add and remove language.    $admin_user = $this->drupalCreateUser(array('administer languages', 'access administration pages'));    // User to change their default language.    $web_user = $this->drupalCreateUser();    // Add custom language.    $this->drupalLogin($admin_user);    // Code for the language.    $langcode = 'xx';    // The English name for the language.    $name = $this->randomName(16);    // The native name for the language.    $native = $this->randomName(16);    // The domain prefix.    $prefix = 'xx';    $edit = array(      'langcode' => $langcode,      'name' => $name,      'native' => $native,      'prefix' => $prefix,      'direction' => '0',    );    $this->drupalPost('admin/config/regional/language/add', $edit, t('Add custom language'));    // Add custom language and disable it.    // Code for the language.    $langcode_disabled = 'xx-yy';    // The English name for the language. This will be translated.    $name_disabled = $this->randomName(16);    // The native name for the language.    $native_disabled = $this->randomName(16);    // The domain prefix.    $prefix_disabled = $langcode_disabled;    $edit = array(      'langcode' => $langcode_disabled,      'name' => $name_disabled,      'native' => $native_disabled,      'prefix' => $prefix_disabled,      'direction' => '0',    );    $this->drupalPost('admin/config/regional/language/add', $edit, t('Add custom language'));    // Disable the language.    $edit = array(      'enabled[' . $langcode_disabled . ']' => FALSE,    );    $this->drupalPost('admin/config/regional/language', $edit, t('Save configuration'));    $this->drupalLogout();    // Login as normal user and edit account settings.    $this->drupalLogin($web_user);    $path = 'user/' . $web_user->uid . '/edit';    $this->drupalGet($path);    // Ensure language settings fieldset is available.    $this->assertText(t('Language settings'), 'Language settings available.');    // Ensure custom language is present.    $this->assertText($name, 'Language present on form.');    // Ensure disabled language isn't present.    $this->assertNoText($name_disabled, 'Disabled language not present on form.');    // Switch to our custom language.    $edit = array(      'language' => $langcode,    );    $this->drupalPost($path, $edit, t('Save'));    // Ensure form was submitted successfully.    $this->assertText(t('The changes have been saved.'), 'Changes were saved.');    // Check if language was changed.    $elements = $this->xpath('//input[@id=:id]', array(':id' => 'edit-language-' . $langcode));    $this->assertTrue(isset($elements[0]) && !empty($elements[0]['checked']), 'Default language successfully updated.');    $this->drupalLogout();  }}/** * Functional test for language handling during user creation. */class LocaleUserCreationTest extends DrupalWebTestCase {  public static function getInfo() {    return array(      'name' => 'User creation',      'description' => 'Tests whether proper language is stored for new users and access to language selector.',      'group' => 'Locale',    );  }  function setUp() {    parent::setUp('locale');    variable_set('user_register', USER_REGISTER_VISITORS);  }  /**   * Functional test for language handling during user creation.   */  function testLocalUserCreation() {    // User to add and remove language and create new users.    $admin_user = $this->drupalCreateUser(array('administer languages', 'access administration pages', 'administer users'));    $this->drupalLogin($admin_user);    // Add predefined language.    $langcode = 'fr';    $edit = array(      'langcode' => 'fr',    );    $this->drupalPost('admin/config/regional/language/add', $edit, t('Add language'));    $this->assertText($langcode, 'Language added successfully.');    $this->assertEqual($this->getUrl(), url('admin/config/regional/language', array('absolute' => TRUE)), 'Correct page redirection.');    // Set language negotiation.    $edit = array(      'language[enabled][locale-url]' => TRUE,    );    $this->drupalPost('admin/config/regional/language/configure', $edit, t('Save settings'));    $this->assertText(t('Language negotiation configuration saved.'), 'Set language negotiation.');    // Check if the language selector is available on admin/people/create and    // set to the currently active language.    $this->drupalGet($langcode . '/admin/people/create');    $this->assertFieldChecked("edit-language-$langcode", 'Global language set in the language selector.');    // Create a user with the admin/people/create form and check if the correct    // language is set.    $username = $this->randomName(10);    $edit = array(      'name' => $username,      'mail' => $this->randomName(4) . '@example.com',      'pass[pass1]' => $username,      'pass[pass2]' => $username,    );    $this->drupalPost($langcode . '/admin/people/create', $edit, t('Create new account'));    $user = user_load_by_name($username);    $this->assertEqual($user->language, $langcode, 'New user has correct language set.');    // Register a new user and check if the language selector is hidden.    $this->drupalLogout();    $this->drupalGet($langcode . '/user/register');    $this->assertNoFieldByName('language[fr]', 'Language selector is not accessible.');    $username = $this->randomName(10);    $edit = array(      'name' => $username,      'mail' => $this->randomName(4) . '@example.com',    );    $this->drupalPost($langcode . '/user/register', $edit, t('Create new account'));    $user = user_load_by_name($username);    $this->assertEqual($user->language, $langcode, 'New user has correct language set.');    // Test if the admin can use the language selector and if the    // correct language is was saved.    $user_edit = $langcode . '/user/' . $user->uid . '/edit';    $this->drupalLogin($admin_user);    $this->drupalGet($user_edit);    $this->assertFieldChecked("edit-language-$langcode", 'Language selector is accessible and correct language is selected.');    // Set pass_raw so we can login the new user.    $user->pass_raw = $this->randomName(10);    $edit = array(      'pass[pass1]' => $user->pass_raw,      'pass[pass2]' => $user->pass_raw,    );    $this->drupalPost($user_edit, $edit, t('Save'));    $this->drupalLogin($user);    $this->drupalGet($user_edit);    $this->assertFieldChecked("edit-language-$langcode", 'Language selector is accessible and correct language is selected.');  }}/** * Functional tests for configuring a different path alias per language. */class LocalePathFunctionalTest extends DrupalWebTestCase {  public static function getInfo() {    return array(      'name' => 'Path language settings',      'description' => 'Checks you can configure a language for individual URL aliases.',      'group' => 'Locale',    );  }  function setUp() {    parent::setUp('locale', 'path');  }  /**   * Test if a language can be associated with a path alias.   */  function testPathLanguageConfiguration() {    global $base_url;    // User to add and remove language.    $admin_user = $this->drupalCreateUser(array('administer languages', 'create page content', 'administer url aliases', 'create url aliases', 'access administration pages'));    // Add custom language.    $this->drupalLogin($admin_user);    // Code for the language.    $langcode = 'xx';    // The English name for the language.    $name = $this->randomName(16);    // The native name for the language.    $native = $this->randomName(16);    // The domain prefix.    $prefix = $langcode;    $edit = array(      'langcode' => $langcode,      'name' => $name,      'native' => $native,      'prefix' => $prefix,      'direction' => '0',    );    $this->drupalPost('admin/config/regional/language/add', $edit, t('Add custom language'));    // Check that the "xx" front page is not available when path prefixes are    // not enabled yet.    $this->drupalPost('admin/config/regional/language/configure', array(), t('Save settings'));    $this->drupalGet($prefix);    $this->assertResponse(404, 'The "xx" front page is not available yet.');    // Enable URL language detection and selection.    $edit = array('language[enabled][locale-url]' => 1);    $this->drupalPost('admin/config/regional/language/configure', $edit, t('Save settings'));    // Create a node.    $node = $this->drupalCreateNode(array('type' => 'page'));    // Create a path alias in default language (English).    $path = 'admin/config/search/path/add';    $english_path = $this->randomName(8);    $edit = array(      'source'   => 'node/' . $node->nid,      'alias'    => $english_path,      'language' => 'en',    );    $this->drupalPost($path, $edit, t('Save'));    // Create a path alias in new custom language.    $custom_language_path = $this->randomName(8);    $edit = array(      'source'   => 'node/' . $node->nid,      'alias'    => $custom_language_path,      'language' => $langcode,    );    $this->drupalPost($path, $edit, t('Save'));    // Confirm English language path alias works.    $this->drupalGet($english_path);    $this->assertText($node->title, 'English alias works.');    // Confirm custom language path alias works.    $this->drupalGet($prefix . '/' . $custom_language_path);    $this->assertText($node->title, 'Custom language alias works.');    // Create a custom path.    $custom_path = $this->randomName(8);    // Check priority of language for alias by source path.    $edit = array(      'source'   => 'node/' . $node->nid,      'alias'    => $custom_path,      'language' => LANGUAGE_NONE,    );    path_save($edit);    $lookup_path = drupal_lookup_path('alias', 'node/' . $node->nid, 'en');    $this->assertEqual($english_path, $lookup_path, 'English language alias has priority.');    // Same check for language 'xx'.    $lookup_path = drupal_lookup_path('alias', 'node/' . $node->nid, $prefix);    $this->assertEqual($custom_language_path, $lookup_path, 'Custom language alias has priority.');    path_delete($edit);    // Create language nodes to check priority of aliases.    $first_node = $this->drupalCreateNode(array('type' => 'article', 'promote' => 1));    $second_node = $this->drupalCreateNode(array('type' => 'article', 'promote' => 1));    // Assign a custom path alias to the first node with the English language.    $edit = array(      'source'   => 'node/' . $first_node->nid,      'alias'    => $custom_path,      'language' => 'en',    );    path_save($edit);    // Assign a custom path alias to second node with LANGUAGE_NONE.    $edit = array(      'source'   => 'node/' . $second_node->nid,      'alias'    => $custom_path,      'language' => LANGUAGE_NONE,    );    path_save($edit);    // Test that both node titles link to our path alias.    $this->drupalGet('<front>');    $custom_path_url = base_path() . (variable_get('clean_url', 0) ? $custom_path : '?q=' . $custom_path);    $elements = $this->xpath('//a[@href=:href and .=:title]', array(':href' => $custom_path_url, ':title' => $first_node->title));    $this->assertTrue(!empty($elements), 'First node links to the path alias.');    $elements = $this->xpath('//a[@href=:href and .=:title]', array(':href' => $custom_path_url, ':title' => $second_node->title));    $this->assertTrue(!empty($elements), 'Second node links to the path alias.');    // Confirm that the custom path leads to the first node.    $this->drupalGet($custom_path);    $this->assertText($first_node->title, 'Custom alias returns first node.');    // Confirm that the custom path with prefix leads to the second node.    $this->drupalGet($prefix . '/' . $custom_path);    $this->assertText($second_node->title, 'Custom alias with prefix returns second node.');  }}/** * Functional tests for multilingual support on nodes. */class LocaleContentFunctionalTest extends DrupalWebTestCase {  public static function getInfo() {    return array(      'name' => 'Content language settings',      'description' => 'Checks you can enable multilingual support on content types and configure a language for a node.',      'group' => 'Locale',    );  }  function setUp() {    parent::setUp('locale');  }  /**   * Verifies that machine name fields are always LTR.   */  function testMachineNameLTR() {    // User to add and remove language.    $admin_user = $this->drupalCreateUser(array('administer languages', 'administer content types', 'access administration pages'));    // Log in as admin.    $this->drupalLogin($admin_user);    // Verify that the machine name field is LTR for a new content type.    $this->drupalGet('admin/structure/types/add');    $this->assertFieldByXpath('//input[@name="type" and @dir="ltr"]', NULL, 'The machine name field is LTR when no additional language is configured.');    // Install the Arabic language (which is RTL) and configure as the default.    $edit = array();    $edit['langcode'] = 'ar';    $this->drupalPost('admin/config/regional/language/add', $edit, t('Add language'));    $edit = array();    $edit['site_default'] = 'ar';    $this->drupalPost(NULL, $edit, t('Save configuration'));    // Verify that the machine name field is still LTR for a new content type.    $this->drupalGet('admin/structure/types/add');    $this->assertFieldByXpath('//input[@name="type" and @dir="ltr"]', NULL, 'The machine name field is LTR when the default language is RTL.');  }  /**   * Test if a content type can be set to multilingual and language setting is   * present on node add and edit forms.   */  function testContentTypeLanguageConfiguration() {    global $base_url;    // User to add and remove language.    $admin_user = $this->drupalCreateUser(array('administer languages', 'administer content types', 'access administration pages'));    // User to create a node.    $web_user = $this->drupalCreateUser(array('create article content', 'create page content', 'edit any page content'));    // Add custom language.    $this->drupalLogin($admin_user);    // Code for the language.    $langcode = 'xx';    // The English name for the language.    $name = $this->randomName(16);    // The native name for the language.    $native = $this->randomName(16);    // The domain prefix.    $prefix = $langcode;    $edit = array(      'langcode' => $langcode,      'name' => $name,      'native' => $native,      'prefix' => $prefix,      'direction' => '0',    );    $this->drupalPost('admin/config/regional/language/add', $edit, t('Add custom language'));    // Add disabled custom language.    // Code for the language.    $langcode_disabled = 'xx-yy';    // The English name for the language.    $name_disabled = $this->randomName(16);    // The native name for the language.    $native_disabled = $this->randomName(16);    // The domain prefix.    $prefix_disabled = $langcode_disabled;    $edit = array(      'langcode' => $langcode_disabled,      'name' => $name_disabled,      'native' => $native_disabled,      'prefix' => $prefix_disabled,      'direction' => '0',    );    $this->drupalPost('admin/config/regional/language/add', $edit, t('Add custom language'));    // Disable second custom language.    $path = 'admin/config/regional/language';    $edit = array(      'enabled[' . $langcode_disabled . ']' => FALSE,    );    $this->drupalPost($path, $edit, t('Save configuration'));    // Set "Basic page" content type to use multilingual support.    $this->drupalGet('admin/structure/types/manage/page');    $this->assertText(t('Multilingual support'), 'Multilingual support fieldset present on content type configuration form.');    $edit = array(      'language_content_type' => 1,    );    $this->drupalPost('admin/structure/types/manage/page', $edit, t('Save content type'));    $this->assertRaw(t('The content type %type has been updated.', array('%type' => 'Basic page')), 'Basic page content type has been updated.');    $this->drupalLogout();    // Verify language selection is not present on add article form.    $this->drupalLogin($web_user);    $this->drupalGet('node/add/article');    // Verify language select list is not present.    $this->assertNoFieldByName('language', NULL, 'Language select not present on add article form.');    // Verify language selection appears on add "Basic page" form.    $this->drupalGet('node/add/page');    // Verify language select list is present.    $this->assertFieldByName('language', NULL, 'Language select present on add Basic page form.');    // Ensure enabled language appears.    $this->assertText($name, 'Enabled language present.');    // Ensure disabled language doesn't appear.    $this->assertNoText($name_disabled, 'Disabled language not present.');    // Create "Basic page" content.    $node_title = $this->randomName();    $node_body =  $this->randomName();    $edit = array(      'type' => 'page',      'title' => $node_title,      'body' => array($langcode => array(array('value' => $node_body))),      'language' => $langcode,    );    $node = $this->drupalCreateNode($edit);    // Edit the content and ensure correct language is selected.    $path = 'node/' . $node->nid . '/edit';    $this->drupalGet($path);    $this->assertRaw('<option value="' . $langcode . '" selected="selected">' .  $name . '</option>', 'Correct language selected.');    // Ensure we can change the node language.    $edit = array(      'language' => 'en',    );    $this->drupalPost($path, $edit, t('Save'));    $this->assertRaw(t('%title has been updated.', array('%title' => $node_title)), 'Basic page content updated.');    $this->drupalLogout();  }  /**   * Verifies that nodes may be created with different languages.   */  function testNodeCreationWithLanguage() {    // Create an admin user and log them in.    $perms = array(      // Standard node permissions.      'create page content',      'administer content types',      'administer nodes',      'bypass node access',      // Locale.      'administer languages',    );    $web_user = $this->drupalCreateUser($perms);    $this->drupalLogin($web_user);    // Create some test nodes using different langcodes.    foreach (array(LANGUAGE_NONE, 'en', 'fr') as $langcode) {      $node_args = array(        'type' => 'page',        'promote' => 1,        'language' => $langcode,      );      $node = $this->drupalCreateNode($node_args);      $node_reloaded = node_load($node->nid, NULL, TRUE);      $this->assertEqual($node_reloaded->language, $langcode, format_string('The language code of the node was successfully set to @langcode.', array('@langcode' => $langcode)));    }  }}/** * Test UI language negotiation * 1. URL (PATH) > DEFAULT *    UI Language base on URL prefix, browser language preference has no *    influence: *      admin/config *        UI in site default language *      zh-hans/admin/config *        UI in Chinese *      blah-blah/admin/config *        404 * 2. URL (PATH) > BROWSER > DEFAULT *        admin/config *          UI in user's browser language preference if the site has that *          language enabled, if not, the default language *        zh-hans/admin/config *          UI in Chinese *        blah-blah/admin/config *          404 * 3. URL (DOMAIN) > DEFAULT *        http://example.com/admin/config *          UI language in site default *        http://example.cn/admin/config *          UI language in Chinese */class LocaleUILanguageNegotiationTest extends DrupalWebTestCase {  public static function getInfo() {    return array(      'name' => 'UI language negotiation',      'description' => 'Test UI language switching by URL path prefix and domain.',      'group' => 'Locale',    );  }  function setUp() {    parent::setUp('locale', 'locale_test');    require_once DRUPAL_ROOT . '/includes/language.inc';    drupal_load('module', 'locale');    $admin_user = $this->drupalCreateUser(array('administer languages', 'translate interface', 'access administration pages', 'administer blocks'));    $this->drupalLogin($admin_user);  }  /**   * Tests for language switching by URL path.   */  function testUILanguageNegotiation() {    // A few languages to switch to.    // This one is unknown, should get the default lang version.    $language_unknown = 'blah-blah';    // For testing browser lang preference.    $language_browser_fallback = 'vi';    // For testing path prefix.    $language = 'zh-hans';    // For setting browser language preference to 'vi'.    $http_header_browser_fallback = array("Accept-Language: $language_browser_fallback;q=1");    // For setting browser language preference to some unknown.    $http_header_blah = array("Accept-Language: blah;q=1");    // This domain should switch the UI to Chinese.    $language_domain = 'example.cn';    // Setup the site languages by installing two languages.    require_once DRUPAL_ROOT . '/includes/locale.inc';    locale_add_language($language_browser_fallback);    locale_add_language($language);    // We will look for this string in the admin/config screen to see if the    // corresponding translated string is shown.    $default_string = 'Configure languages for content and the user interface';    // Set the default language in order for the translated string to be registered    // into database when seen by t(). Without doing this, our target string    // is for some reason not found when doing translate search. This might    // be some bug.    drupal_static_reset('language_list');    $languages = language_list('enabled');    variable_set('language_default', $languages[1]['vi']);    // First visit this page to make sure our target string is searchable.    $this->drupalGet('admin/config');    // Now the t()'ed string is in db so switch the language back to default.    variable_del('language_default');    // Translate the string.    $language_browser_fallback_string = "In $language_browser_fallback In $language_browser_fallback In $language_browser_fallback";    $language_string = "In $language In $language In $language";    // Do a translate search of our target string.    $edit = array( 'string' => $default_string);    $this->drupalPost('admin/config/regional/translate/translate', $edit, t('Filter'));    // Should find the string and now click edit to post translated string.    $this->clickLink('edit');    $edit = array(      "translations[$language_browser_fallback]" => $language_browser_fallback_string,      "translations[$language]" => $language_string,    );    $this->drupalPost(NULL, $edit, t('Save translations'));    // Configure URL language rewrite.    variable_set('locale_language_negotiation_url_type', LANGUAGE_TYPE_INTERFACE);    $tests = array(      // Default, browser preference should have no influence.      array(        'language_negotiation' => array(LOCALE_LANGUAGE_NEGOTIATION_URL, LANGUAGE_NEGOTIATION_DEFAULT),        'path' => 'admin/config',        'expect' => $default_string,        'expected_provider' => LANGUAGE_NEGOTIATION_DEFAULT,        'http_header' => $http_header_browser_fallback,        'message' => 'URL (PATH) > DEFAULT: no language prefix, UI language is default and the browser language preference setting is not used.',      ),      // Language prefix.      array(        'language_negotiation' => array(LOCALE_LANGUAGE_NEGOTIATION_URL, LANGUAGE_NEGOTIATION_DEFAULT),        'path' => "$language/admin/config",        'expect' => $language_string,        'expected_provider' => LOCALE_LANGUAGE_NEGOTIATION_URL,        'http_header' => $http_header_browser_fallback,        'message' => 'URL (PATH) > DEFAULT: with language prefix, UI language is switched based on path prefix',      ),      // Default, go by browser preference.      array(        'language_negotiation' => array(LOCALE_LANGUAGE_NEGOTIATION_URL, LOCALE_LANGUAGE_NEGOTIATION_BROWSER),        'path' => 'admin/config',        'expect' => $language_browser_fallback_string,        'expected_provider' => LOCALE_LANGUAGE_NEGOTIATION_BROWSER,        'http_header' => $http_header_browser_fallback,        'message' => 'URL (PATH) > BROWSER: no language prefix, UI language is determined by browser language preference',      ),      // Prefix, switch to the language.      array(        'language_negotiation' => array(LOCALE_LANGUAGE_NEGOTIATION_URL, LOCALE_LANGUAGE_NEGOTIATION_BROWSER),        'path' => "$language/admin/config",        'expect' => $language_string,        'expected_provider' => LOCALE_LANGUAGE_NEGOTIATION_URL,        'http_header' => $http_header_browser_fallback,        'message' => 'URL (PATH) > BROWSER: with langage prefix, UI language is based on path prefix',      ),      // Default, browser language preference is not one of site's lang.      array(        'language_negotiation' => array(LOCALE_LANGUAGE_NEGOTIATION_URL, LOCALE_LANGUAGE_NEGOTIATION_BROWSER, LANGUAGE_NEGOTIATION_DEFAULT),        'path' => 'admin/config',        'expect' => $default_string,        'expected_provider' => LANGUAGE_NEGOTIATION_DEFAULT,        'http_header' => $http_header_blah,        'message' => 'URL (PATH) > BROWSER > DEFAULT: no language prefix and browser language preference set to unknown language should use default language',      ),    );    foreach ($tests as $test) {      $this->runTest($test);    }    // Unknown language prefix should return 404.    variable_set('language_negotiation_' . LANGUAGE_TYPE_INTERFACE, locale_language_negotiation_info());    $this->drupalGet("$language_unknown/admin/config", array(), $http_header_browser_fallback);    $this->assertResponse(404, "Unknown language path prefix should return 404");    // Setup for domain negotiation, first configure the language to have domain    // URL. We use HTTPS and a port to make sure that only the domain name is used.    $edit = array('prefix' => '', 'domain' => "https://$language_domain:99");    $this->drupalPost("admin/config/regional/language/edit/$language", $edit, t('Save language'));    // Set the site to use domain language negotiation.    $tests = array(      // Default domain, browser preference should have no influence.      array(        'language_negotiation' => array(LOCALE_LANGUAGE_NEGOTIATION_URL, LANGUAGE_NEGOTIATION_DEFAULT),        'locale_language_negotiation_url_part' => LOCALE_LANGUAGE_NEGOTIATION_URL_DOMAIN,        'path' => 'admin/config',        'expect' => $default_string,        'expected_provider' => LANGUAGE_NEGOTIATION_DEFAULT,        'http_header' => $http_header_browser_fallback,        'message' => 'URL (DOMAIN) > DEFAULT: default domain should get default language',      ),      // Language domain specific URL, we set the $_SERVER['HTTP_HOST'] in      // locale_test.module hook_boot() to simulate this.      array(        'language_negotiation' => array(LOCALE_LANGUAGE_NEGOTIATION_URL, LANGUAGE_NEGOTIATION_DEFAULT),        'locale_language_negotiation_url_part' => LOCALE_LANGUAGE_NEGOTIATION_URL_DOMAIN,        'locale_test_domain' => $language_domain . ':88',        'path' => 'admin/config',        'expect' => $language_string,        'expected_provider' => LOCALE_LANGUAGE_NEGOTIATION_URL,        'http_header' => $http_header_browser_fallback,        'message' => 'URL (DOMAIN) > DEFAULT: domain example.cn should switch to Chinese',      ),    );    foreach ($tests as $test) {      $this->runTest($test);    }  }  private function runTest($test) {    if (!empty($test['language_negotiation'])) {      $negotiation = array_flip($test['language_negotiation']);      language_negotiation_set(LANGUAGE_TYPE_INTERFACE, $negotiation);    }    if (!empty($test['locale_language_negotiation_url_part'])) {      variable_set('locale_language_negotiation_url_part', $test['locale_language_negotiation_url_part']);    }    if (!empty($test['locale_test_domain'])) {      variable_set('locale_test_domain', $test['locale_test_domain']);    }    $this->drupalGet($test['path'], array(), $test['http_header']);    $this->assertText($test['expect'], $test['message']);    $this->assertText(t('Language negotiation provider: @name', array('@name' => $test['expected_provider'])));  }  /**   * Test URL language detection when the requested URL has no language.   */  function testUrlLanguageFallback() {    // Add the Italian language.    $language_browser_fallback = 'it';    locale_add_language($language_browser_fallback);    $languages = language_list();    // Enable the path prefix for the default language: this way any unprefixed    // URL must have a valid fallback value.    $edit = array('prefix' => 'en');    $this->drupalPost('admin/config/regional/language/edit/en', $edit, t('Save language'));    // Enable browser and URL language detection.    $edit = array(      'language[enabled][locale-browser]' => TRUE,      'language[enabled][locale-url]' => TRUE,      'language[weight][locale-browser]' => -8,      'language[weight][locale-url]' => -10,    );    $this->drupalPost('admin/config/regional/language/configure', $edit, t('Save settings'));    $this->drupalGet('admin/config/regional/language/configure');    // Enable the language switcher block.    $edit = array('blocks[locale_language][region]' => 'sidebar_first');    $this->drupalPost('admin/structure/block', $edit, t('Save blocks'));    // Access the front page without specifying any valid URL language prefix    // and having as browser language preference a non-default language.    $http_header = array("Accept-Language: $language_browser_fallback;q=1");    $this->drupalGet('', array(), $http_header);    // Check that the language switcher active link matches the given browser    // language.    $args = array(':url' => base_path() . (!empty($GLOBALS['conf']['clean_url']) ? $language_browser_fallback : "?q=$language_browser_fallback"));    $fields = $this->xpath('//div[@id="block-locale-language"]//a[@class="language-link active" and @href=:url]', $args);    $this->assertTrue($fields[0] == $languages[$language_browser_fallback]->native, 'The browser language is the URL active language');    // Check that URLs are rewritten using the given browser language.    $fields = $this->xpath('//div[@id="site-name"]//a[@rel="home" and @href=:url]//span', $args);    $this->assertTrue($fields[0] == 'Drupal', 'URLs are rewritten using the browser language.');  }  /**   * Tests url() when separate domains are used for multiple languages.   */  function testLanguageDomain() {    // Add the Italian language, without protocol.    $langcode = 'it';    locale_add_language($langcode, 'Italian', 'Italian', LANGUAGE_LTR, 'it.example.com', '', TRUE, FALSE);    // Add the French language, with protocol.    $langcode = 'fr';    locale_add_language($langcode, 'French', 'French', LANGUAGE_LTR, 'http://fr.example.com', '', TRUE, FALSE);    // Enable language URL detection.    $negotiation = array_flip(array(LOCALE_LANGUAGE_NEGOTIATION_URL, LANGUAGE_NEGOTIATION_DEFAULT));    language_negotiation_set(LANGUAGE_TYPE_INTERFACE, $negotiation);    variable_set('locale_language_negotiation_url_part', 1);    global $is_https;    $languages = language_list();    foreach (array('it', 'fr') as $langcode) {      // Build the link we're going to test based on the clean URL setting.      $link = (!empty($GLOBALS['conf']['clean_url'])) ? $langcode . '.example.com/admin' : $langcode . '.example.com/?q=admin';      // Test URL in another language.      // Base path gives problems on the testbot, so $correct_link is hard-coded.      // @see UrlAlterFunctionalTest::assertUrlOutboundAlter (path.test).      $url = url('admin', array('language' => $languages[$langcode]));      $url_scheme = ($is_https) ? 'https://' : 'http://';      $correct_link = $url_scheme . $link;      $this->assertTrue($url == $correct_link, format_string('The url() function returns the right url (@url) in accordance with the chosen language', array('@url' => $url . " == " . $correct_link)));      // Test HTTPS via options.      variable_set('https', TRUE);      $url = url('admin', array('https' => TRUE, 'language' => $languages[$langcode]));      $correct_link = 'https://' . $link;      $this->assertTrue($url == $correct_link, format_string('The url() function returns the right https url (via options) (@url) in accordance with the chosen language', array('@url' => $url . " == " . $correct_link)));      variable_set('https', FALSE);      // Test HTTPS via current URL scheme.      $temp_https = $is_https;      $is_https = TRUE;      $url = url('admin', array('language' => $languages[$langcode]));      $correct_link = 'https://' . $link;      $this->assertTrue($url == $correct_link, format_string('The url() function returns the right url (via current url scheme) (@url) in accordance with the chosen language', array('@url' => $url . " == " . $correct_link)));      $is_https = $temp_https;    }  }}/** * Test that URL rewriting works as expected. */class LocaleUrlRewritingTest extends DrupalWebTestCase {  public static function getInfo() {    return array(      'name' => 'URL rewriting',      'description' => 'Test that URL rewriting works as expected.',      'group' => 'Locale',    );  }  function setUp() {    parent::setUp('locale');    // Create and login user.    $this->web_user = $this->drupalCreateUser(array('administer languages', 'access administration pages'));    $this->drupalLogin($this->web_user);    // Install French language.    $edit = array();    $edit['langcode'] = 'fr';    $this->drupalPost('admin/config/regional/language/add', $edit, t('Add language'));    // Install Italian language.    $edit = array();    $edit['langcode'] = 'it';    $this->drupalPost('admin/config/regional/language/add', $edit, t('Add language'));    // Disable Italian language.    $edit = array('enabled[it]' => FALSE);    $this->drupalPost('admin/config/regional/language', $edit, t('Save configuration'));    // Enable URL language detection and selection.    $edit = array('language[enabled][locale-url]' => 1);    $this->drupalPost('admin/config/regional/language/configure', $edit, t('Save settings'));    // Reset static caching.    drupal_static_reset('language_list');    drupal_static_reset('locale_url_outbound_alter');    drupal_static_reset('locale_language_url_rewrite_url');  }  /**   * Check that disabled or non-installed languages are not considered.   */  function testUrlRewritingEdgeCases() {    // Check URL rewriting with a disabled language.    $languages = language_list();    $this->checkUrl($languages['it'], 'Path language is ignored if language is disabled.', 'URL language negotiation does not work with disabled languages');    // Check URL rewriting with a non-installed language.    $non_existing = language_default();    $non_existing->language = $this->randomName();    $non_existing->prefix = $this->randomName();    $this->checkUrl($non_existing, 'Path language is ignored if language is not installed.', 'URL language negotiation does not work with non-installed languages');  }  /**   * Check URL rewriting for the given language.   *   * The test is performed with a fixed URL (the default front page) to simply   * check that language prefixes are not added to it and that the prefixed URL   * is actually not working.   *   * @param string $language   *   The language prefix, e.g. 'es'.   * @param string $message1   *   Message to display in assertion that language prefixes are not added.   * @param string $message2   *   The message to display confirming prefixed URL is not working.   */  private function checkUrl($language, $message1, $message2) {    $options = array('language' => $language);    $base_path = trim(base_path(), '/');    $rewritten_path = trim(str_replace(array('?q=', $base_path), '', url('node', $options)), '/');    $segments = explode('/', $rewritten_path, 2);    $prefix = $segments[0];    $path = isset($segments[1]) ? $segments[1] : $prefix;    // If the rewritten URL has not a language prefix we pick the right one from    // the language object so we can always check the prefixed URL.    if ($this->assertNotEqual($language->prefix, $prefix, $message1)) {      $prefix = $language->prefix;    }    $this->drupalGet("$prefix/$path");    $this->assertResponse(404, $message2);  }  /**   * Check URL rewriting when using a domain name and a non-standard port.   */  function testDomainNameNegotiationPort() {    $language_domain = 'example.fr';    $edit = array(      'locale_language_negotiation_url_part' => 1,    );    $this->drupalPost('admin/config/regional/language/configure/url', $edit, t('Save configuration'));    $edit = array(      'prefix' => '',      'domain' => $language_domain    );    $this->drupalPost('admin/config/regional/language/edit/fr', $edit, t('Save language'));    // Enable domain configuration.    variable_set('locale_language_negotiation_url_part', LOCALE_LANGUAGE_NEGOTIATION_URL_DOMAIN);    // Reset static caching.    drupal_static_reset('language_list');    drupal_static_reset('language_url_outbound_alter');    drupal_static_reset('language_url_rewrite_url');    // In case index.php is part of the URLs, we need to adapt the asserted    // URLs as well.    $index_php = strpos(url('', array('absolute' => TRUE)), 'index.php') !== FALSE;    // Remember current HTTP_HOST.    $http_host = $_SERVER['HTTP_HOST'];    // Fake a different port.    $_SERVER['HTTP_HOST'] .= ':88';    // Create an absolute French link.    $languages = language_list();    $language = $languages['fr'];    $url = url('', array(      'absolute' => TRUE,      'language' => $language    ));    $expected = 'http://example.fr:88/';    $expected .= $index_php ? 'index.php/' : '';    $this->assertEqual($url, $expected, 'The right port is used.');    // If we set the port explicitly in url(), it should not be overriden.    $url = url('', array(      'absolute' => TRUE,      'language' => $language,      'base_url' => $GLOBALS['base_url'] . ':90',    ));    $expected = 'http://example.fr:90/';    $expected .= $index_php ? 'index.php/' : '';    $this->assertEqual($url, $expected, 'A given port is not overriden.');    // Restore HTTP_HOST.    $_SERVER['HTTP_HOST'] = $http_host;  }}/** * Functional test for multilingual fields. */class LocaleMultilingualFieldsFunctionalTest extends DrupalWebTestCase {  public static function getInfo() {    return array(      'name' => 'Multilingual fields',      'description' => 'Test multilingual support for fields.',      'group' => 'Locale',    );  }  function setUp() {    parent::setUp('locale');    // Setup users.    $admin_user = $this->drupalCreateUser(array('administer languages', 'administer content types', 'access administration pages', 'create page content', 'edit own page content'));    $this->drupalLogin($admin_user);    // Add a new language.    require_once DRUPAL_ROOT . '/includes/locale.inc';    locale_add_language('it', 'Italian', 'Italiano', LANGUAGE_LTR, '', '', TRUE, FALSE);    // Enable URL language detection and selection.    $edit = array('language[enabled][locale-url]' => '1');    $this->drupalPost('admin/config/regional/language/configure', $edit, t('Save settings'));    // Set "Basic page" content type to use multilingual support.    $edit = array(      'language_content_type' => 1,    );    $this->drupalPost('admin/structure/types/manage/page', $edit, t('Save content type'));    $this->assertRaw(t('The content type %type has been updated.', array('%type' => 'Basic page')), 'Basic page content type has been updated.');    // Make node body translatable.    $field = field_info_field('body');    $field['translatable'] = TRUE;    field_update_field($field);  }  /**   * Test if field languages are correctly set through the node form.   */  function testMultilingualNodeForm() {    // Create "Basic page" content.    $langcode = LANGUAGE_NONE;    $title_key = "title";    $title_value = $this->randomName(8);    $body_key = "body[$langcode][0][value]";    $body_value = $this->randomName(16);    // Create node to edit.    $edit = array();    $edit[$title_key] = $title_value;    $edit[$body_key] = $body_value;    $edit['language'] = 'en';    $this->drupalPost('node/add/page', $edit, t('Save'));    // Check that the node exists in the database.    $node = $this->drupalGetNodeByTitle($edit[$title_key]);    $this->assertTrue($node, 'Node found in database.');    $assert = isset($node->body['en']) && !isset($node->body[LANGUAGE_NONE]) && $node->body['en'][0]['value'] == $body_value;    $this->assertTrue($assert, 'Field language correctly set.');    // Change node language.    $this->drupalGet("node/$node->nid/edit");    $edit = array(      $title_key => $this->randomName(8),      'language' => 'it'    );    $this->drupalPost(NULL, $edit, t('Save'));    $node = $this->drupalGetNodeByTitle($edit[$title_key]);    $this->assertTrue($node, 'Node found in database.');    $assert = isset($node->body['it']) && !isset($node->body['en']) && $node->body['it'][0]['value'] == $body_value;    $this->assertTrue($assert, 'Field language correctly changed.');    // Enable content language URL detection.    language_negotiation_set(LANGUAGE_TYPE_CONTENT, array(LOCALE_LANGUAGE_NEGOTIATION_URL => 0));    // Test multilingual field language fallback logic.    $this->drupalGet("it/node/$node->nid");    $this->assertRaw($body_value, 'Body correctly displayed using Italian as requested language');    $this->drupalGet("node/$node->nid");    $this->assertRaw($body_value, 'Body correctly displayed using English as requested language');  }  /*   * Test multilingual field display settings.   */  function testMultilingualDisplaySettings() {    // Create "Basic page" content.    $langcode = LANGUAGE_NONE;    $title_key = "title";    $title_value = $this->randomName(8);    $body_key = "body[$langcode][0][value]";    $body_value = $this->randomName(16);    // Create node to edit.    $edit = array();    $edit[$title_key] = $title_value;    $edit[$body_key] = $body_value;    $edit['language'] = 'en';    $this->drupalPost('node/add/page', $edit, t('Save'));    // Check that the node exists in the database.    $node = $this->drupalGetNodeByTitle($edit[$title_key]);    $this->assertTrue($node, 'Node found in database.');    // Check if node body is showed.    $this->drupalGet("node/$node->nid");    $body = $this->xpath('//div[@id=:id]//div[@property="content:encoded"]/p', array(':id' => 'node-' . $node->nid));    $this->assertEqual(current($body), $node->body['en'][0]['value'], 'Node body is correctly showed.');  }}/** * Functional tests for comment language. */class LocaleCommentLanguageFunctionalTest extends DrupalWebTestCase {  public static function getInfo() {    return array(      'name' => 'Comment language',      'description' => 'Tests for comment language.',      'group' => 'Locale',    );  }  function setUp() {    parent::setUp('locale', 'locale_test');    // Create and login user.    $admin_user = $this->drupalCreateUser(array('administer site configuration', 'administer languages', 'access administration pages', 'administer content types', 'administer comments', 'create article content'));    $this->drupalLogin($admin_user);    // Add language.    $edit = array('langcode' => 'fr');    $this->drupalPost('admin/config/regional/language/add', $edit, t('Add language'));    // Set "Article" content type to use multilingual support.    $edit = array('language_content_type' => 1);    $this->drupalPost('admin/structure/types/manage/article', $edit, t('Save content type'));    // Enable content language negotiation UI.    variable_set('locale_test_content_language_type', TRUE);    // Set interface language detection to user and content language detection    // to URL. Disable inheritance from interface language to ensure content    // language will fall back to the default language if no URL language can be    // detected.    $edit = array(      'language[enabled][locale-user]' => TRUE,      'language_content[enabled][locale-url]' => TRUE,      'language_content[enabled][locale-interface]' => FALSE,    );    $this->drupalPost('admin/config/regional/language/configure', $edit, t('Save settings'));    // Change user language preference, this way interface language is always    // French no matter what path prefix the URLs have.    $edit = array('language' => 'fr');    $this->drupalPost("user/{$admin_user->uid}/edit", $edit, t('Save'));    // Make comment body translatable.    $field = field_info_field('comment_body');    $field['translatable'] = TRUE;    field_update_field($field);    $this->assertTrue(field_is_translatable('comment', $field), 'Comment body is translatable.');  }  /**   * Test that comment language is properly set.   */  function testCommentLanguage() {    drupal_static_reset('language_list');    // Create two nodes, one for english and one for french, and comment each    // node using both english and french as content language by changing URL    // language prefixes. Meanwhile interface language is always French, which    // is the user language preference. This way we can ensure that node    // language and interface language do not influence comment language, as    // only content language has to.    foreach (language_list() as $node_langcode => $node_language) {      $language_none = LANGUAGE_NONE;      // Create "Article" content.      $title = $this->randomName();      $edit = array(        "title" => $title,        "body[$language_none][0][value]" => $this->randomName(),        "language" => $node_langcode,      );      $this->drupalPost("node/add/article", $edit, t('Save'));      $node = $this->drupalGetNodeByTitle($title);      foreach (language_list() as $langcode => $language) {        // Post a comment with content language $langcode.        $prefix = empty($language->prefix) ? '' : $language->prefix . '/';        $comment_values[$node_langcode][$langcode] = $this->randomName();        // Initially field form widgets have no language.        $edit = array(          'subject' => $this->randomName(),          "comment_body[$language_none][0][value]" => $comment_values[$node_langcode][$langcode],        );        $this->drupalPost("{$prefix}node/{$node->nid}", $edit, t('Preview'));        // After the first submit the submitted entity language is taken into        // account.        $edit = array(          'subject' => $edit['subject'],          "comment_body[$langcode][0][value]" => $comment_values[$node_langcode][$langcode],        );        $this->drupalPost(NULL, $edit, t('Save'));        // Check that comment language matches the current content language.        $cid = db_select('comment', 'c')          ->fields('c', array('cid'))          ->condition('nid', $node->nid)          ->orderBy('cid', 'DESC')          ->range(0, 1)          ->execute()          ->fetchField();        $comment = comment_load($cid);        $comment_langcode = entity_language('comment', $comment);        $args = array('%node_language' => $node_langcode, '%comment_language' => $comment_langcode, '%langcode' => $langcode);        $this->assertEqual($comment_langcode, $langcode, format_string('The comment posted with content language %langcode and belonging to the node with language %node_language has language %comment_language', $args));        $this->assertEqual($comment->comment_body[$langcode][0]['value'], $comment_values[$node_langcode][$langcode], 'Comment body correctly stored.');      }    }    // Check that comment bodies appear in the administration UI.    $this->drupalGet('admin/content/comment');    foreach ($comment_values as $node_values) {      foreach ($node_values as $value) {        $this->assertRaw($value);      }    }  }}/** * Functional tests for localizing date formats. */class LocaleDateFormatsFunctionalTest extends DrupalWebTestCase {  public static function getInfo() {    return array(      'name' => 'Localize date formats',      'description' => 'Tests for the localization of date formats.',      'group' => 'Locale',    );  }  function setUp() {    parent::setUp('locale');    // Create and login user.    $admin_user = $this->drupalCreateUser(array('administer site configuration', 'administer languages', 'access administration pages', 'create article content'));    $this->drupalLogin($admin_user);  }  /**   * Functional tests for localizing date formats.   */  function testLocalizeDateFormats() {    // Add language.    $edit = array(      'langcode' => 'fr',    );    $this->drupalPost('admin/config/regional/language/add', $edit, t('Add language'));    // Set language negotiation.    $language_type = LANGUAGE_TYPE_INTERFACE;    $edit = array(      "{$language_type}[enabled][locale-url]" => TRUE,    );    $this->drupalPost('admin/config/regional/language/configure', $edit, t('Save settings'));    // Configure date formats.    $this->drupalGet('admin/config/regional/date-time/locale');    $this->assertText('Français', 'Configured languages appear.');    $edit = array(      'date_format_long' => 'd.m.Y - H:i',      'date_format_medium' => 'd.m.Y - H:i',      'date_format_short' => 'd.m.Y - H:i',    );    $this->drupalPost('admin/config/regional/date-time/locale/fr/edit', $edit, t('Save configuration'));    $this->assertText(t('Configuration saved.'), 'French date formats updated.');    $edit = array(      'date_format_long' => 'j M Y - g:ia',      'date_format_medium' => 'j M Y - g:ia',      'date_format_short' => 'j M Y - g:ia',    );    $this->drupalPost('admin/config/regional/date-time/locale/en/edit', $edit, t('Save configuration'));    $this->assertText(t('Configuration saved.'), 'English date formats updated.');    // Create node content.    $node = $this->drupalCreateNode(array('type' => 'article'));    // Configure format for the node posted date changes with the language.    $this->drupalGet('node/' . $node->nid);    $english_date = format_date($node->created, 'custom', 'j M Y');    $this->assertText($english_date, 'English date format appears');    $this->drupalGet('fr/node/' . $node->nid);    $french_date = format_date($node->created, 'custom', 'd.m.Y');    $this->assertText($french_date, 'French date format appears');  }}/** * Functional test for language types/negotiation info. */class LocaleLanguageNegotiationInfoFunctionalTest extends DrupalWebTestCase {  public static function getInfo() {    return array(      'name' => 'Language negotiation info',      'description' => 'Tests alterations to language types/negotiation info.',      'group' => 'Locale',    );  }  function setUp() {    parent::setUp('locale');    require_once DRUPAL_ROOT .'/includes/language.inc';    $admin_user = $this->drupalCreateUser(array('administer languages', 'access administration pages', 'view the administration theme'));    $this->drupalLogin($admin_user);    $this->drupalPost('admin/config/regional/language/add', array('langcode' => 'it'), t('Add language'));  }  /**   * Tests alterations to language types/negotiation info.   */  function testInfoAlterations() {    // Enable language type/negotiation info alterations.    variable_set('locale_test_language_types', TRUE);    variable_set('locale_test_language_negotiation_info', TRUE);    $this->languageNegotiationUpdate();    // Check that fixed language types are properly configured without the need    // of saving the language negotiation settings.    $this->checkFixedLanguageTypes();    // Make the content language type configurable by updating the language    // negotiation settings with the proper flag enabled.    variable_set('locale_test_content_language_type', TRUE);    $this->languageNegotiationUpdate();    $type = LANGUAGE_TYPE_CONTENT;    $language_types = variable_get('language_types', drupal_language_types());    $this->assertTrue($language_types[$type], 'Content language type is configurable.');    // Enable some core and custom language providers. The test language type is    // supposed to be configurable.    $test_type = 'test_language_type';    $provider = LOCALE_LANGUAGE_NEGOTIATION_INTERFACE;    $test_provider = 'test_language_provider';    $form_field = $type . '[enabled]['. $provider .']';    $edit = array(      $form_field => TRUE,      $type . '[enabled][' . $test_provider . ']' => TRUE,      $test_type . '[enabled][' . $test_provider . ']' => TRUE,    );    $this->drupalPost('admin/config/regional/language/configure', $edit, t('Save settings'));    // Remove the interface language provider by updating the language    // negotiation settings with the proper flag enabled.    variable_set('locale_test_language_negotiation_info_alter', TRUE);    $this->languageNegotiationUpdate();    $negotiation = variable_get("language_negotiation_$type", array());    $this->assertFalse(isset($negotiation[$provider]), 'Interface language provider removed from the stored settings.');    $this->assertNoFieldByXPath("//input[@name=\"$form_field\"]", NULL, 'Interface language provider unavailable.');    // Check that type-specific language providers can be assigned only to the    // corresponding language types.    foreach (language_types_configurable() as $type) {      $form_field = $type . '[enabled][test_language_provider_ts]';      if ($type == $test_type) {        $this->assertFieldByXPath("//input[@name=\"$form_field\"]", NULL, format_string('Type-specific test language provider available for %type.', array('%type' => $type)));      }      else {        $this->assertNoFieldByXPath("//input[@name=\"$form_field\"]", NULL, format_string('Type-specific test language provider unavailable for %type.', array('%type' => $type)));      }    }    // Check language negotiation results.    $this->drupalGet('');    $last = variable_get('locale_test_language_negotiation_last', array());    foreach (language_types() as $type) {      $langcode = $last[$type];      $value = $type == LANGUAGE_TYPE_CONTENT || strpos($type, 'test') !== FALSE ? 'it' : 'en';      $this->assertEqual($langcode, $value, format_string('The negotiated language for %type is %language', array('%type' => $type, '%language' => $langcode)));    }    // Disable locale_test and check that everything is set back to the original    // status.    $this->languageNegotiationUpdate('disable');    // Check that only the core language types are available.    foreach (language_types() as $type) {      $this->assertTrue(strpos($type, 'test') === FALSE, format_string('The %type language is still available', array('%type' => $type)));    }    // Check that fixed language types are properly configured, even those    // previously set to configurable.    $this->checkFixedLanguageTypes();    // Check that unavailable language providers are not present in the    // negotiation settings.    $negotiation = variable_get("language_negotiation_$type", array());    $this->assertFalse(isset($negotiation[$test_provider]), 'The disabled test language provider is not part of the content language negotiation settings.');    // Check that configuration page presents the correct options and settings.    $this->assertNoRaw(t('Test language detection'), 'No test language type configuration available.');    $this->assertNoRaw(t('This is a test language provider'), 'No test language provider available.');  }  /**   * Update language types/negotiation information.   *   * Manually invoke locale_modules_enabled()/locale_modules_disabled() since   * they would not be invoked after enabling/disabling locale_test the first   * time.   */  private function languageNegotiationUpdate($op = 'enable') {    static $last_op = NULL;    $modules = array('locale_test');    // Enable/disable locale_test only if we did not already before.    if ($last_op != $op) {      $function = "module_{$op}";      $function($modules);      // Reset hook implementation cache.      module_implements(NULL, FALSE, TRUE);    }    drupal_static_reset('language_types_info');    drupal_static_reset('language_negotiation_info');    $function = "locale_modules_{$op}d";    if (function_exists($function)) {      $function($modules);    }    $this->drupalGet('admin/config/regional/language/configure');  }  /**   * Check that language negotiation for fixed types matches the stored one.   */  private function checkFixedLanguageTypes() {    drupal_static_reset('language_types_info');    foreach (language_types_info() as $type => $info) {      if (isset($info['fixed'])) {        $negotiation = variable_get("language_negotiation_$type", array());        $equal = count($info['fixed']) == count($negotiation);        while ($equal && list($id) = each($negotiation)) {          list(, $info_id) = each($info['fixed']);          $equal = $info_id == $id;        }        $this->assertTrue($equal, format_string('language negotiation for %type is properly set up', array('%type' => $type)));      }    }  }}/** * Functional tests for CSS alter functions. */class LocaleCSSAlterTest extends DrupalWebTestCase {  public static function getInfo() {    return array(      'name' => 'CSS altering',      'description' => 'Test CSS alter functions.',      'group' => 'Locale',    );  }  function setUp() {    parent::setUp('locale');  }  /**   * Verifies that -rtl.css file is added directly after LTR .css file.   */  function testCSSFilesOrderInRTLMode() {    global $base_url;    // User to add and remove language.    $admin_user = $this->drupalCreateUser(array('administer languages', 'administer content types', 'access administration pages'));    // Log in as admin.    $this->drupalLogin($admin_user);    // Install the Arabic language (which is RTL) and configure as the default.    $edit = array();    $edit['langcode'] = 'ar';    $this->drupalPost('admin/config/regional/language/add', $edit, t('Add language'));    $edit = array();    $edit['site_default'] = 'ar';    $this->drupalPost(NULL, $edit, t('Save configuration'));    // Verify that the -rtl.css file is added directly after LTR file.    $this->drupalGet('');    $query_string = '?' . variable_get('css_js_query_string', '0');    $this->assertRaw('@import url("' . $base_url . '/modules/system/system.base.css' . $query_string . '");' . "\n" . '@import url("' . $base_url . '/modules/system/system.base-rtl.css' . $query_string . '");' . "\n", 'CSS: system.base-rtl.css is added directly after system.base.css.');    $this->assertRaw('@import url("' . $base_url . '/modules/system/system.menus.css' . $query_string . '");' . "\n" . '@import url("' . $base_url . '/modules/system/system.menus-rtl.css' . $query_string . '");' . "\n", 'CSS: system.menus-rtl.css is added directly after system.menus.css.');    $this->assertRaw('@import url("' . $base_url . '/modules/system/system.messages.css' . $query_string . '");' . "\n" . '@import url("' . $base_url . '/modules/system/system.messages-rtl.css' . $query_string . '");' . "\n", 'CSS: system.messages-rtl.css is added directly after system.messages.css.');  }}/** * Tests locale translation safe string handling. */class LocaleStringIsSafeTest extends DrupalWebTestCase {  public static function getInfo() {    return array(      'name' => 'Test if a string is safe',      'description' => 'Tests locale translation safe string handling.',      'group' => 'Locale',    );  }  function setUp() {    parent::setUp('locale');  }  /**   * Tests for locale_string_is_safe().   */  public function testLocaleStringIsSafe() {    // Check a translatable string without HTML.    $string = 'Hello world!';    $result = locale_string_is_safe($string);    $this->assertTrue($result);    // Check a translatable string which includes trustable HTML.    $string = 'Hello <strong>world</strong>!';    $result = locale_string_is_safe($string);    $this->assertTrue($result);    // Check an untranslatable string which includes untrustable HTML (according    // to the locale_string_is_safe() function definition).    $string = 'Hello <img src="world.png" alt="world" />!';    $result = locale_string_is_safe($string);    $this->assertFalse($result);    // Check a translatable string which includes a token in an href attribute.    $string = 'Hi <a href="[current-user:url]">user</a>';    $result = locale_string_is_safe($string);    $this->assertTrue($result);  }}
 |