Merge "Remove old comment and unused setting"
authorjenkins-bot <jenkins-bot@gerrit.wikimedia.org>
Mon, 5 Aug 2019 15:19:01 +0000 (15:19 +0000)
committerGerrit Code Review <gerrit@wikimedia.org>
Mon, 5 Aug 2019 15:19:01 +0000 (15:19 +0000)
223 files changed:
RELEASE-NOTES-1.34
autoload.php
includes/DefaultSettings.php
includes/Revision/RenderedRevision.php
includes/Revision/RevisionRenderer.php
includes/Revision/RevisionStore.php
includes/ServiceWiring.php
includes/SiteStatsInit.php
includes/Storage/DerivedPageDataUpdater.php
includes/Storage/PageEditStash.php
includes/Storage/SqlBlobStore.php
includes/actions/RawAction.php
includes/api/ApiHelp.php
includes/api/ApiMain.php
includes/api/ApiQueryQueryPage.php
includes/api/ApiUpload.php
includes/api/i18n/ar.json
includes/api/i18n/en.json
includes/api/i18n/eu.json
includes/api/i18n/fr.json
includes/api/i18n/he.json
includes/api/i18n/mk.json
includes/api/i18n/pt-br.json
includes/api/i18n/pt.json
includes/api/i18n/uk.json
includes/api/i18n/zh-hant.json
includes/block/AbstractBlock.php
includes/block/DatabaseBlock.php
includes/cache/MessageCache.php
includes/changes/ChangesList.php
includes/content/WikitextContent.php
includes/context/RequestContext.php
includes/db/DatabaseOracle.php
includes/db/MWLBFactory.php
includes/deferred/SearchUpdate.php
includes/diff/SlotDiffRenderer.php
includes/externalstore/ExternalStoreDB.php
includes/htmlform/HTMLForm.php
includes/htmlform/fields/HTMLSelectAndOtherField.php
includes/htmlform/fields/HTMLSelectOrOtherField.php
includes/import/UploadSourceAdapter.php
includes/installer/CliInstaller.php
includes/installer/InstallException.php [new file with mode: 0644]
includes/installer/Installer.php
includes/installer/i18n/ast.json
includes/installer/i18n/ca.json
includes/installer/i18n/he.json
includes/installer/i18n/id.json
includes/installer/i18n/ja.json
includes/installer/i18n/sh.json
includes/installer/i18n/tt-cyrl.json
includes/installer/i18n/zh-hant.json
includes/jobqueue/jobs/AssembleUploadChunksJob.php
includes/libs/lockmanager/DBLockManager.php
includes/libs/objectcache/MediumSpecificBagOStuff.php
includes/libs/objectcache/WANObjectCache.php [deleted file]
includes/libs/objectcache/WANObjectCacheReaper.php [deleted file]
includes/libs/objectcache/wancache/WANObjectCache.php [new file with mode: 0644]
includes/libs/objectcache/wancache/WANObjectCacheReaper.php [new file with mode: 0644]
includes/libs/rdbms/database/DBConnRef.php
includes/libs/rdbms/database/Database.php
includes/libs/rdbms/database/DatabaseMssql.php
includes/libs/rdbms/database/DatabaseMysqlBase.php
includes/libs/rdbms/database/DatabaseMysqli.php
includes/libs/rdbms/database/DatabasePostgres.php
includes/libs/rdbms/database/DatabaseSqlite.php
includes/libs/rdbms/database/IDatabase.php
includes/libs/rdbms/lbfactory/ILBFactory.php
includes/libs/rdbms/lbfactory/LBFactory.php
includes/libs/rdbms/loadbalancer/LoadBalancer.php
includes/media/BitmapHandler.php
includes/media/BmpHandler.php
includes/media/DjVuImage.php
includes/media/ExifBitmapHandler.php
includes/media/MediaHandler.php
includes/media/SVGMetadataExtractor.php
includes/media/SvgHandler.php
includes/media/XCFHandler.php
includes/objectcache/SqlBagOStuff.php
includes/pager/AlphabeticPager.php
includes/pager/Pager.php
includes/parser/CoreParserFunctions.php
includes/parser/Parser.php
includes/parser/ParserOptions.php
includes/parser/ParserOutput.php
includes/preferences/DefaultPreferencesFactory.php
includes/resourceloader/MessageBlobStore.php
includes/resourceloader/ResourceLoader.php
includes/resourceloader/ResourceLoaderContext.php
includes/resourceloader/ResourceLoaderImage.php
includes/resourceloader/ResourceLoaderModule.php
includes/resourceloader/ResourceLoaderStartUpModule.php
includes/resourceloader/ResourceLoaderWikiModule.php
includes/search/BaseSearchResultSet.php [new file with mode: 0644]
includes/search/ISearchResultSet.php
includes/search/SearchDatabase.php
includes/search/SearchEngine.php
includes/search/SearchHighlighter.php
includes/search/SearchMssql.php
includes/search/SearchMySQL.php
includes/search/SearchOracle.php
includes/search/SearchResult.php
includes/search/SearchResultSet.php
includes/search/SearchResultSetTrait.php [new file with mode: 0644]
includes/search/SearchSqlite.php
includes/search/SqlSearchResult.php
includes/skins/Skin.php
includes/specialpage/QueryPage.php
includes/specials/SpecialBlock.php
includes/specials/SpecialChangeContentModel.php
includes/specials/SpecialJavaScriptTest.php
includes/specials/SpecialMovepage.php
includes/specials/SpecialUnlinkAccounts.php
includes/templates/EnhancedChangesListGroup.mustache
includes/upload/UploadBase.php
includes/user/UserGroupMembership.php
includes/widget/SelectWithInputWidget.php
languages/i18n/ar.json
languages/i18n/as.json
languages/i18n/ast.json
languages/i18n/az.json
languages/i18n/bcc.json
languages/i18n/bcl.json
languages/i18n/bn.json
languages/i18n/br.json
languages/i18n/ca.json
languages/i18n/ce.json
languages/i18n/cs.json
languages/i18n/de.json
languages/i18n/en.json
languages/i18n/eo.json
languages/i18n/es.json
languages/i18n/et.json
languages/i18n/eu.json
languages/i18n/exif/nl.json
languages/i18n/exif/sdc.json
languages/i18n/exif/tt-cyrl.json
languages/i18n/fi.json
languages/i18n/fr.json
languages/i18n/haw.json
languages/i18n/he.json
languages/i18n/hu.json
languages/i18n/hy.json
languages/i18n/hyw.json
languages/i18n/id.json
languages/i18n/io.json
languages/i18n/is.json
languages/i18n/it.json
languages/i18n/ja.json
languages/i18n/ko.json
languages/i18n/ku-latn.json
languages/i18n/lb.json
languages/i18n/lij.json
languages/i18n/lrc.json
languages/i18n/lt.json
languages/i18n/lv.json
languages/i18n/mk.json
languages/i18n/mni.json
languages/i18n/ms.json
languages/i18n/nb.json
languages/i18n/ne.json
languages/i18n/nl.json
languages/i18n/nqo.json
languages/i18n/pl.json
languages/i18n/pt-br.json
languages/i18n/pt.json
languages/i18n/qqq.json
languages/i18n/roa-tara.json
languages/i18n/ru.json
languages/i18n/rue.json
languages/i18n/sc.json
languages/i18n/sd.json
languages/i18n/sdc.json
languages/i18n/sh.json
languages/i18n/sk.json
languages/i18n/sl.json
languages/i18n/sq.json
languages/i18n/sv.json
languages/i18n/te.json
languages/i18n/tg-cyrl.json
languages/i18n/tt-cyrl.json
maintenance/Maintenance.php
maintenance/includes/BackupDumper.php
maintenance/includes/TextPassDumper.php
maintenance/install.php
maintenance/shell.php
maintenance/sql.php
maintenance/storage/orphanStats.php
maintenance/storage/recompressTracked.php
maintenance/storage/trackBlobs.php
resources/Resources.php
resources/src/mediawiki.apihelp.css [deleted file]
resources/src/mediawiki.apipretty.css [deleted file]
resources/src/mediawiki.apipretty/apihelp.css [new file with mode: 0644]
resources/src/mediawiki.apipretty/apipretty.css [new file with mode: 0644]
resources/src/mediawiki.inspect.js
resources/src/mediawiki.less/mediawiki.mixins.less
resources/src/mediawiki.rcfilters/styles/mw.rcfilters.ui.ChangesListWrapperWidget.less
resources/src/mediawiki.rcfilters/styles/mw.rcfilters.ui.FilterTagMultiselectWidgetMobile.less [new file with mode: 0644]
resources/src/mediawiki.rcfilters/styles/mw.rcfilters.ui.FilterWrapperWidget.less
resources/src/mediawiki.rcfilters/ui/ChangesLimitPopupWidget.js
resources/src/mediawiki.rcfilters/ui/FilterTagMultiselectWidget.js
resources/src/mediawiki.rcfilters/ui/FilterWrapperWidget.js
resources/src/mediawiki.rcfilters/ui/MenuSelectWidget.js
resources/src/mediawiki.special.apisandbox/apisandbox.js
resources/src/mediawiki.widgets/mw.widgets.CopyTextLayout.js
resources/src/mediawiki.widgets/mw.widgets.SelectWithInputWidget.js
resources/src/startup/mediawiki.js
resources/src/startup/startup.js
tests/phpunit/includes/Permissions/PermissionManagerTest.php
tests/phpunit/includes/api/ApiBlockTest.php
tests/phpunit/includes/deferred/SearchUpdateTest.php
tests/phpunit/includes/libs/objectcache/WANObjectCacheTest.php
tests/phpunit/includes/libs/rdbms/database/DatabaseTest.php
tests/phpunit/includes/media/SVGMetadataExtractorTest.php [deleted file]
tests/phpunit/includes/media/SVGReaderTest.php [new file with mode: 0644]
tests/phpunit/includes/parser/ParserOutputTest.php
tests/phpunit/includes/resourceloader/ResourceLoaderContextTest.php
tests/phpunit/includes/resourceloader/ResourceLoaderWikiModuleTest.php
tests/phpunit/includes/search/SearchResultSetTest.php
tests/phpunit/includes/specials/SpecialBlockTest.php
tests/phpunit/structure/ResourcesTest.php
tests/qunit/QUnitTestResources.php

index 9330d10..8ecc469 100644 (file)
@@ -61,6 +61,10 @@ For notes on 1.33.x and older releases, see HISTORY.
   1.23, is now hard-deprecated.
 * $wgProfileOnly — Setting this, deprecated in 1.23, is now hard-deprecated.
   Instead, set the log file in $wgDebugLogGroups['profileoutput'].
+* $wgProxyList — Setting this to an array with IP addresses in the array keys,
+  which was deprecated in 1.30, no longer works. Instead, $wgProxyList should be
+  an array with IP addresses as the values, or a string path to a file
+  containing one IP address per line.
 * …
 
 ==== Removed configuration ====
@@ -72,6 +76,8 @@ For notes on 1.33.x and older releases, see HISTORY.
   configurable via $wgDebugLogFile.
 * $wgPasswordSalt – This setting, used for migrating exceptionally old, insecure
   password setups and deprecated since 1.24, is now removed.
+* $wgDBOracleDRCP - If you must use persistent connections, set DBO_PERSISTENT
+  in the 'flags' field for servers in $wgDBServers (or $wgLBFactoryConf).
 
 === New user-facing features in 1.34 ===
 * Special:Mute has been added as a quick way for users to block unwanted emails
@@ -288,6 +294,8 @@ because of Phabricator reports.
   in JavaScript, use mw.log.deprecate() instead.
 * The 'user.groups' module, deprecated in 1.28, was removed.
   Use the 'user' module instead.
+* The ResourceLoaderContext::expandModuleNames method, deprecated in 1.33, was
+  removed. Use ResourceLoader::expandModuleNames instead.
 * The ability to override User::$mRights has been removed. Use
   PermissionManager::addTemporaryUserRights() instead.
 * Previously, when iterating ResultWrapper with foreach() or a similar
@@ -309,11 +317,24 @@ because of Phabricator reports.
   deprecated since 1.33.
 * The static properties mw.Api.errors and mw.Api.warnings, deprecated in 1.29,
   have been removed.
+* ParserOption::getSpeculativeRevIdCallback(), deprecated in 1.28, has been
+  removed.
 * The UploadVerification hook, deprecated in 1.28, has been removed. Instead,
   use the UploadVerifyFile hook.
 * UploadBase:: and UploadFromChunks::stashFileGetKey() and stashSession(),
   deprecated in 1.28, have been removed. Instead, please use the getFileKey()
   method on the response from doStashFile().
+* LBFactory::setDomainPrefix() and LoadBalancer::setDomainPrefix(), deprecated
+  in 1.33, have been removed. Use setLocalDomainPrefix() instead.
+* IDatabase::implicitGroupby(), deprecated in 1.30, has been removed.
+* IDatabase::doneWrites(), deprecated in 1.31, has been removed.
+  Use IDatabase::lastDoneWrites() instead.
+* Database::reportConnectionError(), deprecated in 1.32, has been removed.
+* LoadBalancer::laggedSlaveUsed(), deprecated in 1.28, has been removed.
+  Use LoadBalancer::laggedReplicaUsed() instead.
+* Database::getProperty(), deprecated in 1.28, has been removed.
+* IDatabase::getWikiId(), deprecated in 1.30, has been removed.
+  Use IDatabase::getDomainID() instead.
 * …
 
 === Deprecations in 1.34 ===
@@ -396,9 +417,24 @@ because of Phabricator reports.
   PermissionManager::getUserPermissions() instead.
 * The LocalisationCacheRecache hook no longer allows purging of message blobs
   to be prevented. Modifying the $purgeBlobs parameter now has no effect.
-* The use of $wgProxyList with IP addresses in the array keys, deprecated in
-  1.30, was removed. Instead, $wgProxyList should be an array with IP addresses
-  as the values, or a string path to a file containing one IP address per line.
+* SVGMetadataExtractor::getMetadata has been deprecated. Instead, you should
+  use SVGReader->getMetadata() directly.
+* The following public properties on AbstractBlock are deprecated: $mReason,
+  $mTimestamp, $mExpiry, $mHideName. Use the getters/setters instead.
+* The following public properties on DatabaseBlock are deprecated: $mAuto,
+  $mParentBlockId. To check for an autoblock use DatabaseBlock::getType; to
+  check for the parent ID, use DatabaseBlock::getParentBlockId.
+* SearchEngine::userHighlightPrefs() is deprecated, simply stop passing
+  $contextlines and $contextchars to the SearchHighlighter methods, they will
+  use proper defaults defined in SearchHighlighter::DEFAULT_CONTEXT_LINES and
+  DEFAULT_CONTEXT_CHARS.
+* SearchUpdate constructor: passing a string as the title param and or a boolean
+  or a string as the content will produce a deprecation warning.
+* SearchEngine::getTextFromContent() is deprecated, use getTextForSearchIndex()
+  directly from the Content object.
+* SearchEngine::textAlreadyUpdatedForIndex() is deprecated, given the
+  deprecation above this method is no longer needed/called and should not be
+  implemented by SearchEngine implementation.
 
 === Other changes in 1.34 ===
 * …
index e6e6504..0208a6d 100644 (file)
@@ -181,6 +181,7 @@ $wgAutoloadLocalClasses = [
        'BadTitleError' => __DIR__ . '/includes/exception/BadTitleError.php',
        'BagOStuff' => __DIR__ . '/includes/libs/objectcache/BagOStuff.php',
        'BaseDump' => __DIR__ . '/includes/export/BaseDump.php',
+       'BaseSearchResultSet' => __DIR__ . '/includes/search/BaseSearchResultSet.php',
        'BaseTemplate' => __DIR__ . '/includes/skins/BaseTemplate.php',
        'BashkirUppercaseCollation' => __DIR__ . '/includes/collation/BashkirUppercaseCollation.php',
        'BatchRowIterator' => __DIR__ . '/includes/utils/BatchRowIterator.php',
@@ -884,6 +885,7 @@ $wgAutoloadLocalClasses = [
        'MediaWiki\\Diff\\WordAccumulator' => __DIR__ . '/includes/diff/WordAccumulator.php',
        'MediaWiki\\HeaderCallback' => __DIR__ . '/includes/HeaderCallback.php',
        'MediaWiki\\Http\\HttpRequestFactory' => __DIR__ . '/includes/http/HttpRequestFactory.php',
+       'MediaWiki\\Installer\\InstallException' => __DIR__ . '/includes/installer/InstallException.php',
        'MediaWiki\\Interwiki\\ClassicInterwikiLookup' => __DIR__ . '/includes/interwiki/ClassicInterwikiLookup.php',
        'MediaWiki\\Interwiki\\InterwikiLookup' => __DIR__ . '/includes/interwiki/InterwikiLookup.php',
        'MediaWiki\\Interwiki\\InterwikiLookupAdapter' => __DIR__ . '/includes/interwiki/InterwikiLookupAdapter.php',
@@ -1321,6 +1323,7 @@ $wgAutoloadLocalClasses = [
        'SearchPostgres' => __DIR__ . '/includes/search/SearchPostgres.php',
        'SearchResult' => __DIR__ . '/includes/search/SearchResult.php',
        'SearchResultSet' => __DIR__ . '/includes/search/SearchResultSet.php',
+       'SearchResultSetTrait' => __DIR__ . '/includes/search/SearchResultSetTrait.php',
        'SearchSqlite' => __DIR__ . '/includes/search/SearchSqlite.php',
        'SearchSuggestion' => __DIR__ . '/includes/search/SearchSuggestion.php',
        'SearchSuggestionSet' => __DIR__ . '/includes/search/SearchSuggestionSet.php',
@@ -1590,8 +1593,8 @@ $wgAutoloadLocalClasses = [
        'VirtualRESTService' => __DIR__ . '/includes/libs/virtualrest/VirtualRESTService.php',
        'VirtualRESTServiceClient' => __DIR__ . '/includes/libs/virtualrest/VirtualRESTServiceClient.php',
        'WANCacheReapUpdate' => __DIR__ . '/includes/deferred/WANCacheReapUpdate.php',
-       'WANObjectCache' => __DIR__ . '/includes/libs/objectcache/WANObjectCache.php',
-       'WANObjectCacheReaper' => __DIR__ . '/includes/libs/objectcache/WANObjectCacheReaper.php',
+       'WANObjectCache' => __DIR__ . '/includes/libs/objectcache/wancache/WANObjectCache.php',
+       'WANObjectCacheReaper' => __DIR__ . '/includes/libs/objectcache/wancache/WANObjectCacheReaper.php',
        'WantedCategoriesPage' => __DIR__ . '/includes/specials/SpecialWantedcategories.php',
        'WantedFilesPage' => __DIR__ . '/includes/specials/SpecialWantedfiles.php',
        'WantedPagesPage' => __DIR__ . '/includes/specials/SpecialWantedpages.php',
index cf8491a..a3772b9 100644 (file)
@@ -738,10 +738,11 @@ $wgUploadDialog = [
  *
  * This is an array of file backend configuration arrays.
  * Each backend configuration has the following parameters:
- *  - 'name'         : A unique name for the backend
- *  - 'class'        : The file backend class to use
- *  - 'wikiId'       : A unique string that identifies the wiki (container prefix)
- *  - 'lockManager'  : The name of a lock manager (see $wgLockManagers)
+ *  - name        : A unique name for the backend
+ *  - class       : The file backend class to use
+ *  - wikiId      : A unique string that identifies the wiki (container prefix)
+ *  - lockManager : The name of a lock manager (see $wgLockManagers) [optional]
+ *  - fileJournal : File journal configuration for FileJournal::__construct() [optional]
  *
  * See FileBackend::__construct() for more details.
  * Additional parameters are specific to the file backend class used.
@@ -774,8 +775,8 @@ $wgFileBackends = [];
 /**
  * Array of configuration arrays for each lock manager.
  * Each backend configuration has the following parameters:
- *  - 'name'        : A unique name for the lock manager
- *  - 'class'       : The lock manger class to use
+ *  - name  : A unique name for the lock manager
+ *  - class : The lock manger class to use
  *
  * See LockManager::__construct() for more details.
  * Additional parameters are specific to the lock manager class used.
@@ -2052,23 +2053,21 @@ $wgSharedSchema = false;
  *                  sent to it. It will be excluded from lag checks in maintenance scripts.
  *                  The only way it can receive traffic is if groupLoads is used.
  *
- *   - groupLoads:  array of load ratios, the key is the query group name. A query may belong
- *                  to several groups, the most specific group defined here is used.
+ *   - groupLoads:  (optional) Array of load ratios, the key is the query group name. A query
+ *                  may belong to several groups, the most specific group defined here is used.
  *
- *   - flags:       bit field
- *                  - DBO_DEFAULT -- turns on DBO_TRX only if "cliMode" is off (recommended)
- *                  - DBO_DEBUG -- equivalent of $wgDebugDumpSql
- *                  - DBO_TRX -- wrap entire request in a transaction
- *                  - DBO_NOBUFFER -- turn off buffering (not useful in LocalSettings.php)
- *                  - DBO_PERSISTENT -- enables persistent database connections
- *                  - DBO_SSL -- uses SSL/TLS encryption in database connections, if available
- *                  - DBO_COMPRESS -- uses internal compression in database connections,
- *                                    if available
+ *   - flags:       (optional) Bit field of properties:
+ *                  - DBO_DEFAULT:    Transactionalize web requests and use autocommit otherwise
+ *                  - DBO_DEBUG:      Equivalent of $wgDebugDumpSql
+ *                  - DBO_SSL:        Use TLS connection encryption if available
+ *                  - DBO_COMPRESS:   Use protocol compression with database connections
+ *                  - DBO_PERSISTENT: Enables persistent database connections
  *
  *   - max lag:     (optional) Maximum replication lag before a replica DB goes out of rotation
  *   - is static:   (optional) Set to true if the dataset is static and no replication is used.
  *   - cliMode:     (optional) Connection handles will not assume that requests are short-lived
  *                  nor that INSERT..SELECT can be rewritten into a buffered SELECT and INSERT.
+ *                  This is what DBO_DEFAULT uses to determine when a web request is present.
  *                  [Default: uses value of $wgCommandLineMode]
  *
  *   These and any other user-defined properties will be assigned to the mLBInfo member
@@ -2138,34 +2137,6 @@ $wgDBerrorLog = false;
  */
 $wgDBerrorLogTZ = false;
 
-/**
- * Set true to enable Oracle DCRP (supported from 11gR1 onward)
- *
- * To use this feature set to true and use a datasource defined as
- * POOLED (i.e. in tnsnames definition set server=pooled in connect_data
- * block).
- *
- * Starting from 11gR1 you can use DCRP (Database Resident Connection
- * Pool) that maintains established sessions and reuses them on new
- * connections.
- *
- * Not completely tested, but it should fall back on normal connection
- * in case the pool is full or the datasource is not configured as
- * pooled.
- * And the other way around; using oci_pconnect on a non pooled
- * datasource should produce a normal connection.
- *
- * When it comes to frequent shortlived DB connections like with MW
- * Oracle tends to s***. The problem is the driver connects to the
- * database reasonably fast, but establishing a session takes time and
- * resources. MW does not rely on session state (as it does not use
- * features such as package variables) so establishing a valid session
- * is in this case an unwanted overhead that just slows things down.
- *
- * @warning EXPERIMENTAL!
- */
-$wgDBOracleDRCP = false;
-
 /**
  * Other wikis on this site, can be administered from a single developer account.
  *
index 818494a..3bc8dda 100644 (file)
@@ -292,6 +292,7 @@ class RenderedRevision implements SlotRenderingProvider {
                $this->setRevisionInternal( $rev );
 
                $this->pruneRevisionSensitiveOutput(
+                       $this->revision->getPageId(),
                        $this->revision->getId(),
                        $this->revision->getTimestamp()
                );
@@ -300,16 +301,25 @@ class RenderedRevision implements SlotRenderingProvider {
        /**
         * Prune any output that depends on the revision ID.
         *
+        * @param int|bool $actualPageId The actual page id, to check the used speculative page ID
+        *        against; false, to not purge on vary-page-id; true, to purge on vary-page-id
+        *        unconditionally.
         * @param int|bool $actualRevId The actual rev id, to check the used speculative rev ID
-        *        against, or false to not purge on vary-revision-id, or true to purge on
+        *        against,; false, to not purge on vary-revision-id; true, to purge on
         *        vary-revision-id unconditionally.
         * @param string|bool $actualRevTimestamp The actual rev timestamp, to check against the
-        *        parser output revision timestamp, or false to not purge on vary-revision-timestamp
+        *        parser output revision timestamp; false, to not purge on vary-revision-timestamp;
+        *        true, to purge on vary-revision-timestamp unconditionally.
         */
-       private function pruneRevisionSensitiveOutput( $actualRevId, $actualRevTimestamp ) {
+       private function pruneRevisionSensitiveOutput(
+               $actualPageId,
+               $actualRevId,
+               $actualRevTimestamp
+       ) {
                if ( $this->revisionOutput ) {
                        if ( $this->outputVariesOnRevisionMetaData(
                                $this->revisionOutput,
+                               $actualPageId,
                                $actualRevId,
                                $actualRevTimestamp
                        ) ) {
@@ -322,6 +332,7 @@ class RenderedRevision implements SlotRenderingProvider {
                foreach ( $this->slotsOutput as $role => $output ) {
                        if ( $this->outputVariesOnRevisionMetaData(
                                $output,
+                               $actualPageId,
                                $actualRevId,
                                $actualRevTimestamp
                        ) ) {
@@ -384,16 +395,20 @@ class RenderedRevision implements SlotRenderingProvider {
 
        /**
         * @param ParserOutput $out
-        * @param int|bool  $actualRevId The actual rev id, to check the used speculative rev ID
-        *        against, false to not purge on vary-revision-id, or true to purge on
+        * @param int|bool $actualPageId The actual page id, to check the used speculative page ID
+        *        against; false, to not purge on vary-page-id; true, to purge on vary-page-id
+        *        unconditionally.
+        * @param int|bool $actualRevId The actual rev id, to check the used speculative rev ID
+        *        against,; false, to not purge on vary-revision-id; true, to purge on
         *        vary-revision-id unconditionally.
         * @param string|bool $actualRevTimestamp The actual rev timestamp, to check against the
-        *        parser output revision timestamp, false to not purge on vary-revision-timestamp,
-        *        or true to purge on vary-revision-timestamp unconditionally.
+        *        parser output revision timestamp; false, to not purge on vary-revision-timestamp;
+        *        true, to purge on vary-revision-timestamp unconditionally.
         * @return bool
         */
        private function outputVariesOnRevisionMetaData(
                ParserOutput $out,
+               $actualPageId,
                $actualRevId,
                $actualRevTimestamp
        ) {
@@ -420,6 +435,13 @@ class RenderedRevision implements SlotRenderingProvider {
                ) {
                        $logger->info( "$varyMsg (vary-revision-timestamp and wrong timestamp)", $context );
                        return true;
+               } elseif (
+                       $out->getFlag( 'vary-page-id' )
+                       && $actualPageId !== false
+                       && ( $actualPageId === true || $out->getSpeculativePageIdUsed() !== $actualPageId )
+               ) {
+                       $logger->info( "$varyMsg (vary-page-id and wrong ID)", $context );
+                       return true;
                } elseif ( $out->getFlag( 'vary-revision-exists' ) ) {
                        // If {{REVISIONID}} resolved to '', it now needs to resolve to '-'.
                        // Note that edit stashing always uses '-', which can be used for both
@@ -432,7 +454,7 @@ class RenderedRevision implements SlotRenderingProvider {
                ) {
                        // If a self-transclusion used the proposed page text, it must match the final
                        // page content after PST transformations and automatically merged edit conflicts
-                       $logger->info( "$varyMsg (vary-revision-sha1 with wrong SHA-1)" );
+                       $logger->info( "$varyMsg (vary-revision-sha1 with wrong SHA-1)", $context );
                        return true;
                }
 
index 99150c1..ca4bb73 100644 (file)
@@ -130,6 +130,9 @@ class RevisionRenderer {
                $options->setSpeculativeRevIdCallback( function () use ( $dbIndex ) {
                        return $this->getSpeculativeRevId( $dbIndex );
                } );
+               $options->setSpeculativePageIdCallback( function () use ( $dbIndex ) {
+                       return $this->getSpeculativePageId( $dbIndex );
+               } );
 
                if ( !$rev->getId() && $rev->getTimestamp() ) {
                        // This is an unsaved revision with an already determined timestamp.
@@ -166,7 +169,8 @@ class RevisionRenderer {
                // HACK: But don't use a fresh connection in unit tests, since it would not have
                // the fake tables. This should be handled by the LoadBalancer!
                $flags = defined( 'MW_PHPUNIT_TEST' ) || $dbIndex === DB_REPLICA
-                       ? 0 : ILoadBalancer::CONN_TRX_AUTOCOMMIT;
+                       ? 0
+                       : ILoadBalancer::CONN_TRX_AUTOCOMMIT;
 
                $db = $this->loadBalancer->getConnectionRef( $dbIndex, [], $this->dbDomain, $flags );
 
@@ -178,6 +182,25 @@ class RevisionRenderer {
                );
        }
 
+       private function getSpeculativePageId( $dbIndex ) {
+               // Use a fresh master connection in order to see the latest data, by avoiding
+               // stale data from REPEATABLE-READ snapshots.
+               // HACK: But don't use a fresh connection in unit tests, since it would not have
+               // the fake tables. This should be handled by the LoadBalancer!
+               $flags = defined( 'MW_PHPUNIT_TEST' ) || $dbIndex === DB_REPLICA
+                       ? 0
+                       : ILoadBalancer::CONN_TRX_AUTOCOMMIT;
+
+               $db = $this->loadBalancer->getConnectionRef( $dbIndex, [], $this->dbDomain, $flags );
+
+               return 1 + (int)$db->selectField(
+                       'page',
+                       'MAX(page_id)',
+                       [],
+                       __METHOD__
+               );
+       }
+
        /**
         * This implements the layout for combining the output of multiple slots.
         *
index 8a4b6dc..fe5b5b9 100644 (file)
@@ -284,7 +284,7 @@ class RevisionStore
         */
        private function getDBConnection( $mode, $groups = [] ) {
                $lb = $this->getDBLoadBalancer();
-               return $lb->getConnection( $mode, $groups, $this->dbDomain );
+               return $lb->getConnectionRef( $mode, $groups, $this->dbDomain );
        }
 
        /**
index b6500a4..9073de1 100644 (file)
@@ -263,6 +263,7 @@ return [
 
        'LocalServerObjectCache' => function ( MediaWikiServices $services ) : BagOStuff {
                $cacheId = \ObjectCache::detectLocalServerCache();
+
                return \ObjectCache::newFromId( $cacheId );
        },
 
@@ -621,9 +622,10 @@ return [
        'SiteStore' => function ( MediaWikiServices $services ) : SiteStore {
                $rawSiteStore = new DBSiteStore( $services->getDBLoadBalancer() );
 
-               // TODO: replace wfGetCache with a CacheFactory service.
-               // TODO: replace wfIsHHVM with a capabilities service.
-               $cache = wfGetCache( wfIsHHVM() ? CACHE_ACCEL : CACHE_ANYTHING );
+               $cache = $services->getLocalServerObjectCache();
+               if ( $cache instanceof EmptyBagOStuff ) {
+                       $cache = ObjectCache::getLocalClusterInstance();
+               }
 
                return new CachingSiteStore( $rawSiteStore, $cache );
        },
index 932e1c3..2252f8f 100644 (file)
@@ -195,8 +195,8 @@ class SiteStatsInit {
         * @return IDatabase
         */
        private static function getDB( $index, $groups = [] ) {
-               $lb = MediaWikiServices::getInstance()->getDBLoadBalancer();
-
-               return $lb->getConnection( $index, $groups );
+               return MediaWikiServices::getInstance()
+                       ->getDBLoadBalancer()
+                       ->getConnectionRef( $index, $groups );
        }
 }
index 5d847b6..68814ef 100644 (file)
@@ -1484,7 +1484,6 @@ class DerivedPageDataUpdater implements IDBAccessObject, LoggerAwareInterface {
 
                $id = $this->getPageId();
                $title = $this->getTitle();
-               $dbKey = $title->getPrefixedDBkey();
                $shortTitle = $title->getDBkey();
 
                if ( !$title->exists() ) {
@@ -1522,7 +1521,7 @@ class DerivedPageDataUpdater implements IDBAccessObject, LoggerAwareInterface {
                // TODO: make search infrastructure aware of slots!
                $mainSlot = $this->revision->getSlot( SlotRecord::MAIN );
                if ( !$mainSlot->isInherited() && !$this->isContentDeleted() ) {
-                       DeferredUpdates::addUpdate( new SearchUpdate( $id, $dbKey, $mainSlot->getContent() ) );
+                       DeferredUpdates::addUpdate( new SearchUpdate( $id, $title, $mainSlot->getContent() ) );
                }
 
                // If this is another user's talk page, update newtalk.
index 4671d99..a0ef07d 100644 (file)
@@ -271,7 +271,7 @@ class PageEditStash {
                        // This can be used for the initial parse, e.g. for filters or doEditContent(),
                        // but a second parse will be triggered in doEditUpdates() no matter what
                        $logger->info(
-                               "Cache for key '{key}' has 'vary-revision'; post-insertion parse inevitable.",
+                               "Cache for key '{key}' has vary-revision; post-insertion parse inevitable.",
                                $context
                        );
                } else {
@@ -281,7 +281,9 @@ class PageEditStash {
                                // Similar to the above if we didn't guess the timestamp correctly
                                'vary-revision-timestamp',
                                // Similar to the above if we didn't guess the content correctly
-                               'vary-revision-sha1'
+                               'vary-revision-sha1',
+                               // Similar to the above if we didn't guess page ID correctly
+                               'vary-page-id'
                        ];
                        foreach ( $flagsMaybeReparse as $flag ) {
                                if ( $editInfo->output->getFlag( $flag ) ) {
index 5260754..d1b688b 100644 (file)
@@ -207,7 +207,7 @@ class SqlBlobStore implements IDBAccessObject, BlobStore {
         */
        private function getDBConnection( $index ) {
                $lb = $this->getDBLoadBalancer();
-               return $lb->getConnection( $index, [], $this->dbDomain );
+               return $lb->getConnectionRef( $index, [], $this->dbDomain );
        }
 
        /**
index f6c4472..abb8ff5 100644 (file)
@@ -269,9 +269,7 @@ class RawAction extends FormlessAction {
         * @return string
         */
        public function getContentType() {
-               // Use getRawVal instead of getVal because we only
-               // need to match against known strings, there is no
-               // storing of localised content or other user input.
+               // Optimisation: Avoid slow getVal(), this isn't user-generated content.
                $ctype = $this->getRequest()->getRawVal( 'ctype' );
 
                if ( $ctype == '' ) {
index cc96f90..988957b 100644 (file)
@@ -100,7 +100,7 @@ class ApiHelp extends ApiBase {
                $out = $context->getOutput();
                $out->addModuleStyles( [
                        'mediawiki.hlist',
-                       'mediawiki.apihelp',
+                       'mediawiki.apipretty',
                ] );
                if ( !empty( $options['toc'] ) ) {
                        $out->addModuleStyles( 'mediawiki.toc.styles' );
index a77136d..554ab6a 100644 (file)
@@ -238,7 +238,8 @@ class ApiMain extends ApiBase {
 
                // Setup uselang. This doesn't use $this->getParameter()
                // because we're not ready to handle errors yet.
-               $uselang = $request->getVal( 'uselang', self::API_DEFAULT_USELANG );
+               // Optimisation: Avoid slow getVal(), this isn't user-generated content.
+               $uselang = $request->getRawVal( 'uselang', self::API_DEFAULT_USELANG );
                if ( $uselang === 'user' ) {
                        // Assume the parent context is going to return the user language
                        // for uselang=user (see T85635).
@@ -257,8 +258,9 @@ class ApiMain extends ApiBase {
 
                // Set up the error formatter. This doesn't use $this->getParameter()
                // because we're not ready to handle errors yet.
-               $errorFormat = $request->getVal( 'errorformat', 'bc' );
-               $errorLangCode = $request->getVal( 'errorlang', 'uselang' );
+               // Optimisation: Avoid slow getVal(), this isn't user-generated content.
+               $errorFormat = $request->getRawVal( 'errorformat', 'bc' );
+               $errorLangCode = $request->getRawVal( 'errorlang', 'uselang' );
                $errorsUseDB = $request->getCheck( 'errorsuselocal' );
                if ( in_array( $errorFormat, [ 'plaintext', 'wikitext', 'html', 'raw', 'none' ], true ) ) {
                        if ( $errorLangCode === 'uselang' ) {
@@ -593,9 +595,13 @@ class ApiMain extends ApiBase {
                // Printer may not be initialized if the extractRequestParams() fails for the main module
                $this->createErrorPrinter();
 
+               // Get desired HTTP code from an ApiUsageException. Don't use codes from other
+               // exception types, as they are unlikely to be intended as an HTTP code.
+               $httpCode = $e instanceof ApiUsageException ? $e->getCode() : 0;
+
                $failed = false;
                try {
-                       $this->printResult( $e->getCode() );
+                       $this->printResult( $httpCode );
                } catch ( ApiUsageException $ex ) {
                        // The error printer itself is failing. Try suppressing its request
                        // parameters and redo.
@@ -617,10 +623,10 @@ class ApiMain extends ApiBase {
                        $this->mPrinter = null;
                        $this->createErrorPrinter();
                        $this->mPrinter->forceDefaultParams();
-                       if ( $e->getCode() ) {
+                       if ( $httpCode ) {
                                $response->statusHeader( 200 ); // Reset in case the fallback doesn't want a non-200
                        }
-                       $this->printResult( $e->getCode() );
+                       $this->printResult( $httpCode );
                }
        }
 
index ea20664..26c17c5 100644 (file)
@@ -122,9 +122,12 @@ class ApiQueryQueryPage extends ApiQueryGeneratorBase {
 
                        $title = Title::makeTitle( $row->namespace, $row->title );
                        if ( is_null( $resultPageSet ) ) {
-                               $data = [ 'value' => $row->value ];
-                               if ( $qp->usesTimestamps() ) {
-                                       $data['timestamp'] = wfTimestamp( TS_ISO_8601, $row->value );
+                               $data = [];
+                               if ( isset( $row->value ) ) {
+                                       $data['value'] = $row->value;
+                                       if ( $qp->usesTimestamps() ) {
+                                               $data['timestamp'] = wfTimestamp( TS_ISO_8601, $row->value );
+                                       }
                                }
                                self::addTitleInfo( $data, $title );
 
index fc41e4e..b15b998 100644 (file)
@@ -658,7 +658,7 @@ class ApiUpload extends ApiBase {
         * @return array
         */
        protected function getApiWarnings() {
-               $warnings = $this->mUpload->checkWarnings();
+               $warnings = UploadBase::makeWarningsSerializable( $this->mUpload->checkWarnings() );
 
                return $this->transformWarnings( $warnings );
        }
@@ -670,9 +670,8 @@ class ApiUpload extends ApiBase {
 
                        if ( isset( $warnings['duplicate'] ) ) {
                                $dupes = [];
-                               /** @var File $dupe */
                                foreach ( $warnings['duplicate'] as $dupe ) {
-                                       $dupes[] = $dupe->getName();
+                                       $dupes[] = $dupe['fileName'];
                                }
                                ApiResult::setIndexedTagName( $dupes, 'duplicate' );
                                $warnings['duplicate'] = $dupes;
@@ -681,27 +680,24 @@ class ApiUpload extends ApiBase {
                        if ( isset( $warnings['exists'] ) ) {
                                $warning = $warnings['exists'];
                                unset( $warnings['exists'] );
-                               /** @var LocalFile $localFile */
                                $localFile = $warning['normalizedFile'] ?? $warning['file'];
-                               $warnings[$warning['warning']] = $localFile->getName();
+                               $warnings[$warning['warning']] = $localFile['fileName'];
                        }
 
                        if ( isset( $warnings['no-change'] ) ) {
-                               /** @var File $file */
                                $file = $warnings['no-change'];
                                unset( $warnings['no-change'] );
 
                                $warnings['nochange'] = [
-                                       'timestamp' => wfTimestamp( TS_ISO_8601, $file->getTimestamp() )
+                                       'timestamp' => wfTimestamp( TS_ISO_8601, $file['timestamp'] )
                                ];
                        }
 
                        if ( isset( $warnings['duplicate-version'] ) ) {
                                $dupes = [];
-                               /** @var File $dupe */
                                foreach ( $warnings['duplicate-version'] as $dupe ) {
                                        $dupes[] = [
-                                               'timestamp' => wfTimestamp( TS_ISO_8601, $dupe->getTimestamp() )
+                                               'timestamp' => wfTimestamp( TS_ISO_8601, $dupe['timestamp'] )
                                        ];
                                }
                                unset( $warnings['duplicate-version'] );
index 70515eb..af97236 100644 (file)
        "apihelp-query+languageinfo-paramvalue-prop-bcp47": "رمز لغة BCP-47.",
        "apihelp-query+languageinfo-paramvalue-prop-dir": "اتجاه كتابة اللغة (إما <code>ltr</code> أو <code>rtl</code>).",
        "apihelp-query+languageinfo-paramvalue-prop-autonym": "الاسم الداخلي للغة، أي: الاسم بتلك اللغة.",
-       "apihelp-query+languageinfo-paramvalue-prop-name": "اسم اللغة ياللغة المحددة بواسطة الوسيط <var>lilang</var>، مع تطبيق احتياطات اللغة إذا لزم الأمر.",
+       "apihelp-query+languageinfo-paramvalue-prop-name": "اسم اللغة ياللغة المحددة بواسطة الوسيط <var>uselang</var>، مع تطبيق احتياطات اللغة إذا لزم الأمر.",
        "apihelp-query+languageinfo-paramvalue-prop-fallbacks": "رموز لغة اللغات الاحتياطية التي تم تكوينها لهذه اللغة، لا يتم تضمين الإرجاع النهائي الضمني إلى \"en\" (ولكن قد ترجع بعض اللغات إلى \"en\" بشكل صريح).",
        "apihelp-query+languageinfo-paramvalue-prop-variants": "رموز اللغات للمتغيرات التي تدعمها هذه اللغة.",
        "apihelp-query+languageinfo-param-code": "رموز اللغات التي يجب إرجاعها، أو <code>*</code> لجميع اللغات.",
index cae7687..6625863 100644 (file)
        "apihelp-query+languageinfo-paramvalue-prop-bcp47": "The BCP-47 language code.",
        "apihelp-query+languageinfo-paramvalue-prop-dir": "The writing direction of the language (either <code>ltr</code> or <code>rtl</code>).",
        "apihelp-query+languageinfo-paramvalue-prop-autonym": "The autonym of the language, that is, the name in that language.",
-       "apihelp-query+languageinfo-paramvalue-prop-name": "The name of the language in the language specified by the <var>lilang</var> parameter, with language fallbacks applied if necessary.",
+       "apihelp-query+languageinfo-paramvalue-prop-name": "The name of the language in the language specified by the <var>uselang</var> parameter, with language fallbacks applied if necessary.",
        "apihelp-query+languageinfo-paramvalue-prop-fallbacks": "The language codes of the fallback languages configured for this language. The implicit final fallback to 'en' is not included (but some languages may fall back to 'en' explicitly).",
        "apihelp-query+languageinfo-paramvalue-prop-variants": "The language codes of the variants supported by this language.",
        "apihelp-query+languageinfo-param-code": "Language codes of the languages that should be returned, or <code>*</code> for all languages.",
index 2f95348..81bd127 100644 (file)
@@ -6,7 +6,8 @@
                        "An13sa",
                        "Gorkaazk",
                        "Mikel Ibaiba",
-                       "Iñaki LL"
+                       "Iñaki LL",
+                       "Xabier Armendaritz"
                ]
        },
        "apihelp-main-param-action": "Zein ekintza burutuko da.",
@@ -37,7 +38,7 @@
        "apihelp-compare-param-torev": "Aldaratzeko bigarren berrikusketa.",
        "apihelp-compare-param-prop": "Hartu beharreko informazio zatiak.",
        "apihelp-compare-paramvalue-prop-diff": "HTML diff-a",
-       "apihelp-compare-paramvalue-prop-diffsize": "HTML diff-aren tamainia, byte-tan",
+       "apihelp-compare-paramvalue-prop-diffsize": "Aldeen HTMLaren tamaina, bytetan",
        "apihelp-compare-paramvalue-prop-size": "\"nondik\" eta \"nora\" berrikuspenen tamaina.",
        "apihelp-compare-example-1": "1. eta 2. berrikusketen arteko \"diff\"-a sortu.",
        "apihelp-createaccount-summary": "Erabiltzaile kontu berria sortu.",
index b04ad1b..591bf31 100644 (file)
        "apihelp-query+languageinfo-paramvalue-prop-bcp47": "Le code de langue BCP-47.",
        "apihelp-query+languageinfo-paramvalue-prop-dir": "La direction d’écriture de la langue (<code>ltr</code> ou <code>rtl</code>).",
        "apihelp-query+languageinfo-paramvalue-prop-autonym": "L’autonyme d’une langue, c’est-à-dire son nom dans cette langue.",
-       "apihelp-query+languageinfo-paramvalue-prop-name": "Le nom de la langue dans la langue spécifiée par le paramètre <var>lilang</var>, avec application des langues de secours si besoin.",
+       "apihelp-query+languageinfo-paramvalue-prop-name": "Le nom de la langue dans la langue spécifiée par le paramètre <var>uselang</var>, avec l'application des langues de repli si besoin.",
        "apihelp-query+languageinfo-paramvalue-prop-fallbacks": "Les codes de langue des langues de secours configurées pour cette langue. Le secours implicite final en 'en' n’est pas inclus (mais certaines langues peuvent avoir 'en' en secours explicitement).",
        "apihelp-query+languageinfo-paramvalue-prop-variants": "Les codes de langue des variantes supportées par cette langue.",
        "apihelp-query+languageinfo-param-code": "Codes de langue des langues qui doivent être renvoyées, ou <code>*</code> pour toutes les langues.",
index d0d5aa2..e2c5fe7 100644 (file)
@@ -16,7 +16,8 @@
                        "שמזן",
                        "Or",
                        "Umherirrender",
-                       "Strayblues"
+                       "Strayblues",
+                       "Steeve815"
                ]
        },
        "apihelp-main-extended-description": "<div class=\"hlist plainlinks api-main-links\">\n* [[mw:Special:MyLanguage/API:Main_page|תיעוד]]\n* [[mw:Special:MyLanguage/API:FAQ|שאלות נפוצות]]\n* [https://lists.wikimedia.org/mailman/listinfo/mediawiki-api רשימת דיוור]\n* [https://lists.wikimedia.org/mailman/listinfo/mediawiki-api-announce הודעות על API]\n* [https://phabricator.wikimedia.org/maniphest/query/GebfyV4uCaLd/#R באגים ובקשות]\n</div>\n<strong>מצב:</strong> ה־API של מדיה־ויקי הוא ממשק ותיק ויציב שנתמך ומשתפר באופן סדיר. למרות שאנחנו משתדלים להימנע מכך, לעתים עלינו לבצע שינויים שעלולים לשבש דברים בפונקציונליות הזו; באפשרותך לעשות מינוי ל[https://lists.wikimedia.org/pipermail/mediawiki-api-announce/ רשימת הדיוור mediawiki-api-announce] כדי לקבל הודעות על עדכונים.\n\n<strong>בקשות שגויות:</strong> כשבקשות שגויות נשלחות ל־API, תישלח כותרת HTTP עם המפתח \"MediaWiki-API-Error\", ואז גם הערך של הכותרת וגם קוד השגיאה יוגדרו לאותו ערך. למידע נוסף, אפשר לעיין בדף [[mw:Special:MyLanguage/API:Errors_and_warnings|API: שגיאות ואזהרות]].\n\n<p class=\"mw-apisandbox-link\"><strong>בדיקה:</strong> לבדיקה קלה יותר של בקשות, אפשר להשתמש ב[[Special:ApiSandbox|ארגז החול של API]].</p>",
        "apihelp-query+langlinks-param-inlanguagecode": "קוד שפה בשביל שמות שפות מתורגמות.",
        "apihelp-query+langlinks-example-simple": "קבלת קישורים בין־לשוניים מהדף <kbd>Main Page</kbd>.",
        "apihelp-query+languageinfo-summary": "מחזירה מידע על שפות זמינות.",
+       "apihelp-query+languageinfo-param-prop": "איזה מידע לקבל עבור כל שפה.",
+       "apihelp-query+languageinfo-paramvalue-prop-dir": "כיוון הכתיבה של השפה (<code>ltr</code> או <code>rtl</code>).",
+       "apihelp-query+languageinfo-example-simple": "קבלת קודי שפה של כל השפות הנתמכות.",
        "apihelp-query+links-summary": "החזרת כל הקישורים מהדפים שצוינו.",
        "apihelp-query+links-param-namespace": "להציג קישורים רק במרחבי השם האלה.",
        "apihelp-query+links-param-limit": "כמה קישורים להחזיר.",
index 9fb5f02..8caea4c 100644 (file)
        "apihelp-query+watchlistraw-param-dir": "Насока на исписот.",
        "apihelp-revisiondelete-param-suppress": "Дали се притајуваат податоци од администраторите на ист начин како и за останатите.",
        "apihelp-revisiondelete-param-tags": "Ознаки за примена врз ставката во дневникот на бришења.",
+       "apihelp-stashedit-param-section": "Број на поднасловот. <kbd>0</kbd> за првиот, <kbd>new</kbd> за нов.",
+       "apihelp-stashedit-param-sectiontitle": "Назив за нов поднаслов.",
+       "apihelp-stashedit-param-text": "Содржина на страницата.",
+       "apihelp-stashedit-param-contentmodel": "Содржински модел на новата содржина.",
+       "apihelp-stashedit-param-contentformat": "Форматот за серијализација на содржината што се користи во вносниот текст.",
+       "apihelp-tag-param-reason": "Причина за промената.",
+       "apihelp-unblock-summary": "Одблокирај корисник.",
+       "apihelp-unblock-param-user": "Корисничко име, IP-адреса или IP-опсег за одблокирање. Не може да се користи заедно со <var>$1id</var> или <var>$1userid</var>.",
+       "apihelp-unblock-param-reason": "Причина за одблокирање.",
+       "apihelp-unblock-param-tags": "Ознаки за примена врз ставката во дневникот на блокирања.",
+       "apihelp-undelete-param-tags": "Ознаки за примена врз ставката во дневникот на бришења.",
+       "apihelp-undelete-param-watchlist": "Безусловно додај или отстрани ја страницата од набљудуваните на тековниот корисник, користете ги нагодувањата или не ги менувајте набљудуваните.",
+       "apihelp-undelete-example-page": "Обнови ја страницата <kbd>Main Page</kbd>.",
+       "apihelp-undelete-example-revisions": "Обнови две преработки на <kbd>Main Page</kbd>.",
        "apihelp-upload-param-filename": "Целно име на податотеката.",
        "apihelp-upload-param-comment": "Коментар при подигање. Се користи и како првичен текст на страницата за нови податотеки ако не е укажано <var>$1text</var>.",
        "apihelp-upload-param-text": "Првичен текст на страницата за нови податотеки.",
        "api-help-param-type-boolean": "Тип: булов ([[Special:ApiHelp/main#main/datatypes|подробно]])",
        "api-help-param-type-timestamp": "Тип: {{PLURAL:$1|1=време и датум|2=список на времиња и датуми}} ([[Special:ApiHelp/main#main/datatypes|допуштени формати]])",
        "api-help-param-type-user": "Тип: {{PLURAL:$1|1=корисничко име|2=список на кориснички имиња}}",
-       "api-help-param-list": "{{PLURAL:$1|1=Ð\95дна Ð²Ñ\80едноÑ\81Ñ\82|2=Ð\92Ñ\80едноÑ\81Ñ\82и (одделеÑ\82е Ð³Ð¸ Ñ\81о <kbd>{{!}}</kbd>)}}: $2",
+       "api-help-param-list": "{{PLURAL:$1|1=Ð\95дна Ð¾Ð´ Ñ\81ледниве Ð²Ñ\80едноÑ\81Ñ\82и|2=Ð\92Ñ\80едноÑ\81Ñ\82и (одделеÑ\82е Ð³Ð¸ Ñ\81о <kbd>{{!}}</kbd> Ð¸Ð»Ð¸ [[Special:ApiHelp/main#main/datatypes|алÑ\82еÑ\80наÑ\82ива]])}}: $2",
        "api-help-param-list-can-be-empty": "{{PLURAL:$1|0=Мора да биде празно|Може да биде празно или $2}}",
        "api-help-param-limit": "Не се допушта повеќе од $1.",
        "api-help-param-limit2": "Не се допушта повеќе од $1 ($2 за ботови).",
        "api-help-param-integer-max": "{{PLURAL:$1|1=Вредноста не може да изнесува|2=Вредностите е може да изнесуваат}} повеќе од $3.",
        "api-help-param-integer-minmax": "{{PLURAL:$1|1=Вредноста мора да изнесува|2=Вредностите мораат да изнесуваат}} помеѓу $2 и $3.",
        "api-help-param-upload": "Мора да биде објавено како податотечно подигање користејќи податоци кои се повеќеделни или од образец.",
-       "api-help-param-multi-separate": "Одделувајте ги вредностите со <kbd>|</kbd>.",
+       "api-help-param-multi-separate": "Одделувајте ги вредностите со <kbd>|</kbd> или [[Special:ApiHelp/main#main/datatypes|алтернатива]].",
        "api-help-param-multi-max": "Највеќе допуштени вредности: {{PLURAL:$1|$1}} ({{PLURAL:$2|$2}} за ботови).",
        "api-help-param-default": "По основно: $1",
        "api-help-param-default-empty": "По основно: <span class=\"apihelp-empty\">(празно)</span>",
        "api-help-permissions": "{{PLURAL:$1|Дозвола|Дозволи}}:",
        "api-help-permissions-granted-to": "{{PLURAL:$1|Доделена на}}: $2",
        "api-help-right-apihighlimits": "Употреба на повисоки ограничувања за приложни барања (бавни барања: $1; брзи барања: $2). Ограничувањата за бавни барања важат и за повеќевредносни параметри.",
+       "apierror-badgenerator-unknown": "Непознат <kbd>generator=$1</kbd>.",
+       "apierror-badquery": "Неважечко барање.",
        "apierror-offline": "Не можев да продолжам поради проблем при поврзувањето со мрежата. Проверете дали сте поврзани со семрежјето и обидете се повторно.",
        "apierror-timeout": "Опслужувачот не одговори во очекуваното време.",
        "api-credits-header": "Признанија",
index af1596d..0ad7687 100644 (file)
        "apihelp-query+languageinfo-paramvalue-prop-bcp47": "O código do idioma BCP-47.",
        "apihelp-query+languageinfo-paramvalue-prop-dir": "A direção de escrita do idioma (<code>ltr</code>, da esquerda para a direita, ou <code>rtl</code>, da direita para a esquerda).",
        "apihelp-query+languageinfo-paramvalue-prop-autonym": "O autônimo do idioma, isto é, o seu nome nesse idioma.",
-       "apihelp-query+languageinfo-paramvalue-prop-name": "O nome do idioma no idioma especificado pelo parâmetro <var>lilang</var>, com a aplicação de idiomas de recurso se necessário.",
+       "apihelp-query+languageinfo-paramvalue-prop-name": "O nome do idioma no idioma especificado pelo parâmetro <var>uselang</var>, com a aplicação de idiomas de recurso se necessário.",
        "apihelp-query+languageinfo-paramvalue-prop-fallbacks": "Os códigos de idioma das idiomas de recurso configuradas para esta língua. O recurso final implícito para 'en' não é incluído (mas algum idiomas podem especificar 'en' como último recurso explicitamente).",
        "apihelp-query+languageinfo-paramvalue-prop-variants": "Os códigos de idioma das variantes suportadas por esse idioma.",
        "apihelp-query+languageinfo-param-code": "Códigos de idioma dos idiomas que devem ser devolvidas, ou <code>*</code> para todos os idiomas.",
index 029a6dc..6c30749 100644 (file)
        "apihelp-query+languageinfo-paramvalue-prop-bcp47": "O código de língua BCP-47.",
        "apihelp-query+languageinfo-paramvalue-prop-dir": "A direção de escrita da língua (<code>ltr</code>, da esquerda para a direita, ou <code>rtl</code>, da direita para a esquerda).",
        "apihelp-query+languageinfo-paramvalue-prop-autonym": "O autónimo da língua, isto é, o seu nome nessa língua.",
-       "apihelp-query+languageinfo-paramvalue-prop-name": "O nome da língua na língua especificada pelo parâmetro <var>lilang</var>, com a aplicação de línguas de recurso se necessário.",
+       "apihelp-query+languageinfo-paramvalue-prop-name": "O nome da língua na língua especificada pelo parâmetro <var>uselang</var>, com a aplicação de línguas de recurso se necessário.",
        "apihelp-query+languageinfo-paramvalue-prop-fallbacks": "Os códigos de língua das línguas de recurso configuradas para esta língua. O recurso final implícito para 'en' não é incluído (mas algumas línguas podem especificar 'en' como último recurso explicitamente).",
        "apihelp-query+languageinfo-paramvalue-prop-variants": "Os códigos de língua das variantes suportadas por esta língua.",
        "apihelp-query+languageinfo-param-code": "Códigos de língua das línguas que devem ser devolvidas, ou <code>*</code> para todas as línguas.",
index 3ca00ee..d62f1ad 100644 (file)
        "api-help-param-templated-var-first": "<var>&#x7B;$1&#x7D;</var> у назві параметра слід замінити значеннями <var>$2</var>",
        "api-help-param-templated-var": "<var>&#x7B;$1&#x7D;</var> — значеннями <var>$2</var>",
        "api-help-datatypes-header": "Типи даних",
-       "api-help-datatypes": "Вхідні дані у MediaWiki мають бути в NFC-нормалізованому UTF-8. MediaWiki може спробувати конвертувати вхідні дані іншого вигляду, але від цього можуть постраждати деякі операції (як [[Special:ApiHelp/edit|редагування]] з перевіркою MD5).\n\nДеякі типи параметрів у запитах API потребують ширшого пояснення:\n;boolean\n:Логічні параметри працюють як галочки HTML: якщо параметр вказано, не залежно від значення, він вважається істинним. Щоб значення було хибним, пропустіть параметр зовсім.\n;timestamp\n:Часові мітки можуть бути вказані у кількох форматах. Рекомендується час і дата в ISO 8601. Усі значення часу в UTC, будь-які часові пояси ігноруються.\n:* Дата і час ISO 8601, <kbd><var>2001</var>-<var>01</var>-<var>15</var>T<var>14</var>:<var>56</var>:<var>00</var>Z</kbd> (пунктуація і <kbd>Z</kbd> необов'язокві)\n:* Дата і час ISO 8601 з (ігнорованими) частками секунди, <kbd><var>2001</var>-<var>01</var>-<var>15</var>T<var>14</var>:<var>56</var>:<var>00</var>.<var>00001</var>Z</kbd> (дефіси, двокрапки та <kbd>Z</kbd> необов'язкові)\n:* Формат MediaWiki, <kbd><var>2001</var><var>01</var><var>15</var><var>14</var><var>56</var><var>00</var></kbd>\n:* Загальний числовий формат, <kbd><var>2001</var>-<var>01</var>-<var>15</var> <var>14</var>:<var>56</var>:<var>00</var></kbd> (необов'язковий часовий пояс <kbd>GMT</kbd>, <kbd>+<var>##</var></kbd> або <kbd>-<var>##</var></kbd> ігнорується)\n:* Формат EXIF, <kbd><var>2001</var>:<var>01</var>:<var>15</var> <var>14</var>:<var>56</var>:<var>00</var></kbd>\n:* Формат RFC 2822 (часовий пояс може бути опущений), <kbd><var>Mon</var>, <var>15</var> <var>Jan</var> <var>2001</var> <var>14</var>:<var>56</var>:<var>00</var></kbd>\n:* Формат RFC 850 (часовий пояс може бути опущений), <kbd><var>Monday</var>, <var>15</var>-<var>Jan</var>-<var>2001</var> <var>14</var>:<var>56</var>:<var>00</var></kbd>\n:* Формат C ctime, <kbd><var>Mon</var> <var>Jan</var> <var>15</var> <var>14</var>:<var>56</var>:<var>00</var> <var>2001</var></kbd>\n:* Секунди від 1970-01-01T00:00:00Z у вигляді цілого числа від 1 до 13 цифр (без <kbd>0</kbd>)\n:* Рядок <kbd>now</kbd>\n;альтернативний роздільник багатьох значень\n:Параметри, що приймають багато значень, зазвичай подаються зі значеннями, розділеними вертикальною рискою, наприклад, <kbd>param=value1|value2</kbd> або <kbd>param=value1%7Cvalue2</kbd>. Якщо значення повинне містити вертикальну риску, використовуйте як роздільник U+001F (роздільник одиниць) ''та'' поставте U+001F перед значенням, наприклад, <kbd>param=%1Fvalue1%1Fvalue2</kbd>.",
+       "api-help-datatypes": "Вхідні дані у MediaWiki мають бути в NFC-нормалізованому UTF-8. MediaWiki може спробувати конвертувати вхідні дані іншого вигляду, але від цього можуть постраждати деякі операції (як [[Special:ApiHelp/edit|редагування]] з перевіркою MD5).\n\nДеякі типи параметрів у запитах API потребують ширшого пояснення:\n;boolean\n:Логічні параметри працюють як галочки HTML: якщо параметр вказано, не залежно від значення, він вважається істинним. Щоб значення було хибним, пропустіть параметр зовсім.\n;timestamp\n:Часові мітки можуть бути вказані у кількох форматах. Рекомендується час і дата в ISO 8601, див. детальніше [[mw:Special:MyLanguage/Timestamp|про формати введення бібліотеки часових міток на mediawiki.org]]. Рекомендована дата і час у форматі ISO 8601, <kbd><var>2001</var>-<var>01</var>-<var>15</var>T<var>14</var>:<var>56</var>:<var>00</var>Z</kbd>. Додатково, рядок <kbd>now</kbd> можна використовувати, щоб вказати поточну часову мітку.\n;альтернативний роздільник багатьох значень\n:Параметри, що приймають багато значень, зазвичай подаються зі значеннями, розділеними вертикальною рискою, наприклад, <kbd>param=value1|value2</kbd> або <kbd>param=value1%7Cvalue2</kbd>. Якщо значення повинне містити вертикальну риску, використовуйте як роздільник U+001F (роздільник одиниць) ''та'' поставте U+001F перед значенням, наприклад, <kbd>param=%1Fvalue1%1Fvalue2</kbd>.",
        "api-help-templatedparams-header": "Шаблонні параметри",
        "api-help-templatedparams": "Шаблонні параметри підтримують випадки, в яких модулю API необхідне значення для кожного значення якогось іншого параметра. Наприклад, якби був модуль API для запитів фруктів, у нього був би параметр <var>fruits</var> для зазначення, запит на які саме фрукти надсилається і шаблонний параметр <var>{fruit}-quantity</var> для зазначення, скільки саме кожного різновиду фруктів потрібно. Клієнт API, якому потрібні 1 яблуко, 5 бананів і 20 полуниць, таким чином, міг би надіслати запит у формі <kbd>fruits=apples|bananas|strawberries&apples-quantity=1&bananas-quantity=5&strawberries-quantity=20</kbd>.",
        "api-help-param-type-limit": "Тип: ціле число або <kbd>max</kbd>",
index 1bdac01..b7c60ed 100644 (file)
@@ -67,7 +67,7 @@
        "apihelp-clearhasmsg-example-1": "清除目前使用者的 <code>hasmsg</code> 標記。",
        "apihelp-clientlogin-summary": "使用互動流程來登入 wiki。",
        "apihelp-clientlogin-example-login": "開始以使用者 <kbd>Example</kbd> 與密碼 <kbd>ExamplePassword</kbd> 來登入至 wiki 的過程。",
-       "apihelp-clientlogin-example-login2": "在 <samp>UI</samp> 回應雙重認證後繼續登入,提供 <kbd>987654</kbd> 的 <var>OATHToken</var>。",
+       "apihelp-clientlogin-example-login2": "在 <samp>UI</samp> 回應雙因素驗證後繼續登入,提供 <kbd>987654</kbd> 的 <var>OATHToken</var>。",
        "apihelp-compare-summary": "比較 2 個頁面間的差異。",
        "apihelp-compare-extended-description": "\"from\" 以及 \"to\" 的修訂編號,頁面標題或頁面 ID 為必填。",
        "apihelp-compare-param-fromtitle": "要比對的第一個標題。",
        "apihelp-query+languageinfo-paramvalue-prop-bcp47": "BCP-47 語言代碼。",
        "apihelp-query+languageinfo-paramvalue-prop-dir": "語言的書寫方向(<code>ltr</code> 或 <code>rtl</code>)。",
        "apihelp-query+languageinfo-paramvalue-prop-autonym": "語言的本語稱呼,也就是該語言用自己語言本身寫出的名稱。",
-       "apihelp-query+languageinfo-paramvalue-prop-name": "在由 <var>lilang</var> 參數所指定語言裡的語言名稱,如有需要可套用語言遞補。",
+       "apihelp-query+languageinfo-paramvalue-prop-name": "在由 <var>uselang</var> 參數所指定語言裡的語言名稱,如有需要可套用語言遞補。",
        "apihelp-query+languageinfo-paramvalue-prop-fallbacks": "替此語言設置的遞補語言之語言代碼。「en」不包括在內含的最後遞補(但一些語言可明確地指定「en」為最後遞補)。",
        "apihelp-query+languageinfo-paramvalue-prop-variants": "由此語言所支援的變體語言代碼。",
        "apihelp-query+languageinfo-param-code": "所應要回傳的語言該語言代碼,或是以 <code>*</code> 來表示為全部語言。",
index d24a2a5..f654404 100644 (file)
@@ -31,13 +31,22 @@ use User;
  * @since 1.34 Factored out from DatabaseBlock (previously Block).
  */
 abstract class AbstractBlock {
-       /** @var string */
+       /**
+        * @deprecated since 1.34. Use getReason and setReason instead.
+        * @var string
+        */
        public $mReason;
 
-       /** @var string */
+       /**
+        * @deprecated since 1.34. Use getTimestamp and setTimestamp instead.
+        * @var string
+        */
        public $mTimestamp;
 
-       /** @var string */
+       /**
+        * @deprecated since 1.34. Use getExpiry and setExpiry instead.
+        * @var string
+        */
        public $mExpiry = '';
 
        /** @var bool */
@@ -49,7 +58,10 @@ abstract class AbstractBlock {
        /** @var bool */
        protected $blockCreateAccount = false;
 
-       /** @var bool */
+       /**
+        * @deprecated since 1.34. Use getHideName and setHideName instead.
+        * @var bool
+        */
        public $mHideName = false;
 
        /** @var User|string */
index fbf9a07..2fd62ee 100644 (file)
@@ -53,10 +53,16 @@ use Wikimedia\Rdbms\IDatabase;
  * @since 1.34 Renamed from Block.
  */
 class DatabaseBlock extends AbstractBlock {
-       /** @var bool */
+       /**
+        * @deprecated since 1.34. Use getType to check whether a block is autoblocking.
+        * @var bool
+        */
        public $mAuto;
 
-       /** @var int */
+       /**
+        * @deprecated since 1.34. Use getParentBlockId instead.
+        * @var int
+        */
        public $mParentBlockId;
 
        /** @var int */
index a8bcfc6..5745451 100644 (file)
@@ -98,6 +98,12 @@ class MessageCache {
        /** @var Language */
        protected $contLang;
 
+       /**
+        * Track which languages have been loaded by load().
+        * @var array
+        */
+       private $loadedLanguages = [];
+
        /**
         * Singleton instance
         *
@@ -264,23 +270,12 @@ class MessageCache {
                }
 
                # Don't do double loading...
-               if ( $this->cache->has( $code ) && $mode != self::FOR_UPDATE ) {
+               if ( isset( $this->loadedLanguages[$code] ) && $mode != self::FOR_UPDATE ) {
                        return true;
                }
 
                $this->overridable = array_flip( Language::getMessageKeysFor( $code ) );
 
-               // T208897 array_flip can fail and return null
-               if ( is_null( $this->overridable ) ) {
-                       LoggerFactory::getInstance( 'MessageCache' )->error(
-                               __METHOD__ . ': $this->overridable is null',
-                               [
-                                       'message_keys' => Language::getMessageKeysFor( $code ),
-                                       'code' => $code
-                               ]
-                       );
-               }
-
                # 8 lines of code just to say (once) that message cache is disabled
                if ( $this->mDisable ) {
                        static $shownDisabled = false;
@@ -396,6 +391,9 @@ class MessageCache {
                        wfDebugLog( 'MessageCacheError', __METHOD__ . ": Failed to load $code\n" );
                        # This used to throw an exception, but that led to nasty side effects like
                        # the whole wiki being instantly down if the memcached server died
+               } else {
+                       # All good, just record the success
+                       $this->loadedLanguages[$code] = true;
                }
 
                if ( !$this->cache->has( $code ) ) { // sanity
@@ -1300,6 +1298,7 @@ class MessageCache {
                        $this->wanCache->touchCheckKey( $this->getCheckKey( $code ) );
                }
                $this->cache->clear();
+               $this->loadedLanguages = [];
        }
 
        /**
index d97abca..31b4443 100644 (file)
@@ -662,15 +662,20 @@ class ChangesList extends ContextSource {
         * field of this revision, if it's marked as deleted.
         * @param RCCacheEntry|RecentChange $rc
         * @param int $field
-        * @param User|null $user User object to check, or null to use $wgUser
+        * @param User|null $user User object to check against. If null, the global RequestContext's
+        * User is assumed instead.
         * @return bool
         */
        public static function userCan( $rc, $field, User $user = null ) {
+               if ( $user === null ) {
+                       $user = RequestContext::getMain()->getUser();
+               }
+
                if ( $rc->mAttribs['rc_type'] == RC_LOG ) {
                        return LogEventsList::userCanBitfield( $rc->mAttribs['rc_deleted'], $field, $user );
-               } else {
-                       return Revision::userCanBitfield( $rc->mAttribs['rc_deleted'], $field, $user );
                }
+
+               return RevisionRecord::userCanBitfield( $rc->mAttribs['rc_deleted'], $field, $user );
        }
 
        /**
index 455eb0d..8e5e0a8 100644 (file)
@@ -329,7 +329,7 @@ class WikitextContent extends TextContent {
         * using the global Parser service.
         *
         * @param Title $title
-        * @param int $revId Revision to pass to the parser (default: null)
+        * @param int|null $revId Revision to pass to the parser (default: null)
         * @param ParserOptions $options (default: null)
         * @param bool $generateHtml (default: true)
         * @param ParserOutput &$output ParserOutput representing the HTML form of the text,
index 4393abb..6eeac1c 100644 (file)
@@ -332,7 +332,8 @@ class RequestContext implements IContextSource, MutableContext {
                                $request = $this->getRequest();
                                $user = $this->getUser();
 
-                               $code = $request->getVal( 'uselang', 'user' );
+                               // Optimisation: Avoid slow getVal(), this isn't user-generated content.
+                               $code = $request->getRawVal( 'uselang', 'user' );
                                if ( $code === 'user' ) {
                                        $code = $user->getOption( 'language' );
                                }
@@ -371,33 +372,28 @@ class RequestContext implements IContextSource, MutableContext {
                        Hooks::run( 'RequestContextCreateSkin', [ $this, &$skin ] );
                        $factory = MediaWikiServices::getInstance()->getSkinFactory();
 
-                       // If the hook worked try to set a skin from it
                        if ( $skin instanceof Skin ) {
+                               // The hook provided a skin object
                                $this->skin = $skin;
                        } elseif ( is_string( $skin ) ) {
+                               // The hook provided a skin name
                                // Normalize the key, just in case the hook did something weird.
                                $normalized = Skin::normalizeKey( $skin );
                                $this->skin = $factory->makeSkin( $normalized );
-                       }
-
-                       // If this is still null (the hook didn't run or didn't work)
-                       // then go through the normal processing to load a skin
-                       if ( $this->skin === null ) {
+                       } else {
+                               // No hook override, go through normal processing
                                if ( !in_array( 'skin', $this->getConfig()->get( 'HiddenPrefs' ) ) ) {
-                                       # get the user skin
                                        $userSkin = $this->getUser()->getOption( 'skin' );
-                                       $userSkin = $this->getRequest()->getVal( 'useskin', $userSkin );
+                                       // Optimisation: Avoid slow getVal(), this isn't user-generated content.
+                                       $userSkin = $this->getRequest()->getRawVal( 'useskin', $userSkin );
                                } else {
-                                       # if we're not allowing users to override, then use the default
                                        $userSkin = $this->getConfig()->get( 'DefaultSkin' );
                                }
 
-                               // Normalize the key in case the user is passing gibberish
-                               // or has old preferences (T71566).
+                               // Normalize the key in case the user is passing gibberish query params
+                               // or has old user preferences (T71566).
+                               // Skin::normalizeKey will also validate it, so makeSkin() won't throw.
                                $normalized = Skin::normalizeKey( $userSkin );
-
-                               // Skin::normalizeKey will also validate it, so
-                               // this won't throw an exception
                                $this->skin = $factory->makeSkin( $normalized );
                        }
 
index 501f01a..82fff6b 100644 (file)
@@ -28,7 +28,6 @@ use Wikimedia\Rdbms\DatabaseDomain;
 use Wikimedia\Rdbms\Blob;
 use Wikimedia\Rdbms\ResultWrapper;
 use Wikimedia\Rdbms\IResultWrapper;
-use Wikimedia\Rdbms\DBConnectionError;
 use Wikimedia\Rdbms\DBUnexpectedError;
 use Wikimedia\Rdbms\DBExpectedError;
 
@@ -80,101 +79,96 @@ class DatabaseOracle extends Database {
                return 'oracle';
        }
 
-       function implicitGroupby() {
-               return false;
-       }
-
        function implicitOrderby() {
                return false;
        }
 
        protected function open( $server, $user, $password, $dbName, $schema, $tablePrefix ) {
                if ( !function_exists( 'oci_connect' ) ) {
-                       throw new DBConnectionError(
-                               $this,
+                       throw $this->newExceptionAfterConnectError(
                                "Oracle functions missing, have you compiled PHP with the --with-oci8 option?\n " .
-                                       "(Note: if you recently installed PHP, you may need to restart your webserver\n " .
-                                       "and database)\n" );
+                               "(Note: if you recently installed PHP, you may need to restart your webserver\n " .
+                               "and database)"
+                       );
                }
 
+               $this->close();
+
                if ( $schema !== null ) {
-                       // We use the *database* aspect of $domain for schema, not the domain schema
-                       throw new DBExpectedError(
-                               $this,
-                               __CLASS__ . ": cannot use schema '$schema'; " .
-                               "the database component '$dbName' is actually interpreted as the Oracle schema."
+                       // This uses the *database* aspect of $domain for schema, not the domain schema
+                       throw $this->newExceptionAfterConnectError(
+                               "Got schema '$schema'; not supported. " .
+                               "The database component '$dbName' is actually interpreted as the Oracle schema."
                        );
                }
 
-               $this->close();
                $this->user = $user;
                $this->password = $password;
-               if ( !$server ) {
-                       // Backward compatibility (server used to be null and TNS was supplied in dbname)
+               if ( strlen( $server ) ) {
+                       // Transparent Network Substrate (TNS) endpoint
+                       $this->server = $server;
+                       // Database name, defaulting to the user name
+                       $realDatabase = strlen( $dbName ) ? $dbName : $user;
+               } else {
+                       // Backward compatibility; $server used to be null and $dbName was the TNS
                        $this->server = $dbName;
                        $realDatabase = $user;
-               } else {
-                       // $server now holds the TNS endpoint
-                       $this->server = $server;
-                       // $dbName is schema name if different from username
-                       $realDatabase = $dbName ?: $user;
-               }
-
-               if ( !strlen( $user ) ) { # e.g. the class is being loaded
-                       return null;
                }
-
                $session_mode = ( $this->flags & DBO_SYSDBA ) ? OCI_SYSDBA : OCI_DEFAULT;
 
-               Wikimedia\suppressWarnings();
-               if ( $this->flags & DBO_PERSISTENT ) {
-                       $this->conn = oci_pconnect(
-                               $this->user,
-                               $this->password,
-                               $this->server,
-                               $this->defaultCharset,
-                               $session_mode
-                       );
-               } elseif ( $this->flags & DBO_DEFAULT ) {
-                       $this->conn = oci_new_connect(
-                               $this->user,
-                               $this->password,
-                               $this->server,
-                               $this->defaultCharset,
-                               $session_mode
-                       );
-               } else {
-                       $this->conn = oci_connect(
-                               $this->user,
-                               $this->password,
-                               $this->server,
-                               $this->defaultCharset,
-                               $session_mode
-                       );
-               }
-               Wikimedia\restoreWarnings();
-
-               if ( $this->user != $realDatabase ) {
-                       // change current schema in session
-                       $this->selectDB( $realDatabase );
-               } else {
-                       $this->currentDomain = new DatabaseDomain(
-                               $realDatabase,
-                               null,
-                               $tablePrefix
-                       );
-               }
+               $this->installErrorHandler();
+               try {
+                       $this->conn = $this->getFlag( DBO_PERSISTENT )
+                               ? oci_pconnect(
+                                       $this->user,
+                                       $this->password,
+                                       $this->server,
+                                       $this->defaultCharset,
+                                       $session_mode
+                               )
+                               : oci_new_connect(
+                                       $this->user,
+                                       $this->password,
+                                       $this->server,
+                                       $this->defaultCharset,
+                                       $session_mode
+                               );
+               } catch ( Exception $e ) {
+                       $this->restoreErrorHandler();
+                       throw $this->newExceptionAfterConnectError( $e->getMessage() );
+               }
+               $error = $this->restoreErrorHandler();
 
                if ( !$this->conn ) {
-                       throw new DBConnectionError( $this, $this->lastError() );
+                       throw $this->newExceptionAfterConnectError( $error ?: $this->lastError() );
                }
 
-               # removed putenv calls because they interfere with the system globaly
-               $this->doQuery( 'ALTER SESSION SET NLS_TIMESTAMP_FORMAT=\'DD-MM-YYYY HH24:MI:SS.FF6\'' );
-               $this->doQuery( 'ALTER SESSION SET NLS_TIMESTAMP_TZ_FORMAT=\'DD-MM-YYYY HH24:MI:SS.FF6\'' );
-               $this->doQuery( 'ALTER SESSION SET NLS_NUMERIC_CHARACTERS=\'.,\'' );
-
-               return (bool)$this->conn;
+               try {
+                       if ( $this->user != $realDatabase ) {
+                               // Change current schema for the entire session
+                               $this->selectDomain( new DatabaseDomain(
+                                       $realDatabase,
+                                       $this->currentDomain->getSchema(),
+                                       $this->currentDomain->getTablePrefix()
+                               ) );
+                       } else {
+                               $this->currentDomain = new DatabaseDomain( $realDatabase, null, $tablePrefix );
+                       }
+                       $set = [
+                               'NLS_TIMESTAMP_FORMAT' => 'DD-MM-YYYY HH24:MI:SS.FF6',
+                               'NLS_TIMESTAMP_TZ_FORMAT' => 'DD-MM-YYYY HH24:MI:SS.FF6',
+                               'NLS_NUMERIC_CHARACTERS' => '.,'
+                       ];
+                       foreach ( $set as $var => $val ) {
+                               $this->query(
+                                       "ALTER SESSION SET {$var}=" . $this->addQuotes( $val ),
+                                       __METHOD__,
+                                       self::QUERY_IGNORE_DBO_TRX | self::QUERY_NO_RETRY
+                               );
+                       }
+               } catch ( Exception $e ) {
+                       throw $this->newExceptionAfterConnectError( $e->getMessage() );
+               }
        }
 
        /**
index 3d404d3..0c17840 100644 (file)
@@ -212,9 +212,6 @@ abstract class MWLBFactory {
                $flags = DBO_DEFAULT;
                $flags |= $options->get( 'DebugDumpSql' ) ? DBO_DEBUG : 0;
                $flags |= $options->get( 'DebugLogFile' ) ? DBO_DEBUG : 0;
-               if ( $server['type'] === 'oracle' ) {
-                       $flags |= $options->get( 'DBOracleDRCP' ) ? DBO_PERSISTENT : 0;
-               }
 
                $server += [
                        'tablePrefix' => $options->get( 'DBprefix' ),
index 3625476..611469c 100644 (file)
@@ -37,7 +37,7 @@ class SearchUpdate implements DeferrableUpdate {
        /** @var Title Title we're updating */
        private $title;
 
-       /** @var Content|bool Content of the page (not text) */
+       /** @var Content|null Content of the page (not text) */
        private $content;
 
        /** @var WikiPage **/
@@ -45,30 +45,30 @@ class SearchUpdate implements DeferrableUpdate {
 
        /**
         * @param int $id Page id to update
-        * @param Title|string $title Title of page to update
-        * @param Content|string|bool $c Content of the page to update. Default: false.
-        *  If a Content object, text will be gotten from it. String is for back-compat.
-        *  Passing false tells the backend to just update the title, not the content
+        * @param Title $title Title of page to update
+        * @param Content|null $c Content of the page to update.
         */
-       public function __construct( $id, $title, $c = false ) {
+       public function __construct( $id, $title, $c = null ) {
                if ( is_string( $title ) ) {
-                       $nt = Title::newFromText( $title );
+                       wfDeprecated( __METHOD__ . " with a string for the title", 1.34 );
+                       $this->title = Title::newFromText( $title );
+                       if ( $this->title === null ) {
+                               throw new InvalidArgumentException( "Cannot construct the title: $title" );
+                       }
                } else {
-                       $nt = $title;
+                       $this->title = $title;
                }
 
-               if ( $nt ) {
-                       $this->id = $id;
-                       // is_string() check is back-compat for ApprovedRevs
-                       if ( is_string( $c ) ) {
-                               $this->content = new TextContent( $c );
-                       } else {
-                               $this->content = $c ?: false;
-                       }
-                       $this->title = $nt;
-               } else {
-                       wfDebug( "SearchUpdate object created with invalid title '$title'\n" );
+               $this->id = $id;
+               // is_string() check is back-compat for ApprovedRevs
+               if ( is_string( $c ) ) {
+                       wfDeprecated( __METHOD__ . " with a string for the content", 1.34 );
+                       $c = new TextContent( $c );
+               } elseif ( is_bool( $c ) ) {
+                       wfDeprecated( __METHOD__ . " with a boolean for the content", 1.34 );
+                       $c = null;
                }
+               $this->content = $c;
        }
 
        /**
@@ -94,15 +94,13 @@ class SearchUpdate implements DeferrableUpdate {
                        if ( $this->getLatestPage() === null ) {
                                $search->delete( $this->id, $normalTitle );
                                continue;
-                       } elseif ( $this->content === false ) {
+                       } elseif ( $this->content === null ) {
                                $search->updateTitle( $this->id, $normalTitle );
                                continue;
                        }
 
-                       $text = $search->getTextFromContent( $this->title, $this->content );
-                       if ( !$search->textAlreadyUpdatedForIndex() ) {
-                               $text = $this->updateText( $text, $search );
-                       }
+                       $text = $this->content !== null ? $this->content->getTextForSearchIndex() : '';
+                       $text = $this->updateText( $text, $search );
 
                        # Perform the actual update
                        $search->update( $this->id, $normalTitle, $search->normalizeText( $text ) );
index b30607f..969e0ba 100644 (file)
@@ -25,13 +25,13 @@ use Wikimedia\Assert\Assert;
 /**
  * Renders a diff for a single slot (that is, a diff between two content objects).
  *
- * Callers should obtain this class by invoking ContentHandler::getSlotDiffRendererClass
+ * Callers should obtain instances of this class by invoking ContentHandler::getSlotDiffRenderer
  * on the content handler of the new content object (ie. the one shown on the right side
  * of the diff), or of the old one if the new one does not exist.
  *
  * The default implementation just does a text diff on the native text representation.
  * Content handler extensions can subclass this to provide a more appropriate diff method by
- * overriding ContentHandler::getSlotDiffRendererClass. Other extensions that want to interfere
+ * overriding ContentHandler::getSlotDiffRendererInternal. Other extensions that want to interfere
  * with diff generation in some way can use the GetSlotDiffRenderer hook.
  *
  * @ingroup DifferenceEngine
index 4d70d66..4db351b 100644 (file)
@@ -154,12 +154,13 @@ class ExternalStoreDB extends ExternalStoreMedium {
         */
        public function getSlave( $cluster ) {
                $lb = $this->getLoadBalancer( $cluster );
-               $domainId = $this->getDomainId( $lb->getServerInfo( $lb->getWriterIndex() ) );
-
-               $db = $lb->getConnectionRef( DB_REPLICA, [], $domainId );
-               $db->clearFlag( DBO_TRX ); // sanity
 
-               return $db;
+               return $lb->getConnectionRef(
+                       DB_REPLICA,
+                       [],
+                       $this->getDomainId( $lb->getServerInfo( $lb->getWriterIndex() ) ),
+                       $lb::CONN_TRX_AUTOCOMMIT
+               );
        }
 
        /**
@@ -170,12 +171,13 @@ class ExternalStoreDB extends ExternalStoreMedium {
         */
        public function getMaster( $cluster ) {
                $lb = $this->getLoadBalancer( $cluster );
-               $domainId = $this->getDomainId( $lb->getServerInfo( $lb->getWriterIndex() ) );
-
-               $db = $lb->getMaintenanceConnectionRef( DB_MASTER, [], $domainId );
-               $db->clearFlag( DBO_TRX ); // sanity
 
-               return $db;
+               return $lb->getMaintenanceConnectionRef(
+                       DB_MASTER,
+                       [],
+                       $this->getDomainId( $lb->getServerInfo( $lb->getWriterIndex() ) ),
+                       $lb::CONN_TRX_AUTOCOMMIT
+               );
        }
 
        /**
index a7cef3c..ed151e6 100644 (file)
@@ -23,7 +23,7 @@
 
 /**
  * Object handling generic submission, CSRF protection, layout and
- * other logic for UI forms. in a reusable manner.
+ * other logic for UI forms in a reusable manner.
  *
  * In order to generate the form, the HTMLForm object takes an array
  * structure detailing the form fields available. Each element of the
index 85cbbb1..354432b 100644 (file)
@@ -130,6 +130,7 @@ class HTMLSelectAndOtherField extends HTMLSelectField {
                        'textinput' => $textAttribs,
                        'dropdowninput' => $dropdownInputAttribs,
                        'or' => false,
+                       'required' => $this->mParams[ 'required' ] ?? false,
                        'classes' => [ 'mw-htmlform-select-and-other-field' ],
                        'data' => [
                                'maxlengthUnit' => $this->mParams['maxlength-unit'] ?? 'bytes'
index 47c1f18..c928df7 100644 (file)
@@ -135,6 +135,7 @@ class HTMLSelectOrOtherField extends HTMLTextField {
                        'disabled' => $disabled,
                        'textinput' => $textAttribs,
                        'dropdowninput' => $dropdownAttribs,
+                       'required' => $this->mParams[ 'required' ] ?? false,
                        'or' => true,
                ] );
        }
index 7ac895c..8768f07 100644 (file)
@@ -29,7 +29,7 @@
  * @ingroup SpecialPage
  */
 class UploadSourceAdapter {
-       /** @var array */
+       /** @var ImportSource[] */
        public static $sourceRegistrations = [];
 
        /** @var ImportSource */
@@ -56,7 +56,7 @@ class UploadSourceAdapter {
        /**
         * @param string $path
         * @param string $mode
-        * @param array $options
+        * @param int $options
         * @param string &$opened_path
         * @return bool
         */
@@ -104,14 +104,14 @@ class UploadSourceAdapter {
 
        /**
         * @param string $data
-        * @return bool
+        * @return false
         */
        function stream_write( $data ) {
                return false;
        }
 
        /**
-        * @return mixed
+        * @return int
         */
        function stream_tell() {
                return $this->mPosition;
@@ -125,7 +125,7 @@ class UploadSourceAdapter {
        }
 
        /**
-        * @return array
+        * @return int[]
         */
        function url_stat() {
                $result = [];
index 567fb10..99d594d 100644 (file)
@@ -21,6 +21,7 @@
  * @ingroup Deployment
  */
 
+use MediaWiki\Installer\InstallException;
 use MediaWiki\MediaWikiServices;
 
 /**
@@ -51,6 +52,7 @@ class CliInstaller extends Installer {
         * @param string $siteName
         * @param string|null $admin
         * @param array $options
+        * @throws InstallException
         */
        function __construct( $siteName, $admin = null, array $options = [] ) {
                global $wgContLang;
@@ -114,7 +116,7 @@ class CliInstaller extends Installer {
                        $status = $this->validateExtensions(
                                'extension', 'extensions', $options['extensions'] );
                        if ( !$status->isOK() ) {
-                               $this->showStatusMessage( $status );
+                               throw new InstallException( $status );
                        }
                        $this->setVar( '_Extensions', $status->value );
                } elseif ( isset( $options['with-extensions'] ) ) {
@@ -125,7 +127,7 @@ class CliInstaller extends Installer {
                if ( isset( $options['skins'] ) ) {
                        $status = $this->validateExtensions( 'skin', 'skins', $options['skins'] );
                        if ( !$status->isOK() ) {
-                               $this->showStatusMessage( $status );
+                               throw new InstallException( $status );
                        }
                        $skins = $status->value;
                } else {
@@ -176,15 +178,23 @@ class CliInstaller extends Installer {
 
                $vars = Installer::getExistingLocalSettings();
                if ( $vars ) {
-                       $this->showStatusMessage(
-                               Status::newFatal( "config-localsettings-cli-upgrade" )
-                       );
+                       $status = Status::newFatal( "config-localsettings-cli-upgrade" );
+                       $this->showStatusMessage( $status );
+                       return $status;
                }
 
-               $this->performInstallation(
+               $result = $this->performInstallation(
                        [ $this, 'startStage' ],
                        [ $this, 'endStage' ]
                );
+               // PerformInstallation bails on a fatal, so make sure the last item
+               // completed before giving 'next.' Likewise, only provide back on failure
+               $lastStepStatus = end( $result );
+               if ( $lastStepStatus->isOk() ) {
+                       return Status::newGood();
+               } else {
+                       return $lastStepStatus;
+               }
        }
 
        /**
@@ -248,11 +258,6 @@ class CliInstaller extends Installer {
                                $this->showMessage( ...$w );
                        }
                }
-
-               if ( !$status->isOK() ) {
-                       echo "\n";
-                       exit( 1 );
-               }
        }
 
        public function envCheckPath() {
diff --git a/includes/installer/InstallException.php b/includes/installer/InstallException.php
new file mode 100644 (file)
index 0000000..a03a5aa
--- /dev/null
@@ -0,0 +1,51 @@
+<?php
+/**
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License along
+ * with this program; if not, write to the Free Software Foundation, Inc.,
+ * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ * http://www.gnu.org/copyleft/gpl.html
+ *
+ * @file
+ */
+
+namespace MediaWiki\Installer;
+
+use Throwable;
+
+/**
+ * Exception thrown if an error occur which installation
+ * @ingroup Exception
+ */
+class InstallException extends \MWException {
+       /**
+        * @var \Status State when an exception occurs
+        */
+       private $status;
+
+       /**
+        * InstallException constructor.
+        * @param \Status $status State when an exception occurs
+        * @param string $message The Exception message to throw
+        * @param int $code The Exception code
+        * @param Throwable|null $previous The previous throwable used for the exception chaining
+        */
+       public function __construct( \Status $status, $message = '', $code = 0,
+               Throwable $previous = null ) {
+               parent::__construct( $message, $code, $previous );
+               $this->status = $status;
+       }
+
+       public function getStatus() : \Status {
+               return $this->status;
+       }
+}
index f6a5c41..7452b7c 100644 (file)
@@ -756,6 +756,8 @@ abstract class Installer {
         */
        protected function envCheckDB() {
                global $wgLang;
+               /** @var string|null $dbType The user-specified database type */
+               $dbType = $this->getVar( 'wgDBtype' );
 
                $allNames = [];
 
@@ -768,25 +770,27 @@ abstract class Installer {
                $databases = $this->getCompiledDBs();
 
                $databases = array_flip( $databases );
+               $ok = true;
                foreach ( array_keys( $databases ) as $db ) {
                        $installer = $this->getDBInstaller( $db );
                        $status = $installer->checkPrerequisites();
                        if ( !$status->isGood() ) {
+                               if ( !$this instanceof WebInstaller && $db === $dbType ) {
+                                       // Strictly check the key database type instead of just outputting message
+                                       // Note: No perform this check run from the web installer, since this method always called by
+                                       // the welcome page under web installation, so $dbType will always be 'mysql'
+                                       $ok = false;
+                               }
                                $this->showStatusMessage( $status );
-                       }
-                       if ( !$status->isOK() ) {
                                unset( $databases[$db] );
                        }
                }
                $databases = array_flip( $databases );
                if ( !$databases ) {
                        $this->showError( 'config-no-db', $wgLang->commaList( $allNames ), count( $allNames ) );
-
-                       // @todo FIXME: This only works for the web installer!
                        return false;
                }
-
-               return true;
+               return $ok;
        }
 
        /**
@@ -1586,7 +1590,7 @@ abstract class Installer {
         * @param callable $startCB A callback array for the beginning of each step
         * @param callable $endCB A callback array for the end of each step
         *
-        * @return array Array of Status objects
+        * @return Status[] Array of Status objects
         */
        public function performInstallation( $startCB, $endCB ) {
                $installResults = [];
index 6b2024f..26d2ea7 100644 (file)
        "config-restart": "Sí, reanicialu",
        "config-welcome": "=== Comprobaciones del entornu ===\nAgora van facese unes comprobaciones básiques para ver si l'entornu ye afayadizu pa la instalación de MediaWiki.\nAlcuérdese d'incluir esta información si necesita encontu pa completar la instalación.",
        "config-welcome-section-copyright": "=== Drechos d'autor y condiciones d'usu ===\n\n$1\n\nEsti programa ye software llibre; puedes redistribuilu y/o camudalu baxo les condiciones de la llicencia pública xeneral GNU tal como la publica la Free Software Foundation; versión 2 o (como prefieras) cualquier versión posterior.\n\nEsti programa distribúese cola esperanza de que pueda ser útil, pero <strong>ensin garantía denguna</strong>; nin siquiera la garantía implícita de <strong>comercialidá</strong> o \n<strong>adautación a un fin determináu</strong>.\nVer la Llicencia pública xeneral GNU pa más detalles.\n\nHabríes de tener recibío [$2 una copia de la llicencia pública xeneral GNU] xunto con esti programa; sinón, escribi a la Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA, o [https://www.gnu.org/copyleft/gpl.html lléila en llinia].",
-       "config-sidebar": "* [https://www.mediawiki.org/wiki/MediaWiki/gl Páxina principal de MediaWiki]\n* [https://www.mediawiki.org/wiki/Special:MyLanguage/Help:Contents Guía del usuariu]\n* [https://www.mediawiki.org/wiki/Special:MyLanguage/Manual:Contents Guía del alministrador]\n* [https://www.mediawiki.org/wiki/Special:MyLanguage/Manual:FAQ Entrugues frecuentes]\n----\n* <doclink href=Readme>Lléame</doclink>\n* <doclink href=ReleaseNotes>Notes de llanzamientu</doclink>\n* <doclink href=Copying>Copia</doclink>\n* <doclink href=UpgradeDoc>Anovamientu</doclink>",
+       "config-sidebar": "* [https://www.mediawiki.org Páxina principal de MediaWiki]\n* [https://www.mediawiki.org/wiki/Special:MyLanguage/Help:Contents Guía del usuariu]\n* [https://www.mediawiki.org/wiki/Special:MyLanguage/Manual:Contents Guía p'alministradores]\n* [https://www.mediawiki.org/wiki/Special:MyLanguage/Manual:FAQ Entrugues frecuentes]",
+       "config-sidebar-readme": "Llei-me",
+       "config-sidebar-relnotes": "Notes de la versión",
+       "config-sidebar-license": "Copiar",
+       "config-sidebar-upgrade": "Anovar",
        "config-env-good": "Comprobóse l'entornu.\nPues instalar MediaWiki.",
        "config-env-bad": "Comprobóse l'entornu.\nNun pues instalar MediaWiki.",
        "config-env-php": "PHP $1 ta instaláu.",
        "config-env-hhvm": "HHVM $1 ta instaláu.",
-       "config-unicode-using-intl": "Usando la [https://pecl.php.net/intl estensión intl PECL] pa la normalización Unicode.",
-       "config-unicode-pure-php-warning": "'''Avisu:''' La [https://pecl.php.net/intl estensión intl PECL] nun ta disponible pa xestionar la normalización Unicode; volviendo a la implementación lenta en PHP puru.\nSi xestiona un sitiu con un tráficu altu, tendría de lleer una migaya sobro la [https://www.mediawiki.org/wiki/Special:MyLanguage/Unicode_normalization_considerations normalización Unicode].",
+       "config-unicode-using-intl": "Usando la [https://php.net/manual/en/book.intl.php estensión intl de PHP] pa la normalización Unicode.",
+       "config-unicode-pure-php-warning": "<strong>Avisu:</strong> La [https://php.net/manual/en/book.intl.php estensión intl de PHP] nun ta disponible pa xestionar la normalización Unicode; volviendo a la implementación lenta en PHP puru.\nSi alministres un sitiu con un tráficu altu, tendríes de lleer sobro la [https://www.mediawiki.org/wiki/Special:MyLanguage/Unicode_normalization_considerations normalización Unicode].",
        "config-unicode-update-warning": "'''Avisu:''' La versión instalada del envoltoriu de normalización Unicode usa una versión antigua de la biblioteca [http://site.icu-project.org/ de los proyeutos ICU].\nTendría [https://www.mediawiki.org/wiki/Special:MyLanguage/Unicode_normalization_considerations d'anovala] si ye importante pa vusté usar Unicode.",
        "config-no-db": "¡Nun pudo alcontrase un controlador de base de datos afayadizu! Necesites instalar un controlador de base de datos pa PHP.\n{{PLURAL:$2|Tien sofitu el tipu de base de datos siguiente|Tienen sofitu los tipos de base de datos siguientes}}: $1.\n\nSi compilasti PHP tu mesmu, reconfigúralu con un cliente de base de datos activáu, por exemplu, usando <code>./configure --with-mysqli</code>.\nSi instalasti PHP dende un paquete de Debian o Ubuntu, necesites instalar tamién,por exemplu, el paquete <code>php-mysql</code>.",
-       "config-outdated-sqlite": "'''Avisu:''' tien SQLite $1, que ye inferior a la versión mínima necesaria $2. SQLite nun tará disponible.",
+       "config-outdated-sqlite": "<strong>Avisu:</strong> tienes SQLite $1, que ye menor que la versión mínima necesaria $2. SQLite nun tará disponible.",
        "config-no-fts3": "'''Avisu:''' SQLite ta compiláu ensin el [//sqlite.org/fts3.html módulu FTS3]; les funciones de gueta nun tarán disponibles nesti sistema.",
        "config-pcre-old": "<strong>Fatal:</strong> Ríquese PCRE $1 o posterior.\nEl binariu de PHP ta enllazáu con PCRE $2.\n[https://www.mediawiki.org/wiki/Manual:Errors_and_symptoms/PCRE Más información].",
        "config-pcre-no-utf8": "<strong>Erru fatal:</strong> Paez que'l módulu PCRE de PHP foi compiláu ensin el soporte PCRE_UTF8.\nMediaWiki requier compatibilidá con UTF_8 pa furrular correutamente.",
        "config-type-oracle": "Oracle",
        "config-type-mssql": "Microsoft SQL Server",
        "config-support-info": "MediaWiki ye compatible colos siguientes sistemes de bases de datos:\n\n$1\n\nSi nun atopes na llista el sistema de base de datos que tas intentando utilizar, sigue les instrucciones enllazaes enriba p'activar la compatibilidá.",
+       "config-dbsupport-mysql": "* [{{int:version-db-mariadb-url}} MariaDB] ye la base de datos primaria pa MediaWiki y la que tien mayor encontu. MediaWiki tamién funciona con [{{int:version-db-myslql-url}} MySQL] y [{{int:version-db-percona-url}} Percona Server], que son compatibles con MariaDB. ([https://www.php.net/manual/en/mysqli.installation.php Cómo compilar PHP con compatibilidá MySQL])",
+       "config-dbsupport-postgres": "* [{{int:version-db-postgres-url}} PostgreSQL] ye un sistema popular de base de datos de códigu abiertu, como alternativa a MySQL. ([https://www.php.net/manual/en/pgsql.installation.php Cómo compilar PHP con compatibilidá PostgreSQL]).",
+       "config-dbsupport-sqlite": "* [{{int:version-db-sqlite-url}} SQLite] ye un sistema de base de datos llixeru que tien encontu perbonu. ([https://https://www.php.net/manual/en/pdo.installation.php Cómo compilar PHP con encontu pa SQLite], usa PDO)",
+       "config-dbsupport-oracle": "* [{{int:version-db-oracle-url}} Oracle] ye una base de datos comercial a nivel empresarial. ([https://www.php.net/manual/en/oci8.installation.php Cómo compilar PHP con encontu pa OCI8])",
        "config-dbsupport-mssql": "* [{{int:version-db-mssql-url}} Microsoft SQL Server] ye un sistema comercial de base de datos empresariales pa Windows. ([https://www.php.net/manual/en/sqlsrv.installation.php Cómo compilar PHP con compatibilidá pa SQLSRV])",
        "config-header-mysql": "Axustes de MariaDB/MySQL",
        "config-header-postgres": "Axustes de PostgreSQL",
        "config-sqlite-readonly": "El ficheru <code>$1</code> nun puede escribise.",
        "config-sqlite-cant-create-db": "Nun pudo crease'l ficheru de la base de datos <code>$1</code>.",
        "config-sqlite-fts3-downgrade": "PHP nun tien compatibilidá pa FTS3, baxando a una versión anterior les tables.",
+       "config-can-upgrade": "Esta base de datos contien tables de MediaWiki.\nP'anovales a MediaWiki $1, pulsia <strong>Siguir</strong>.",
+       "config-upgrade-error": "Hebo un error al anovar les tables de MediaWiki de la base de datos.\n\nPa más información, consulta'l rexistru de más arriba; pa tentalo nuevamente, pulsia <strong>Siguir</strong>.",
+       "config-upgrade-done": "Anovamientu completáu.\n\nYá puedes [$1 principiar a usar la wiki].\n\nSi quies rexenerar el ficheru <code>LocalSettings.php</code>, pulsia nel botón d'abaxo.\nEsto <strong>nun s'encamienta</strong> sacante que teas teniendo problemes cola wiki.",
+       "config-upgrade-done-no-regenerate": "Anovamientu completáu.\n\nYá puedes [$1 principiar a usar la wiki].",
        "config-regenerate": "Rexenerar LocalSettings.php →",
        "config-show-table-status": "¡Falló la consulta <code>SHOW TABLE STATUS</code>!",
        "config-unknown-collation": "<strong>Avisu:</strong> La base de datos utiliza un orde alfabéticu ensin reconocer.",
        "config-admin-help": "Escribe equí'l nome d'usuariu que desees, como por casu \"Nel Bloggs\".\nEsti ye'l nome que vas usar pa entrar na wiki.",
        "config-admin-name-blank": "Escribe'l nome d'usuariu d'un alministrador.",
        "config-admin-password-mismatch": "Les dos contraseñes qu'introduxesti nun concasen.",
+       "config-admin-email": "Direición de corréu electrónicu:",
+       "config-admin-email-help": "Escribi equí una dirección de corréu electrónicu pa que te permita recibir mensaxes d'otros usuarios de la wiki, reaniciar la contraseña y recibir avisos de cambeos nes páxines de la to llista de siguimientu. Puedes dexar esti campu vacíu.",
+       "config-admin-error-user": "Error internu al crear un alministrador col nome «<nowiki>$1</nowiki>»",
+       "config-admin-error-password": "Error internu al configurar una contraseña pal alministrador «<nowiki>$1</nowiki>»: <pre>$2</pre>",
+       "config-admin-error-bademail": "Escribisti una dirección de corréu electrónicu inválida.",
        "config-optional-skip": "Yá toi aburríu, namái instala la wiki.",
        "config-profile-wiki": "Wiki públicu",
        "config-profile-no-anon": "Ríquese crear una cuenta",
        "config-install-tables-failed": "<strong>Error:</strong> La creación de tables falló col error siguiente: $1",
        "config-install-interwiki": "Enllenando la tabla d'interwiki predeterminada",
        "config-install-interwiki-list": "Nun pudo lleese'l ficheru <code>interwiki.list</code>.",
+       "config-install-interwiki-exists": "<strong>Avisu:</strong> Paez que la tabla d'interwikis yá contien entraes.\nSaltando la llista predeterminada.",
+       "config-install-stats": "Aniciando les estadístiques",
+       "config-install-keys": "Xenerando les claves secretes",
+       "config-install-updates": "Torgar que s'executen anovamientos innecesarios",
        "config-download-localsettings": "Descargar <code>LocalSettings.php</code>",
        "config-help": "ayuda",
+       "config-help-tooltip": "pulsia p'ampliar",
        "config-nofile": "Nun pudo atopase'l ficheru \"$1\". ¿Desaniciose?",
        "config-skins-screenshots": "$1 (imaxes de pantalla: $2)",
        "config-skins-screenshot": "$1 ($2)",
index d49fd9f..d56295b 100644 (file)
@@ -52,7 +52,7 @@
        "config-restart": "Sí, torna a començar",
        "config-welcome": "=== Comprovacions de l'entorn ===\nS'efectuaran comprovacions bàsiques per veure si l'entorn és adequat per a la instal·lació del MediaWiki.\nRecordeu d'incloure aquesta informació si heu de demanar ajuda sobre com completar la instal·lació.",
        "config-welcome-section-copyright": "=== Drets d'autor i condicions ===\n\n$1\n\nAquest programa és de programari lliure; podeu redistribuir-lo i/o modificar-lo sota les condicions de la Llicència Pública General GNU com es publicada per la Free Software Foundation; qualsevol versió 2 de la llicència, o (opcionalment) qualsevol versió posterior.\n\nAquest programa és distribueix amb l'esperança que serà útil, però <strong>sense cap garantia</strong>; sense ni tan sols la garantia implícita de <strong>\ncomerciabilitat</strong> o <strong>idoneïtat per a un propòsit particular</strong>.\nConsulteu la Llicència Pública General GNU, per a més detalls.\n\nHauríeu d'haver rebut [$2 una còpia de la Llicència Pública General GNU] amb aquest programa; si no, escriviu a la Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA o [https://www.gnu.org/copyleft/gpl.html per llegir-lo en línia].",
-       "config-sidebar": "* [https://www.mediawiki.org la Pàgina d'inici]\n* [https://www.mediawiki.org/wiki/Special:MyLanguage/Help:Contents Guia de l'usuari]\n* [https://www.mediawiki.org/wiki/Special:MyLanguage/Manual:Contents Guia de l'administrador]\n* [https://www.mediawiki.org/wiki/Special:MyLanguage/Manual:FAQ PMF]\n----\n* <doclink href=Readme>Llegeix-me</doclink>\n* <doclink href=ReleaseNotes>Notes de la versió</doclink>\n* <doclink href=Còpia>Còpia</doclink>\n* <doclink href=UpgradeDoc>Actualització</doclink>",
+       "config-sidebar": "* [https://www.mediawiki.org la Pàgina d'inici]\n* [https://www.mediawiki.org/wiki/Special:MyLanguage/Help:Contents Guia de l'usuari]\n* [https://www.mediawiki.org/wiki/Special:MyLanguage/Manual:Contents Guia de l'administrador]\n* [https://www.mediawiki.org/wiki/Special:MyLanguage/Manual:FAQ PMF]",
        "config-sidebar-readme": "Llegeix-me",
        "config-sidebar-relnotes": "Notes de la versió",
        "config-sidebar-license": "Còpia",
@@ -88,6 +88,7 @@
        "config-uploads-not-safe": "<strong>Avís:</strong> El directori de càrregues per defecte <code>$1</code> és vulnerable a l'execució d'scripts arbitraris.\nEncara que el MediaWiki comprova tots els fitxers que es carreguen davant d'amenaces de seguretat, és molt recomanable [https://www.mediawiki.org/ wiki/Special:MyLanguage/Manual:Security#Upload_security tancar aquesta vulnerabilitat de seguretat] abans d'habilitar les càrregues.",
        "config-no-cli-uploads-check": "<strong>Avís</strong>: no s'ha comprovat el directori per defecte per a càrregues (<code><span class=\"notranslate\">$1</span></code>$) per vulnerabilitats en l'execució arbitrària durant la instal·lació amb la línia d'ordres.",
        "config-brokenlibxml": "El vostre sistema té una combinació de versions de PHP i libxml2 que són problemàtiques i que poden causar corrupció de dades no aparent a MediaWiki i a altres aplicacions web.\nActualitzeu-vos a libxml2 2.7.3 o superior ([https://bugs.php.net/bug.php?id=45996 informe d'error al projecte PHP]).\nS'ha interromput la instal·lació.",
+       "config-using-32bit": "<strong>Avís</strong>: el vostre sistema sembla que s'executa amb enters de 32 bits. Això [https://www.mediawiki.org/wiki/special:MyLanguage/Manual:32-bit no és aconsellable].",
        "config-db-type": "Tipus de base de dades:",
        "config-db-host": "Servidor de la base de dades:",
        "config-db-host-help": "Si el servidor de base de dades és en un servidor diferent, introduïu el nom del servidor o l'adreça IP a continuació.\n\nSi feu servir un hostatge web compartit, el vostre proveïdor us hauria de proporcionar el nom del servidor a la documentació.\n\nSi feu servir MySQL, «localhost» podria no funcionar com a nom de servidor. Si no funciona, proveu «127.0.0.1» com a adreça IP local.\n\nSi feu servir PostgreSQL, deixeu aquest camp en blanc per a connectar-vos a través d'un sòcol Unix.",
        "config-postgres-old": "Cal el PostgreSQL $1 o posterior. Teniu el $2.",
        "config-mssql-old": "Cal utilitzar el Microsoft SQL Server $1 o posterior. Teniu la versió $2.",
        "config-sqlite-name-help": "Trieu un nom per identificar el wiki.\nNo feu servir espais ni guionets.\nAquest nom s’utilitzarà per a denominar el fitxer de les dades de l’SQLite.",
+       "config-sqlite-parent-unwritable-group": "No es pot crear el directori de dades <code><nowiki>$1</nowiki></code>, perquè el directori pare <code><nowiki>$2</nowiki></code> no el pot escriure el servidor web.\n\nL'instal·lador no pot determinar l'usuari amb què s'executa el servidor web.\nFeu el directori <code><nowiki>$3</nowiki></code> escrivible globalment per l'usuari del servidor web (i altres) per continuar.\nEn un sistema Unix/Linux feu:\n\n<pre>cd $2\nmkdir $3\nchgrp $4 $3\nchmod g+w $3</pre>",
+       "config-sqlite-parent-unwritable-nogroup": "No es pot crear el directori de dades <code><nowiki>$1</nowiki></code>, perquè el directori pare <code><nowiki>$2</nowiki></code> no el pot escriure el servidor web.\n\nL'instal·lador no pot determinar l'usuari amb què s'executa el servidor web.\nFeu el directori <code><nowiki>$3</nowiki></code> escrivible globalment per l'usuari del servidor web (i altres) per continuar.\nEn un sistema Unix/Linux feu:\n\n<pre>cd $2\nmkdir $3\nchmod a+w $3</pre>",
        "config-sqlite-mkdir-error": "S'ha produït un error en crear el directori de dades «$1».\nComproveu la ubicació i torneu-ho a provar.",
        "config-sqlite-dir-unwritable": "No s'ha pogut escriure al directori «$1».\nCanvieu els permisos perquè el servidor web pugui escriure-hi i torneu-ho a provar.",
        "config-sqlite-connection-error": "$1. \n\nComproveu el directori de dades i el nom de la base de dades a continuació i torneu-ho a provar.",
        "config-admin-error-password": "S'ha produït un error intern en definir una contrasenya per a l'administrador «<nowiki>$1</nowiki>»: <pre>$2</pre>",
        "config-admin-error-bademail": "Heu introduït una adreça electrònica no vàlida.",
        "config-subscribe": "Subscriu a la [https://lists.wikimedia.org/mailman/listinfo/mediawiki-announce llista de correu d'anunci de noves versions].",
+       "config-subscribe-help": "És una llista de poc volum que s'utilitza per fer anuncis de noves versions, incloent-hi aquells de seguretat importants.\nCal que us hi subscriviu i actualitzeu la instal·lació de MediaWiki quan apareguin noves versions.",
        "config-subscribe-noemail": "Us heu provat de subscriure a la llista de correu d'anuncis de noves versions sense proporcionar-hi una adreça electrònica.\nProporcioneu-ne una si voleu subscriure-us a la llista de correu electrònic.",
        "config-pingback": "Comparteix dades d'aquesta instal·lació amb els desenvolupadors de MediaWiki.",
+       "config-pingback-help": "Si seleccioneu aquesta opció, el MediaWiki farà ping periòdicament a https://www.mediawiki.org amb dades bàsiques d'aquesta instància. Les dades inclouen, per exemple, el tipus de sistema, la versió PHP i el sistema de bases de dades que s'utilitza. La Fundació Wikimedia comparteix les dades amb els desenvolupadors de MediaWiki per tal d'ajudar-los a guiar empreses de desenvolupament futures. Les dades següents s'enviaran del vostre sistema:\n<pre>$1</pre>",
        "config-almost-done": "Gairebé ja heu acabat!\nPodeu ometre el que queda de la configuració i procedir amb la instal·lació del wiki.",
        "config-optional-continue": "Fes-me més preguntes.",
        "config-optional-skip": "Ja estic avorrit. Simplement instal·leu el wiki.",
index 7d361bc..3a869cf 100644 (file)
@@ -11,7 +11,8 @@
                        "Macofe",
                        "Guycn2",
                        "שמזן",
-                       "חיים"
+                       "חיים",
+                       "Steeve815"
                ]
        },
        "config-desc": "תכנית ההתקנה של מדיה־ויקי",
        "config-welcome": "=== בדיקות סביבה ===\nבדיקות בסיסיות תתבצענה עכשיו כדי לראות אם הסביבה הזאת מתאימה להתקנת מדיה־ויקי.\nנא לזכור לכלול את המידע הזה בעת בקשת תמיכה עם השלמת ההתקנה.",
        "config-welcome-section-copyright": "=== זכויות יוצרים ותנאים ===\n\n$1\n\nתכנית זו היא תכנה חופשית; באפשרותך להפיצה מחדש ו/או לשנות אותה על פי תנאי הרישיון הציבורי הכללי של GNU כפי שפורסם על ידי קרן התכנה החופשית; בין אם גרסה 2 של הרישיון, ובין אם (לפי בחירתך) כל גרסה מאוחרת שלו.\n\nתכנית זו מופצת בתקווה שתהיה מועילה, אבל '''בלא אחריות כלשהי'''; ואפילו ללא האחריות המשתמעת בדבר '''מסחריותה''' או '''התאמתה למטרה '''מסוימת'''. לפרטים נוספים, ניתן לעיין ברישיון הציבורי הכללי של GNU.\n\nלתכנית זו אמור היה להיות מצורף [$2 עותק של הרישיון הציבורי הכללי של GNU]; אם לא קיבלת אותו, אפשר לכתוב ל־Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA או [https://www.gnu.org/copyleft/gpl.html לקרוא אותו דרך האינטרנט].",
        "config-sidebar": "* [https://www.mediawiki.org אתר הבית של מדיה־ויקי]\n* [https://www.mediawiki.org/wiki/Special:MyLanguage/Help:Contents המדריך למשתמש]\n* [https://www.mediawiki.org/wiki/Special:MyLanguage/Manual:Contents המדריך למנהל]\n* [https://www.mediawiki.org/wiki/Special:MyLanguage/Manual:FAQ שו״ת]\n----\n* <doclink href=Readme>קרא אותי</doclink>\n* <doclink href=ReleaseNotes>הערות גרסה</doclink>\n* <doclink href=Copying>העתקה</doclink>\n* <doclink href=UpgradeDoc>שדרוג</doclink>",
+       "config-sidebar-readme": "קרא־אותי",
+       "config-sidebar-relnotes": "הערות גרסה",
+       "config-sidebar-license": "העתקה",
+       "config-sidebar-upgrade": "שדרוג",
        "config-env-good": "הסביבה שלכם נבדקה.\nאפשר להתקין מדיה־ויקי.",
        "config-env-bad": "הסביבה שלכם נבדקה.\nאי־אפשר להתקין מדיה־ויקי.",
        "config-env-php": "מותקנת <span dir=\"ltr\">PHP $1</span>.",
        "config-env-hhvm": "מותקנת <span dir=\"ltr\">HHVM $1</span>.",
-       "config-unicode-using-intl": "משתמש ב[https://pecl.php.net/intl הרחבת intl PECL] לנרמול יוניקוד.",
+       "config-unicode-using-intl": "משתמש ב[https://php.net/manual/en/book.intl.php הרחבת PHP intl] לנרמול יוניקוד.",
        "config-unicode-pure-php-warning": "<strong>אזהרה:</strong> [https://pecl.php.net/intl הרחבת intl PECL] אינה זמינה לטיפול בנרמול יוניקוד. משתמש ביישום PHP טהור ואטי יותר.\nאם זהו אתר בעל תעבורה גבוהה, כדאי לקרוא את המסמך הבא: [https://www.mediawiki.org/wiki/Special:MyLanguage/Unicode_normalization_considerations Unicode normalization].",
        "config-unicode-update-warning": "'''אזהרה''': הגרסה המותקנת של מעטפת נרמול יוניקוד משתמשת בגרסה ישנה של הספרייה של [http://site.icu-project.org/ פרויקט ICU].\nכדאי [https://www.mediawiki.org/wiki/Special:MyLanguage/Unicode_normalization_considerations לעדכן] אם הטיפול ביוניקוד חשוב לך.",
        "config-no-db": "לא נמצא דרייבר מסד נתונים מתאים. יש להתקין דרייבר מסד נתונים ל־PHP.\n{{PLURAL:$2|נתמך הסוג הבא של מסד נתונים|נתמכים הסוגים הבאים של מסדי נתונים}}: $1.\n\nאם קִמפלת את PHP בעצמך, יש להגדיר אותו מחדש ולהפעיל את לקוח מסד נתונים, למשל באמצעות <code dir=\"ltr\">./configure --with-mysqli</code>.\nאם התקנת את PHP מחבילה של דביאן או של אובונטו, יש להתקין, למשל, גם את המודול <code dir=\"ltr\">php-mysql</code>.",
        "config-install-done": "<strong>מזל טוב!</strong>\nהתקנת את תוכנת מדיה־ויקי.\n\nתוכנת ההתקנה יצרה את הקובץ <code>LocalSettings.php</code>.\nהוא מכיל את כל ההגדרות שלך.\n\nיש להוריד אותו ולהכניס אותו לתיקיית הבסיס שבה הותקן הוויקי שלך (אותה התיקייה שבה נמצא הקובץ index.php). ההורדה אמורה להתחיל באופן אוטומטי.\n\nאם ההורדה לא התחילה, או אם ביטלת אותה, אפשר להתחיל אותה מחדש באמצעות לחיצה על הקישור הבא:\n\n$3\n\n<strong>לתשומת לבך:</strong> אם ההורדה לא תבוצע כעת, קובץ ההגדרות <strong>לא</strong> יהיה זמין מאוחר יותר אם תוכנת ההתקנה תיסגר לפני שהקובץ יורד.\n\nלאחר שביצעת את הפעולות שלהלן, באפשרותך <strong>[$2 להיכנס לאתר הוויקי שלך]</strong>.",
        "config-install-done-path": "<strong>מזל טוב!</strong>\nהתקנת את תוכנת מדיה־ויקי.\n\nתוכנת ההתקנה יצרה את הקובץ <code>LocalSettings.php</code>.\nהוא מכיל את כל ההגדרות שלך.\n\nיש להוריד אותו ולהכניס אותו לתיקייה <code>$4</code>. ההורדה אמורה להתחיל באופן אוטומטי.\n\nאם ההורדה לא התחילה, או אם ביטלת אותה, אפשר להתחיל אותה מחדש באמצעות לחיצה על הקישור הבא:\n\n$3\n\n<strong>לתשומת לבך:</strong> אם ההורדה לא תבוצע כעת, קובץ ההגדרות <strong>לא</strong> יהיה זמין מאוחר יותר אם תוכנת ההתקנה תיסגר לפני שהקובץ יורד.\n\nלאחר שביצעת את הפעולות שלהלן, באפשרותך <strong>[$2 להיכנס לאתר הוויקי שלך]</strong>.",
        "config-install-success": "מדיה־ויקי הותקנה בהצלחה. עכשיו אפשר\nלבקר בכתובת <$1$2> כדי לצפות בוויקי שלך.\nאם יש לך שאלות, ר' את רשימת השאלות הנפוצות שלנו:\n<https://www.mediawiki.org/wiki/Special:MyLanguage/Manual:FAQ> ואפשר גם להשתמש\nבאתרי התמיכה שקישורים אליהם מופיעים באותו הדף.",
+       "config-install-db-success": "בסיס הנתונים הוגדר בהצלחה",
        "config-download-localsettings": "הורדת <code>LocalSettings.php</code>",
        "config-help": "עזרה",
        "config-help-tooltip": "להרחיב",
index 7fd77ee..2a98729 100644 (file)
@@ -15,7 +15,8 @@
                        "Macofe",
                        "Rachmat.Wahidi",
                        "Gombang",
-                       "Rachmat04"
+                       "Rachmat04",
+                       "ArlandGa"
                ]
        },
        "config-desc": "Penginstal untuk MediaWiki",
@@ -55,7 +56,7 @@
        "config-help-restart": "Apakah Anda ingin menghapus semua data tersimpan yang telah Anda masukkan dan mengulang proses instalasi?",
        "config-restart": "Ya, nyalakan ulang",
        "config-welcome": "=== Pengecekan lingkungan ===\nPengecekan dasar kini akan dilakukan untuk melihat apakah lingkungan ini memadai untuk instalasi MediaWiki.\nIngatlah untuk menyertakan informasi ini jika Anda mencari bantuan tentang cara menyelesaikan instalasi.",
-       "config-welcome-section-copyright": "=== Hak cipta dan persyaratan ===\n\n$1\n\nProgram ini adalah perangkat lunak bebas; Anda dapat mendistribusikan dan/atau memodifikasi di bawah persyaratan GNU General Public License seperti yang diterbitkan oleh Free Software Foundation; baik versi 2 lisensi, atau (sesuai pilihan Anda) versi yang lebih baru.\n\nProgram ini didistribusikan dengan harapan bahwa itu akan berguna, tetapi <strong>tanpa jaminan apa pun</strong>; bahkan tanpa jaminan tersirat untuk <strong>dapat diperjualbelikan</strong> atau <strong>sesuai untuk tujuan tertentu</strong>.\nLihat GNU General Public License untuk lebih jelasnya.\n\nAnda seharusnya telah menerima [$2 salinan dari GNU General Public License] bersama dengan program ini; jika tidak, kirimkan surat untuk Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA, atau [https://www.gnu.org/copyleft/gpl.html baca versi daring].",
+       "config-welcome-section-copyright": "=== Hak cipta dan persyaratan ===\n\n$1\n\nProgram ini adalah perangkat lunak bebas; Anda dapat mendistribusikan dan/atau memodifikasinya di bawah persyaratan GNU General Public License seperti yang diterbitkan oleh Free Software Foundation; baik versi 2 lisensi, atau (sesuai pilihan Anda) versi yang lebih baru.\n\nProgram ini didistribusikan dengan harapan bahwa itu akan berguna, tetapi <strong>tanpa jaminan apa pun</strong>; bahkan tanpa jaminan tersirat untuk <strong>dapat diperjualbelikan</strong> atau <strong>sesuai untuk tujuan tertentu</strong>.\nLihat GNU General Public License untuk lebih jelasnya.\n\nAnda seharusnya telah menerima [$2 salinan dari GNU General Public License] bersama dengan program ini; jika tidak, kirimkan surat untuk Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA, atau [https://www.gnu.org/copyleft/gpl.html baca versi daring].",
        "config-sidebar": "* [https://www.mediawiki.org/wiki/MediaWiki/id Situs MediaWiki]\n* [https://www.mediawiki.org/wiki/Special:MyLanguage/Help:Contents/id Pedoman Pengguna]\n* [https://www.mediawiki.org/wiki/Special:MyLanguage/Manual:Contents/id Pedoman Administrator]\n* [https://www.mediawiki.org/wiki/Special:MyLanguage/Manual:FAQ/id FAQ]\n----\n* <doclink href=Readme>Read me</doclink>\n* <doclink href=ReleaseNotes>Release notes</doclink>\n* <doclink href=Copying>Copying</doclink>\n* <doclink href=UpgradeDoc>Upgrading</doclink>",
        "config-env-good": "Kondisi telah diperiksa.\nAnda dapat menginstal MediaWiki.",
        "config-env-bad": "Kondisi telah diperiksa.\nAnda tidak dapat menginstal MediaWiki.",
index 9516433..55e8d78 100644 (file)
        "config-restart": "はい、再起動します",
        "config-welcome": "=== 環境の確認 ===\n基本的な確認では、現在の環境が MediaWiki のインストールに適しているかを確認します。\nインストール方法について助けが必要になった場合は、必ずこの確認結果を添えてください。",
        "config-welcome-section-copyright": "=== 著作権および規約 ===\n$1\n\nこの作品はフリーソフトウェアです。あなたは、フリーソフトウェア財団の発行する GNU 一般公衆利用許諾書 (GNU General Public License) (バージョン 2、またはそれ以降のライセンス) の規約に基づき、このライブラリを再配布および改変できます。\n\nこの作品は、有用であることを期待して配布されていますが、<strong>商用または特定の目的に適するかどうか</strong>も含めて、暗黙的にも、<strong>一切保証されません</strong>。\n詳しくは、 GNU 一般公衆利用許諾書をご覧ください。\n\nあなたはこのプログラムと共に、[$2 GNU 一般公衆利用許諾契約書の複製]を受け取ったはずです。受け取っていない場合は、フリーソフトウェア財団 (宛先は the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA) まで請求するか、または[https://www.gnu.org/copyleft/gpl.html オンラインでお読みください]。",
-       "config-sidebar": "* [https://www.mediawiki.org MediaWikiのホーム]\n* [https://www.mediawiki.org/wiki/Special:MyLanguage/Help:Contents 利用者向け案内]\n* [https://www.mediawiki.org/wiki/Special:MyLanguage/Manual:Contents 管理者向け案内]\n* [https://www.mediawiki.org/wiki/Special:MyLanguage/Manual:FAQ FAQ]\n----\n* <doclink href=Readme>お読みください</doclink>\n* <doclink href=ReleaseNotes>リリースノート</doclink>\n* <doclink href=Copying>コピー</doclink>\n* <doclink href=UpgradeDoc>アップグレード</doclink>",
+       "config-sidebar": "* [https://www.mediawiki.org MediaWikiのホーム]\n* [https://www.mediawiki.org/wiki/Special:MyLanguage/Help:Contents 利用者向け案内]\n* [https://www.mediawiki.org/wiki/Special:MyLanguage/Manual:Contents 管理者向け案内]\n* [https://www.mediawiki.org/wiki/Special:MyLanguage/Manual:FAQ FAQ]",
+       "config-sidebar-readme": "お読みください",
+       "config-sidebar-relnotes": "リリースノート",
+       "config-sidebar-license": "複製",
+       "config-sidebar-upgrade": "アップグレード",
        "config-env-good": "環境を確認しました。\nMediaWiki をインストールできます。",
        "config-env-bad": "環境を確認しました。\nMediaWiki のインストールはできません。",
        "config-env-php": "PHP $1がインストールされています。",
index c718b75..06090ca 100644 (file)
@@ -58,7 +58,7 @@
        "config-unicode-pure-php-warning": "'''Upozorenje''': Dodatak [https://pecl.php.net/intl intl PECL] nije dostupan za normalizaciju Unicode, vraćajući se na sporu primjenu čistog PHP-a.\n\nAko imate web-lokaciju s visokim prometom, morat ćete pročitati više o [https://www.mediawiki.org/wiki/Special:MyLanguage/Unicode_normalization_considerations Unicode normalizacije].",
        "config-unicode-update-warning": "'''Upozorenje:''' Uspostavljena verzija omotnice Unicode normalizacije koristi stariju verziju biblioteke [http://site.icu-project.org/ projekta ICU].\nDa biste koristili Unicode, trebate napraviti [https://www.mediawiki.org/wiki/Special:MyLanguage/Unicode_normalization_considerations nadogradnju].",
        "config-no-db": "Nisam mogao pronaći odgovarajući upravljački program za bazu podataka! Morat ćete uspostaviti upravljački program za PHP-bazu.\n{{PLURAL:$2|Podržana je sljedeća vrsta|Podržane su sljedeće vrste}} baze: $1.\n\nAko ste sami kompilirali PHP, omogućite bazni klijent u postavkama - npr. s <code>./configure --with-mysqli</code>.\nAko ovaj PHP uspostavite iz Debian ili Ubuntu paketa, tada ćete ga morati uspostaviti, npr., paket <code>php-mysql</code>.",
-       "config-outdated-sqlite": "'''Upozorenje''': imate SQLite $1. Najstarija dopuštena verzija je $2. Stoga će SQLite biti nedostupan.",
+       "config-outdated-sqlite": "<strong>Upozorenje</strong>: imate SQLite $2. Najstarija dopuštena verzija je $1. Stoga, SQLite će biti nedostupan.",
        "config-no-fts3": "'''Upozorenje''': SQLite se kompilira bez modula [//sqlite.org/fts3.html FTS3] - za tu bazu podataka neće biti mogućnosti pretrage.",
        "config-pcre-old": "'''Kobno:''' Potreban je PCRE $1 ili novija verzija.\nVaš PHP-binarni je svezan s PCRE $2.\n[https://www.mediawiki.org/wiki/Manual:Errors_and_symptoms/PCRE Više informacija].",
        "config-pcre-no-utf8": "<strong>Kobno</strong>: PCRE modul PHP-a je hitan bez podrške za PCRE_UTF8.\nMediaWiki zahtijeva podršku za UTF-8 kako bi ispravno funkcionirao.",
        "config-mssql-windowsauth": "Potvrda identiteta za Windows",
        "config-site-name": "Ime wikija:",
        "config-site-name-help": "Ovo će se pojaviti u naslovnoj traci pregledača i na raznim drugim mestima.",
+       "config-project-namespace": "Projektni imenski prostor:",
+       "config-ns-generic": "Projekat",
+       "config-ns-site-name": "Isto ime kao wikija: $1",
+       "config-ns-other": "Drugo (navedite)",
+       "config-ns-other-default": "MyWiki",
+       "config-admin-box": "Administratoski račun",
+       "config-admin-name": "Vaše korisničko ime:",
        "config-admin-password": "Lozinka:",
+       "config-admin-password-confirm": "Ponovite lozinku:",
+       "config-admin-name-blank": "Upišite administratorsko korisničko ime.",
+       "config-admin-password-blank": "Upišite lozinku za administratorski račun",
+       "config-admin-password-mismatch": "Lozinke što ste upisali se ne poklapaju.",
+       "config-admin-email": "E-mail adresa:",
+       "config-profile": "Profil korisničkih prava:",
+       "config-profile-wiki": "Otvoren wiki",
+       "config-profile-no-anon": "Neophodno otvaranje računa",
+       "config-profile-fishbowl": "Samo ovlašteni urednici",
+       "config-profile-private": "Privatan wiki",
+       "config-license": "Autorska prava i licenca:",
+       "config-license-none": "Bez podnožja za licencu",
+       "config-email-settings": "Podešavanja e-pošte",
+       "config-enable-email": "Omogući odlaznu e-poštu",
+       "config-email-user": "Omogući slanje e-poruka među korisnicima",
+       "config-email-user-help": "Dozvoli svim korisnicima da međusobno šalju e-poruke ako imaju omogućeno u podešavanjima.",
+       "config-email-usertalk": "Omogući obaveštenja o promjenama u korisničkim stranicama za razgovor",
+       "config-email-usertalk-help": "Omogući korisnicima da primaju obaveštenja o promenama u njihovim korisničkim razgovornim stranicama ako su ih omogućili u podešavanjima.",
+       "config-email-watchlist": "Omogući obaveštenja o spisku praćenja",
+       "config-email-watchlist-help": "Omogući korisnicima da primaju obaveštenja o svojim nadgledanim stranicama ako su ih omogućili u podešavanjima.",
+       "config-upload-settings": "Otpremanja slika i datoteka",
+       "config-upload-enable": "Omogući postavljanje datoteka",
+       "config-upload-deleted": "Folder za obrisane datoteke:",
+       "config-upload-deleted-help": "Odaberite u kojem folderu će se arhivirati izbrisane datoteke.\nNajbolje bi bilo ako taj nije dostupan putem svemrežja.",
+       "config-logo": "URL za logotipa:",
+       "config-instantcommons": "Omogući Instant Commons",
+       "config-cc-again": "Odaberite ponovo...",
+       "config-cc-not-chosen": "Odaberite željenu licencu Creative Commons i kliknite na „proceed”.",
+       "config-advanced-settings": "Napredna podešavanja",
+       "config-cache-options": "Podešavanja za međuspremanje objekta:",
+       "config-cache-none": "Nema međuspremanja (ne uklanja se nijedna funkcija, ali može uticati na brzinu na veće wiki lokacije)",
+       "config-cache-accel": "Međuspremanje PHP-objekta (APC, APCu ili WinCache)",
+       "config-cache-memcached": "Koristi Memcached (zahtijeva dodatno postavljivanje i podešavanje)",
+       "config-memcached-servers": "Memcached-serveri:",
+       "config-memcached-help": "Lista IP adresa za uporabu u Memcached.\nTreba da se navede jednu u svaki red, kao i port što će se koristiti. Na primer:\n 127.0.0.1:11211\n 192.168.1.25:1234",
+       "config-memcache-needservers": "Odabrali ste Memcached kao vaš tip međuspremnika (keša), ali niste naveli nijedan server.",
        "mainpagetext": "<strong>MediaWiki je uspješno instaliran.</strong>",
        "mainpagedocfooter": "Za informacije o korištenju wiki softvera konzultirajte [https://meta.wikimedia.org/wiki/Help:Contents Vodič za korisnike].\n\n== Uvod u rad ==\n* [https://www.mediawiki.org/wiki/Special:MyLanguage/Manual:Configuration_settings Lista konfiguracije postavki]\n* [https://www.mediawiki.org/wiki/Special:MyLanguage/Manual:FAQ MediaWiki FAQ]\n* [https://lists.wikimedia.org/mailman/listinfo/mediawiki-announce Lista primatelja izdanja MediaWikija]\n* [https://www.mediawiki.org/wiki/Special:MyLanguage/Localisation#Translation_resources Lokalizirajte MediaWiki za svoj jezik]\n* [https://www.mediawiki.org/wiki/Special:MyLanguage/Manual:Combating_spam Saznajte kako se boriti protiv spama na svojem wikiju]"
 }
index 85aa98f..d9c615d 100644 (file)
@@ -7,16 +7,22 @@
                        "Ерней"
                ]
        },
-       "config-desc": "MediaWiki йөкләүче",
-       "config-title": "MediaWiki $1 куелышы",
+       "config-desc": "MediaWiki урнаштыручы",
+       "config-title": "MediaWiki $1 урнаштыруы",
        "config-information": "Мәгълүмат",
+       "config-localsettings-upgrade": "<code>LocalSettings.php</code> файлы табылды.\nБу урнаштыру яңартырга өчен, түбәндәге кырда <code>$wgUpgradeKey</code> кыйммәтен кертегез әле.\nСез аны <code>LocalSettings.php</code> файлында табасыз.",
+       "config-localsettings-cli-upgrade": "<code>LocalSettings.php</code> файлы табылды.\nБу урнаштыру яңартыга өчен, <code>update.php</code> җибәрегез әле",
        "config-localsettings-key": "Яңарту ачкычы:",
+       "config-localsettings-badkey": "Сез ялгыш яңарту ачкычы бирдегез.",
        "config-your-language": "Телегез:",
        "config-wiki-language": "Вики теле:",
-       "config-back": "â\86\90 Ð\90Ñ\80Ñ\82ка",
-       "config-continue": "Ð\9aилÓ\99Ñ\81е →",
+       "config-back": "â\86\90 Ð\9aиÑ\80егÓ\99",
+       "config-continue": "Ð\94Ó\99вам Ð¸Ñ\82Ò¯ →",
        "config-page-language": "Тел",
        "config-page-welcome": "MediaWiki проектына рәхим итегез!",
+       "config-page-dbconnect": "Мәгълүматлар базасына тоташтыру",
+       "config-page-upgrade": "Булган урнаштыруны яңарту",
+       "config-page-dbsettings": "Мәгълүматлар базасы көйләнмәләре",
        "config-page-name": "Исем",
        "config-page-options": "Көйләнмәләр",
        "config-page-install": "Урнаштыру",
        "config-page-copying": "Лицензия",
        "config-page-upgradedoc": "Яңарту",
        "config-page-existingwiki": "Хәзерге вики",
-       "config-restart": "Әйе, яңадан башларга",
+       "config-restart": "Әйе, яңадан башлау",
+       "config-sidebar-readme": "Укып чык",
+       "config-sidebar-relnotes": "Чыгарыш турында мәгълүмат",
+       "config-sidebar-license": "Күчермә алу",
+       "config-sidebar-upgrade": "Яңарту",
        "config-env-php": "PHP $1 куелды.",
        "config-env-hhvm": "HHVM $1 куелды.",
        "config-apc": "[https://www.php.net/apc APC] куелды",
+       "config-apcu": "[https://www.php.net/apcu APCu] куелды",
        "config-wincache": "[https://www.iis.net/downloads/microsoft/wincache-extension WinCache] куелды",
        "config-diff3-bad": "GNU diff3 табылмады.",
        "config-git": "Git юрамалар идарә итү системасы табылды: <code>$1</code>.",
        "config-using-server": "«<nowiki>$1</nowiki>» сервер исеме файдаланыла.",
        "config-using-uri": "«<nowiki>$1$2</nowiki>» URL исемле сервер файдаланыла.",
-       "config-db-type": "Мәгълүмат базасы төре:",
-       "config-db-host": "Мәгълүмат базасы хосты:",
+       "config-db-type": "Мәгълүматлар базасы төре:",
+       "config-db-host": "Мәгълүматлар базасы хосты:",
        "config-db-host-oracle": "TNS мәгълүмат базасы:",
-       "config-db-wiki-settings": "Бу вики идентификациясе",
+       "config-db-wiki-settings": "Бу вики тәңгәлләштерү",
        "config-db-name": "Мәгълүматлар базасы исеме (сызыкчасыз):",
-       "config-db-name-oracle": "Мәгълүмат базасы төзелеше:",
+       "config-db-name-oracle": "Мәгълүматлар базасы төзелеше:",
        "config-db-username": "Мәгълүмат базасын кулланучы исеме:",
        "config-db-password": "Мәгълүмат базасының серсүзе:",
-       "config-db-port": "Мәгълүмат базасы порты:",
+       "config-db-port": "Мәгълүматлар базасы порты:",
        "config-db-schema": "MediaWiki өчен (сызыкчасыз) төзелеш:",
-       "config-type-mysql": "MariaDB, MySQL Ñ\8fиÑ\81Ó\99 ярашлы",
+       "config-type-mysql": "MariaDB, MySQL Ñ\8fки ярашлы",
        "config-type-mssql": "Microsoft SQL Server",
        "config-header-mysql": "MariaDB/MySQL көйләнмәләре",
        "config-header-postgres": "PostgreSQL көйләнмәләре",
        "config-header-sqlite": "SQLite көйләнмәләре",
        "config-header-oracle": "Oracle көйләнмәләре",
        "config-header-mssql": "Microsoft SQL Server көйләнмәләре",
-       "config-invalid-db-type": "Ялган Ð±Ð¸Ñ\80елмÓ\99лÓ\99Ñ\80 Ð±Ð°Ð·Ð°Ñ\81Ñ\8b Ñ\82Ó©Ñ\80е",
+       "config-invalid-db-type": "Ð\9cÓ\99гÑ\8aлүмаÑ\82лаÑ\80 Ð±Ð°Ð·Ð°Ñ\81Ñ\8b Ñ\8fлгÑ\8bÑ\88 Ñ\82Ó©Ñ\80е.",
        "config-upgrade-done-no-regenerate": "Яңартү тәмамланды.\n\nХәзер сез [$1 вики] белән эшли аласыз.",
        "config-regenerate": "LocalSettings.php яңадан төзү →",
        "config-show-table-status": "«<code>SHOW TABLE STATUS</code>» таләбе эшләнмәде!",
        "config-mysql-engine": "Саклау системасы:",
-       "config-mysql-innodb": "InnoDB (тәкъдим ителә)",
+       "config-mysql-innodb": "InnoDB (киңәш ителә)",
        "config-mssql-auth": "Аутентификация төре:",
        "config-mssql-sqlauth": "SQL Server чынлыгын раслау",
        "config-mssql-windowsauth": "Windows чынлыгын раслау",
        "config-admin-password": "Серсүз:",
        "config-admin-password-confirm": "Серсүзне кабатлагыз:",
        "config-admin-email": "Электрон почта адресы:",
+       "config-profile": "Кулланучы хокуклары профиле:",
        "config-profile-wiki": "Ачык вики",
        "config-profile-private": "Ябык вики",
-       "config-license": "Автор хокуклары һәм лицензияләр:",
+       "config-license": "Авторлык хокукы һәм рөхсәтнамә:",
        "config-license-cc-by-sa": "Creative Commons Attribution Share Alike",
        "config-license-cc-by": "Creative Commons Attribution",
        "config-license-cc-by-nc-sa": "Creative Commons Attribution Non-Commercial Share Alike",
        "config-license-cc-0": "Creative Commons Zero (җәмгыять мирасы)",
        "config-license-gfdl": "GNU Free Documentation License 1.3 яки яңарагы",
        "config-license-pd": "Җәмгыять мирасы",
+       "config-email-settings": "E-mail көйләнмәләре",
        "config-logo": "Логотип URL:",
-       "config-cc-again": "Кабат сайлагыз...",
+       "config-cc-again": "Кабат сайлагыз",
        "config-advanced-settings": "Өстәмә көйләнмәләр",
        "config-memcached-servers": "Memcached серверлары:",
        "config-extensions": "Киңәйтүләр",
        "config-skins": "Бизәлеш",
-       "config-install-step-done": "әзер",
+       "config-install-step-done": "тәмам",
        "config-install-step-failed": "булмады",
+       "config-install-database": "Мәгълүматлар базасы көйләве",
        "config-install-schema": "Схема төзү",
-       "config-install-tables": "Табын төзү",
+       "config-install-user-alreadyexists": "Кулланучы «$1» бар инде",
+       "config-install-user-create-failed": "«$1» кулланучыны китереп булмады: $2",
+       "config-install-tables": "Җәдвәлләр төзү",
        "config-install-stats": "Инициализация статистикасы",
        "config-download-localsettings": "<code>LocalSettings.php</code> йөкләү",
-       "config-help": "ярдәм",
+       "config-help": "белешмә",
        "config-help-tooltip": "ачу өчен басыгыз",
        "config-skins-screenshots": "$1 (скриншотлар: $2)",
        "config-extensions-requires": "$1 ($2 кирәк)",
index 0443cc4..28774f8 100644 (file)
@@ -21,7 +21,8 @@
                        "Wehwei",
                        "Wwycheuk",
                        "蘭斯特",
-                       "Kly"
+                       "Kly",
+                       "Winston Sung"
                ]
        },
        "config-desc": "MediaWiki 安裝程式",
        "config-memcache-badport": "Memcached 埠號應介於 $1 到 $2 之間。",
        "config-extensions": "擴充套件",
        "config-extensions-help": "已在您的 <code>./extensions</code> 目錄中發現下列擴充套件。\n\n這些擴充套件可能需要做額外的設定,但您可以現在先開啟功能。",
-       "config-skins": "外觀",
+       "config-skins": "佈景主題",
        "config-skins-help": "系統偵測到您於 <code>./skins</code> 資料夾中含有外觀如上清單。 您必須開啟其中一項並設為預設值。",
        "config-skins-use-as-default": "使用此外觀作為預設",
        "config-skins-missing": "沒有發現任何外觀;MediaWiki 在您安裝一些恰當的外觀前將會使用備用外觀。",
index e2914be..9519b7f 100644 (file)
@@ -76,7 +76,9 @@ class AssembleUploadChunksJob extends Job {
 
                        // We can only get warnings like 'duplicate' after concatenating the chunks
                        $status = Status::newGood();
-                       $status->value = [ 'warnings' => $upload->checkWarnings() ];
+                       $status->value = [
+                               'warnings' => UploadBase::makeWarningsSerializable( $upload->checkWarnings() )
+                       ];
 
                        // We have a new filekey for the fully concatenated file
                        $newFileKey = $upload->getStashFile()->getFileKey();
index 5e8c22b..37d53e2 100644 (file)
@@ -150,12 +150,12 @@ abstract class DBLockManager extends QuorumLockManager {
                        } elseif ( is_array( $this->dbServers[$lockDb] ) ) {
                                // Parameters to construct a new database connection
                                $config = $this->dbServers[$lockDb];
+                               $config['flags'] = ( $config['flags'] ?? 0 );
+                               $config['flags'] &= ~( IDatabase::DBO_TRX | IDatabase::DBO_DEFAULT );
                                $db = Database::factory( $config['type'], $config );
                        } else {
                                throw new UnexpectedValueException( "No server called '$lockDb'." );
                        }
-
-                       $db->clearFlag( DBO_TRX );
                        # If the connection drops, try to avoid letting the DB rollback
                        # and release the locks before the file operations are finished.
                        # This won't handle the case of DB server restarts however.
index e742432..23cf607 100644 (file)
@@ -798,8 +798,8 @@ abstract class MediumSpecificBagOStuff extends BagOStuff {
         *   - positive (< 10 years): relative TTL; return UNIX timestamp offset by this value
         *   - positive (>= 10 years): absolute UNIX timestamp; return this value
         *
-        * @param int $exptime Absolute TTL or 0 for indefinite
-        * @return int
+        * @param int $exptime
+        * @return int Absolute TTL or 0 for indefinite
         */
        final protected function convertToExpiry( $exptime ) {
                return $this->expiryIsRelative( $exptime )
@@ -811,11 +811,17 @@ abstract class MediumSpecificBagOStuff extends BagOStuff {
         * Convert an optionally absolute expiry time to a relative time. If an
         * absolute time is specified which is in the past, use a short expiry time.
         *
+        * The input value will be cast to an integer and interpreted as follows:
+        *   - zero: no expiry; return zero (e.g. TTL_INDEFINITE)
+        *   - negative: relative TTL; return a short expiry time (1 second)
+        *   - positive (< 10 years): relative TTL; return this value
+        *   - positive (>= 10 years): absolute UNIX timestamp; return offset to current time
+        *
         * @param int $exptime
-        * @return int
+        * @return int Relative TTL or 0 for indefinite
         */
        final protected function convertToRelative( $exptime ) {
-               return $this->expiryIsRelative( $exptime )
+               return $this->expiryIsRelative( $exptime ) || !$exptime
                        ? (int)$exptime
                        : max( $exptime - (int)$this->getCurrentTime(), 1 );
        }
diff --git a/includes/libs/objectcache/WANObjectCache.php b/includes/libs/objectcache/WANObjectCache.php
deleted file mode 100644 (file)
index 69edb11..0000000
+++ /dev/null
@@ -1,2628 +0,0 @@
-<?php
-/**
- * This program is free software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License as published by
- * the Free Software Foundation; either version 2 of the License, or
- * (at your option) any later version.
- *
- * This program is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU General Public License for more details.
- *
- * You should have received a copy of the GNU General Public License along
- * with this program; if not, write to the Free Software Foundation, Inc.,
- * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
- * http://www.gnu.org/copyleft/gpl.html
- *
- * @file
- * @ingroup Cache
- */
-
-use Liuggio\StatsdClient\Factory\StatsdDataFactoryInterface;
-use Psr\Log\LoggerAwareInterface;
-use Psr\Log\LoggerInterface;
-use Psr\Log\NullLogger;
-
-/**
- * Multi-datacenter aware caching interface
- *
- * ### Using WANObjectCache
- *
- * All operations go to the local datacenter cache, except for delete(),
- * touchCheckKey(), and resetCheckKey(), which broadcast to all datacenters.
- *
- * This class is intended for caching data from primary stores.
- * If the get() method does not return a value, then the caller
- * should query the new value and backfill the cache using set().
- * The preferred way to do this logic is through getWithSetCallback().
- * When querying the store on cache miss, the closest DB replica
- * should be used. Try to avoid heavyweight DB master or quorum reads.
- *
- * To ensure consumers of the cache see new values in a timely manner,
- * you either need to follow either the validation strategy, or the
- * purge strategy.
- *
- * The validation strategy refers to the natural avoidance of stale data
- * by one of the following means:
- *
- *   - A) The cached value is immutable.
- *        If the consumer has access to an identifier that uniquely describes a value,
- *        cached value need not change. Instead, the key can change. This also allows
- *        all servers to access their perceived current version. This is important
- *        in context of multiple deployed versions of your application and/or cross-dc
- *        database replication, to ensure deterministic values without oscillation.
- *   - B) Validity is checked against the source after get().
- *        This is the inverse of A. The unique identifier is embedded inside the value
- *        and validated after on retreival. If outdated, the value is recomputed.
- *   - C) The value is cached with a modest TTL (without validation).
- *        If value recomputation is reasonably performant, and the value is allowed to
- *        be stale, one should consider using TTL only – using the value's age as
- *        method of validation.
- *
- * The purge strategy refers to the the approach whereby your application knows that
- * source data has changed and can react by purging the relevant cache keys.
- * As purges are expensive, this strategy should be avoided if possible.
- * The simplest purge method is delete().
- *
- * No matter which strategy you choose, callers must not rely on updates or purges
- * being immediately visible to other servers. It should be treated similarly as
- * one would a database replica.
- *
- * The need for immediate updates should be avoided. If needed, solutions must be
- * sought outside WANObjectCache.
- *
- * ### Deploying WANObjectCache
- *
- * There are two supported ways to set up broadcasted operations:
- *
- *   - A) Set up mcrouter as the underlying cache backend, using a memcached BagOStuff class
- *        for the 'cache' parameter. The 'region' and 'cluster' parameters must be provided
- *        and 'mcrouterAware' must be set to `true`.
- *        Configure mcrouter as follows:
- *          - 1) Use Route Prefixing based on region (datacenter) and cache cluster.
- *               See https://github.com/facebook/mcrouter/wiki/Routing-Prefix and
- *               https://github.com/facebook/mcrouter/wiki/Multi-cluster-broadcast-setup.
- *          - 2) To increase the consistency of delete() and touchCheckKey() during cache
- *               server membership changes, you can use the OperationSelectorRoute to
- *               configure 'set' and 'delete' operations to go to all servers in the cache
- *               cluster, instead of just one server determined by hashing.
- *               See https://github.com/facebook/mcrouter/wiki/List-of-Route-Handles.
- *   - B) Set up dynomite as a cache middleware between the web servers and either memcached
- *        or redis and use it as the underlying cache backend, using a memcached BagOStuff
- *        class for the 'cache' parameter. This will broadcast all key setting operations,
- *        not just purges, which can be useful for cache warming. Writes are eventually
- *        consistent via the Dynamo replication model. See https://github.com/Netflix/dynomite.
- *
- * Broadcasted operations like delete() and touchCheckKey() are intended to run
- * immediately in the local datacenter and asynchronously in remote datacenters.
- *
- * This means that callers in all datacenters may see older values for however many
- * milliseconds that the purge took to reach that datacenter. As with any cache, this
- * should not be relied on for cases where reads are used to determine writes to source
- * (e.g. non-cache) data stores, except when reading immutable data.
- *
- * All values are wrapped in metadata arrays. Keys use a "WANCache:" prefix
- * to avoid collisions with keys that are not wrapped as metadata arrays. The
- * prefixes are as follows:
- *   - a) "WANCache:v" : used for regular value keys
- *   - b) "WANCache:i" : used for temporarily storing values of tombstoned keys
- *   - c) "WANCache:t" : used for storing timestamp "check" keys
- *   - d) "WANCache:m" : used for temporary mutex keys to avoid cache stampedes
- *
- * @ingroup Cache
- * @since 1.26
- */
-class WANObjectCache implements IExpiringStore, IStoreKeyEncoder, LoggerAwareInterface {
-       /** @var BagOStuff The local datacenter cache */
-       protected $cache;
-       /** @var MapCacheLRU[] Map of group PHP instance caches */
-       protected $processCaches = [];
-       /** @var LoggerInterface */
-       protected $logger;
-       /** @var StatsdDataFactoryInterface */
-       protected $stats;
-       /** @var callable|null Function that takes a WAN cache callback and runs it later */
-       protected $asyncHandler;
-
-       /** @bar bool Whether to use mcrouter key prefixing for routing */
-       protected $mcrouterAware;
-       /** @var string Physical region for mcrouter use */
-       protected $region;
-       /** @var string Cache cluster name for mcrouter use */
-       protected $cluster;
-       /** @var bool Whether to use "interim" caching while keys are tombstoned */
-       protected $useInterimHoldOffCaching = true;
-       /** @var float Unix timestamp of the oldest possible valid values */
-       protected $epoch;
-       /** @var string Stable secret used for hasing long strings into key components */
-       protected $secret;
-
-       /** @var int Callback stack depth for getWithSetCallback() */
-       private $callbackDepth = 0;
-       /** @var mixed[] Temporary warm-up cache */
-       private $warmupCache = [];
-       /** @var int Key fetched */
-       private $warmupKeyMisses = 0;
-
-       /** @var float|null */
-       private $wallClockOverride;
-
-       /** @var int Max expected seconds to pass between delete() and DB commit finishing */
-       const MAX_COMMIT_DELAY = 3;
-       /** @var int Max expected seconds of combined lag from replication and view snapshots */
-       const MAX_READ_LAG = 7;
-       /** @var int Seconds to tombstone keys on delete() and treat as volatile after invalidation */
-       const HOLDOFF_TTL = self::MAX_COMMIT_DELAY + self::MAX_READ_LAG + 1;
-
-       /** @var int Idiom for getWithSetCallback() meaning "do not store the callback result" */
-       const TTL_UNCACHEABLE = -1;
-
-       /** @var int Consider regeneration if the key will expire within this many seconds */
-       const LOW_TTL = 30;
-       /** @var int Max TTL, in seconds, to store keys when a data sourced is lagged */
-       const TTL_LAGGED = 30;
-
-       /** @var int Expected time-till-refresh, in seconds, if the key is accessed once per second */
-       const HOT_TTR = 900;
-       /** @var int Minimum key age, in seconds, for expected time-till-refresh to be considered */
-       const AGE_NEW = 60;
-
-       /** @var int Idiom for getWithSetCallback() meaning "no cache stampede mutex required" */
-       const TSE_NONE = -1;
-
-       /** @var int Idiom for set()/getWithSetCallback() meaning "no post-expiration persistence" */
-       const STALE_TTL_NONE = 0;
-       /** @var int Idiom for set()/getWithSetCallback() meaning "no post-expiration grace period" */
-       const GRACE_TTL_NONE = 0;
-       /** @var int Idiom for delete()/touchCheckKey() meaning "no hold-off period" */
-       const HOLDOFF_TTL_NONE = 0;
-       /** @var int Alias for HOLDOFF_TTL_NONE (b/c) (deprecated since 1.34) */
-       const HOLDOFF_NONE = self::HOLDOFF_TTL_NONE;
-
-       /** @var float Idiom for getWithSetCallback() meaning "no minimum required as-of timestamp" */
-       const MIN_TIMESTAMP_NONE = 0.0;
-
-       /** @var string Default process cache name and max key count */
-       const PC_PRIMARY = 'primary:1000';
-
-       /** @var int Idion for get()/getMulti() to return extra information by reference */
-       const PASS_BY_REF = -1;
-
-       /** @var int Seconds to keep dependency purge keys around */
-       private static $CHECK_KEY_TTL = self::TTL_YEAR;
-       /** @var int Seconds to keep interim value keys for tombstoned keys around */
-       private static $INTERIM_KEY_TTL = 1;
-
-       /** @var int Seconds to keep lock keys around */
-       private static $LOCK_TTL = 10;
-       /** @var int Seconds to no-op key set() calls to avoid large blob I/O stampedes */
-       private static $COOLOFF_TTL = 1;
-       /** @var int Seconds to ramp up the chance of regeneration due to expected time-till-refresh */
-       private static $RAMPUP_TTL = 30;
-
-       /** @var float Tiny negative float to use when CTL comes up >= 0 due to clock skew */
-       private static $TINY_NEGATIVE = -0.000001;
-       /** @var float Tiny positive float to use when using "minTime" to assert an inequality */
-       private static $TINY_POSTIVE = 0.000001;
-
-       /** @var int Milliseconds of key fetch/validate/regenerate delay prone to set() stampedes */
-       private static $SET_DELAY_HIGH_MS = 50;
-       /** @var int Min millisecond set() backoff during hold-off (far less than INTERIM_KEY_TTL) */
-       private static $RECENT_SET_LOW_MS = 50;
-       /** @var int Max millisecond set() backoff during hold-off (far less than INTERIM_KEY_TTL) */
-       private static $RECENT_SET_HIGH_MS = 100;
-
-       /** @var int Consider value generation slow if it takes more than this many seconds */
-       private static $GENERATION_SLOW_SEC = 3;
-
-       /** @var int Key to the tombstone entry timestamp */
-       private static $PURGE_TIME = 0;
-       /** @var int Key to the tombstone entry hold-off TTL */
-       private static $PURGE_HOLDOFF = 1;
-
-       /** @var int Cache format version number */
-       private static $VERSION = 1;
-
-       /** @var int Key to WAN cache version number */
-       private static $FLD_FORMAT_VERSION = 0;
-       /** @var int Key to the cached value */
-       private static $FLD_VALUE = 1;
-       /** @var int Key to the original TTL */
-       private static $FLD_TTL = 2;
-       /** @var int Key to the cache timestamp */
-       private static $FLD_TIME = 3;
-       /** @var int Key to the flags bit field (reserved number) */
-       private static /** @noinspection PhpUnusedPrivateFieldInspection */ $FLD_FLAGS = 4;
-       /** @var int Key to collection cache version number */
-       private static $FLD_VALUE_VERSION = 5;
-       /** @var int Key to how long it took to generate the value */
-       private static $FLD_GENERATION_TIME = 6;
-
-       private static $VALUE_KEY_PREFIX = 'WANCache:v:';
-       private static $INTERIM_KEY_PREFIX = 'WANCache:i:';
-       private static $TIME_KEY_PREFIX = 'WANCache:t:';
-       private static $MUTEX_KEY_PREFIX = 'WANCache:m:';
-       private static $COOLOFF_KEY_PREFIX = 'WANCache:c:';
-
-       private static $PURGE_VAL_PREFIX = 'PURGED:';
-
-       /**
-        * @param array $params
-        *   - cache    : BagOStuff object for a persistent cache
-        *   - logger   : LoggerInterface object
-        *   - stats    : StatsdDataFactoryInterface object
-        *   - asyncHandler : A function that takes a callback and runs it later. If supplied,
-        *       whenever a preemptive refresh would be triggered in getWithSetCallback(), the
-        *       current cache value is still used instead. However, the async-handler function
-        *       receives a WAN cache callback that, when run, will execute the value generation
-        *       callback supplied by the getWithSetCallback() caller. The result will be saved
-        *       as normal. The handler is expected to call the WAN cache callback at an opportune
-        *       time (e.g. HTTP post-send), though generally within a few 100ms. [optional]
-        *   - region: the current physical region. This is required when using mcrouter as the
-        *       backing store proxy. [optional]
-        *   - cluster: name of the cache cluster used by this WAN cache. The name must be the
-        *       same in all datacenters; the ("region","cluster") tuple is what distinguishes
-        *       the counterpart cache clusters among all the datacenter. The contents of
-        *       https://github.com/facebook/mcrouter/wiki/Config-Files give background on this.
-        *       This is required when using mcrouter as the backing store proxy. [optional]
-        *   - mcrouterAware: set as true if mcrouter is the backing store proxy and mcrouter
-        *       is configured to interpret /<region>/<cluster>/ key prefixes as routes. This
-        *       requires that "region" and "cluster" are both set above. [optional]
-        *   - epoch: lowest UNIX timestamp a value/tombstone must have to be valid. [optional]
-        *   - secret: stable secret used for hashing long strings into key components. [optional]
-        */
-       public function __construct( array $params ) {
-               $this->cache = $params['cache'];
-               $this->region = $params['region'] ?? 'main';
-               $this->cluster = $params['cluster'] ?? 'wan-main';
-               $this->mcrouterAware = !empty( $params['mcrouterAware'] );
-               $this->epoch = $params['epoch'] ?? 0;
-               $this->secret = $params['secret'] ?? (string)$this->epoch;
-
-               $this->setLogger( $params['logger'] ?? new NullLogger() );
-               $this->stats = $params['stats'] ?? new NullStatsdDataFactory();
-               $this->asyncHandler = $params['asyncHandler'] ?? null;
-       }
-
-       /**
-        * @param LoggerInterface $logger
-        */
-       public function setLogger( LoggerInterface $logger ) {
-               $this->logger = $logger;
-       }
-
-       /**
-        * Get an instance that wraps EmptyBagOStuff
-        *
-        * @return WANObjectCache
-        */
-       public static function newEmpty() {
-               return new static( [ 'cache' => new EmptyBagOStuff() ] );
-       }
-
-       /**
-        * Fetch the value of a key from cache
-        *
-        * If supplied, $curTTL is set to the remaining TTL (current time left):
-        *   - a) INF; if $key exists, has no TTL, and is not invalidated by $checkKeys
-        *   - b) float (>=0); if $key exists, has a TTL, and is not invalidated by $checkKeys
-        *   - c) float (<0); if $key is tombstoned, stale, or existing but invalidated by $checkKeys
-        *   - d) null; if $key does not exist and is not tombstoned
-        *
-        * If a key is tombstoned, $curTTL will reflect the time since delete().
-        *
-        * The timestamp of $key will be checked against the last-purge timestamp
-        * of each of $checkKeys. Those $checkKeys not in cache will have the last-purge
-        * initialized to the current timestamp. If any of $checkKeys have a timestamp
-        * greater than that of $key, then $curTTL will reflect how long ago $key
-        * became invalid. Callers can use $curTTL to know when the value is stale.
-        * The $checkKeys parameter allow mass invalidations by updating a single key:
-        *   - a) Each "check" key represents "last purged" of some source data
-        *   - b) Callers pass in relevant "check" keys as $checkKeys in get()
-        *   - c) When the source data that "check" keys represent changes,
-        *        the touchCheckKey() method is called on them
-        *
-        * Source data entities might exists in a DB that uses snapshot isolation
-        * (e.g. the default REPEATABLE-READ in innoDB). Even for mutable data, that
-        * isolation can largely be maintained by doing the following:
-        *   - a) Calling delete() on entity change *and* creation, before DB commit
-        *   - b) Keeping transaction duration shorter than the delete() hold-off TTL
-        *   - c) Disabling interim key caching via useInterimHoldOffCaching() before get() calls
-        *
-        * However, pre-snapshot values might still be seen if an update was made
-        * in a remote datacenter but the purge from delete() didn't relay yet.
-        *
-        * Consider using getWithSetCallback() instead of get() and set() cycles.
-        * That method has cache slam avoiding features for hot/expensive keys.
-        *
-        * Pass $info as WANObjectCache::PASS_BY_REF to transform it into a cache key metadata map.
-        * This map includes the following metadata:
-        *   - asOf: UNIX timestamp of the value or null if the key is nonexistant
-        *   - tombAsOf: UNIX timestamp of the tombstone or null if the key is not tombstoned
-        *   - lastCKPurge: UNIX timestamp of the highest check key or null if none provided
-        *   - version: cached value version number or null if the key is nonexistant
-        *
-        * Otherwise, $info will transform into the cached value timestamp.
-        *
-        * @param string $key Cache key made from makeKey() or makeGlobalKey()
-        * @param mixed|null &$curTTL Approximate TTL left on the key if present/tombstoned [returned]
-        * @param string[] $checkKeys The "check" keys used to validate the value
-        * @param mixed|null &$info Key info if WANObjectCache::PASS_BY_REF [returned]
-        * @return mixed Value of cache key or false on failure
-        */
-       final public function get(
-               $key, &$curTTL = null, array $checkKeys = [], &$info = null
-       ) {
-               $curTTLs = self::PASS_BY_REF;
-               $infoByKey = self::PASS_BY_REF;
-               $values = $this->getMulti( [ $key ], $curTTLs, $checkKeys, $infoByKey );
-               $curTTL = $curTTLs[$key] ?? null;
-               if ( $info === self::PASS_BY_REF ) {
-                       $info = [
-                               'asOf' => $infoByKey[$key]['asOf'] ?? null,
-                               'tombAsOf' => $infoByKey[$key]['tombAsOf'] ?? null,
-                               'lastCKPurge' => $infoByKey[$key]['lastCKPurge'] ?? null,
-                               'version' => $infoByKey[$key]['version'] ?? null
-                       ];
-               } else {
-                       $info = $infoByKey[$key]['asOf'] ?? null; // b/c
-               }
-
-               return $values[$key] ?? false;
-       }
-
-       /**
-        * Fetch the value of several keys from cache
-        *
-        * Pass $info as WANObjectCache::PASS_BY_REF to transform it into a map of cache keys
-        * to cache key metadata maps, each having the same style as those of WANObjectCache::get().
-        * All the cache keys listed in $keys will have an entry.
-        *
-        * Othwerwise, $info will transform into a map of (cache key => cached value timestamp).
-        * Only the cache keys listed in $keys that exists or are tombstoned will have an entry.
-        *
-        * $checkKeys holds the "check" keys used to validate values of applicable keys. The integer
-        * indexes hold "check" keys that apply to all of $keys while the string indexes hold "check"
-        * keys that only apply to the cache key with that name.
-        *
-        * @see WANObjectCache::get()
-        *
-        * @param string[] $keys List of cache keys made from makeKey() or makeGlobalKey()
-        * @param mixed|null &$curTTLs Map of (key => TTL left) for existing/tombstoned keys [returned]
-        * @param string[]|string[][] $checkKeys Map of (integer or cache key => "check" key(s))
-        * @param mixed|null &$info Map of (key => info) if WANObjectCache::PASS_BY_REF [returned]
-        * @return mixed[] Map of (key => value) for existing values; order of $keys is preserved
-        */
-       final public function getMulti(
-               array $keys,
-               &$curTTLs = [],
-               array $checkKeys = [],
-               &$info = null
-       ) {
-               $result = [];
-               $curTTLs = [];
-               $infoByKey = [];
-
-               $vPrefixLen = strlen( self::$VALUE_KEY_PREFIX );
-               $valueKeys = self::prefixCacheKeys( $keys, self::$VALUE_KEY_PREFIX );
-
-               $checkKeysForAll = [];
-               $checkKeysByKey = [];
-               $checkKeysFlat = [];
-               foreach ( $checkKeys as $i => $checkKeyGroup ) {
-                       $prefixed = self::prefixCacheKeys( (array)$checkKeyGroup, self::$TIME_KEY_PREFIX );
-                       $checkKeysFlat = array_merge( $checkKeysFlat, $prefixed );
-                       // Are these check keys for a specific cache key, or for all keys being fetched?
-                       if ( is_int( $i ) ) {
-                               $checkKeysForAll = array_merge( $checkKeysForAll, $prefixed );
-                       } else {
-                               $checkKeysByKey[$i] = $prefixed;
-                       }
-               }
-
-               // Fetch all of the raw values
-               $keysGet = array_merge( $valueKeys, $checkKeysFlat );
-               if ( $this->warmupCache ) {
-                       $wrappedValues = array_intersect_key( $this->warmupCache, array_flip( $keysGet ) );
-                       $keysGet = array_diff( $keysGet, array_keys( $wrappedValues ) ); // keys left to fetch
-                       $this->warmupKeyMisses += count( $keysGet );
-               } else {
-                       $wrappedValues = [];
-               }
-               if ( $keysGet ) {
-                       $wrappedValues += $this->cache->getMulti( $keysGet );
-               }
-               // Time used to compare/init "check" keys (derived after getMulti() to be pessimistic)
-               $now = $this->getCurrentTime();
-
-               // Collect timestamps from all "check" keys
-               $purgeValuesForAll = $this->processCheckKeys( $checkKeysForAll, $wrappedValues, $now );
-               $purgeValuesByKey = [];
-               foreach ( $checkKeysByKey as $cacheKey => $checks ) {
-                       $purgeValuesByKey[$cacheKey] =
-                               $this->processCheckKeys( $checks, $wrappedValues, $now );
-               }
-
-               // Get the main cache value for each key and validate them
-               foreach ( $valueKeys as $vKey ) {
-                       $key = substr( $vKey, $vPrefixLen ); // unprefix
-                       list( $value, $keyInfo ) = $this->unwrap( $wrappedValues[$vKey] ?? false, $now );
-                       // Force dependent keys to be seen as stale for a while after purging
-                       // to reduce race conditions involving stale data getting cached
-                       $purgeValues = $purgeValuesForAll;
-                       if ( isset( $purgeValuesByKey[$key] ) ) {
-                               $purgeValues = array_merge( $purgeValues, $purgeValuesByKey[$key] );
-                       }
-
-                       $lastCKPurge = null; // timestamp of the highest check key
-                       foreach ( $purgeValues as $purge ) {
-                               $lastCKPurge = max( $purge[self::$PURGE_TIME], $lastCKPurge );
-                               $safeTimestamp = $purge[self::$PURGE_TIME] + $purge[self::$PURGE_HOLDOFF];
-                               if ( $value !== false && $safeTimestamp >= $keyInfo['asOf'] ) {
-                                       // How long ago this value was invalidated by *this* check key
-                                       $ago = min( $purge[self::$PURGE_TIME] - $now, self::$TINY_NEGATIVE );
-                                       // How long ago this value was invalidated by *any* known check key
-                                       $keyInfo['curTTL'] = min( $keyInfo['curTTL'], $ago );
-                               }
-                       }
-                       $keyInfo[ 'lastCKPurge'] = $lastCKPurge;
-
-                       if ( $value !== false ) {
-                               $result[$key] = $value;
-                       }
-                       if ( $keyInfo['curTTL'] !== null ) {
-                               $curTTLs[$key] = $keyInfo['curTTL'];
-                       }
-
-                       $infoByKey[$key] = ( $info === self::PASS_BY_REF )
-                               ? $keyInfo
-                               : $keyInfo['asOf']; // b/c
-               }
-
-               $info = $infoByKey;
-
-               return $result;
-       }
-
-       /**
-        * @since 1.27
-        * @param string[] $timeKeys List of prefixed time check keys
-        * @param mixed[] $wrappedValues
-        * @param float $now
-        * @return array[] List of purge value arrays
-        */
-       private function processCheckKeys( array $timeKeys, array $wrappedValues, $now ) {
-               $purgeValues = [];
-               foreach ( $timeKeys as $timeKey ) {
-                       $purge = isset( $wrappedValues[$timeKey] )
-                               ? $this->parsePurgeValue( $wrappedValues[$timeKey] )
-                               : false;
-                       if ( $purge === false ) {
-                               // Key is not set or malformed; regenerate
-                               $newVal = $this->makePurgeValue( $now, self::HOLDOFF_TTL );
-                               $this->cache->add( $timeKey, $newVal, self::$CHECK_KEY_TTL );
-                               $purge = $this->parsePurgeValue( $newVal );
-                       }
-                       $purgeValues[] = $purge;
-               }
-               return $purgeValues;
-       }
-
-       /**
-        * Set the value of a key in cache
-        *
-        * Simply calling this method when source data changes is not valid because
-        * the changes do not replicate to the other WAN sites. In that case, delete()
-        * should be used instead. This method is intended for use on cache misses.
-        *
-        * If the data was read from a snapshot-isolated transactions (e.g. the default
-        * REPEATABLE-READ in innoDB), use 'since' to avoid the following race condition:
-        *   - a) T1 starts
-        *   - b) T2 updates a row, calls delete(), and commits
-        *   - c) The HOLDOFF_TTL passes, expiring the delete() tombstone
-        *   - d) T1 reads the row and calls set() due to a cache miss
-        *   - e) Stale value is stuck in cache
-        *
-        * Setting 'lag' and 'since' help avoids keys getting stuck in stale states.
-        *
-        * Be aware that this does not update the process cache for getWithSetCallback()
-        * callers. Keys accessed via that method are not generally meant to also be set
-        * using this primitive method.
-        *
-        * Do not use this method on versioned keys accessed via getWithSetCallback().
-        *
-        * Example usage:
-        * @code
-        *     $dbr = wfGetDB( DB_REPLICA );
-        *     $setOpts = Database::getCacheSetOptions( $dbr );
-        *     // Fetch the row from the DB
-        *     $row = $dbr->selectRow( ... );
-        *     $key = $cache->makeKey( 'building', $buildingId );
-        *     $cache->set( $key, $row, $cache::TTL_DAY, $setOpts );
-        * @endcode
-        *
-        * @param string $key Cache key
-        * @param mixed $value
-        * @param int $ttl Seconds to live. Special values are:
-        *   - WANObjectCache::TTL_INDEFINITE: Cache forever (default)
-        *   - WANObjectCache::TTL_UNCACHEABLE: Do not cache (if the key exists, it is not deleted)
-        * @param array $opts Options map:
-        *   - lag: Seconds of replica DB lag. Typically, this is either the replica DB lag
-        *      before the data was read or, if applicable, the replica DB lag before
-        *      the snapshot-isolated transaction the data was read from started.
-        *      Use false to indicate that replication is not running.
-        *      Default: 0 seconds
-        *   - since: UNIX timestamp of the data in $value. Typically, this is either
-        *      the current time the data was read or (if applicable) the time when
-        *      the snapshot-isolated transaction the data was read from started.
-        *      Default: 0 seconds
-        *   - pending: Whether this data is possibly from an uncommitted write transaction.
-        *      Generally, other threads should not see values from the future and
-        *      they certainly should not see ones that ended up getting rolled back.
-        *      Default: false
-        *   - lockTSE: If excessive replication/snapshot lag is detected, then store the value
-        *      with this TTL and flag it as stale. This is only useful if the reads for this key
-        *      use getWithSetCallback() with "lockTSE" set. Note that if "staleTTL" is set
-        *      then it will still add on to this TTL in the excessive lag scenario.
-        *      Default: WANObjectCache::TSE_NONE
-        *   - staleTTL: Seconds to keep the key around if it is stale. The get()/getMulti()
-        *      methods return such stale values with a $curTTL of 0, and getWithSetCallback()
-        *      will call the regeneration callback in such cases, passing in the old value
-        *      and its as-of time to the callback. This is useful if adaptiveTTL() is used
-        *      on the old value's as-of time when it is verified as still being correct.
-        *      Default: WANObjectCache::STALE_TTL_NONE
-        *   - creating: Optimize for the case where the key does not already exist.
-        *      Default: false
-        *   - version: Integer version number signifiying the format of the value.
-        *      Default: null
-        *   - walltime: How long the value took to generate in seconds. Default: 0.0
-        * @note Options added in 1.28: staleTTL
-        * @note Options added in 1.33: creating
-        * @note Options added in 1.34: version, walltime
-        * @return bool Success
-        */
-       final public function set( $key, $value, $ttl = self::TTL_INDEFINITE, array $opts = [] ) {
-               $now = $this->getCurrentTime();
-               $lag = $opts['lag'] ?? 0;
-               $age = isset( $opts['since'] ) ? max( 0, $now - $opts['since'] ) : 0;
-               $pending = $opts['pending'] ?? false;
-               $lockTSE = $opts['lockTSE'] ?? self::TSE_NONE;
-               $staleTTL = $opts['staleTTL'] ?? self::STALE_TTL_NONE;
-               $creating = $opts['creating'] ?? false;
-               $version = $opts['version'] ?? null;
-               $walltime = $opts['walltime'] ?? 0.0;
-
-               if ( $ttl < 0 ) {
-                       return true;
-               }
-
-               // Do not cache potentially uncommitted data as it might get rolled back
-               if ( $pending ) {
-                       $this->logger->info(
-                               'Rejected set() for {cachekey} due to pending writes.',
-                               [ 'cachekey' => $key ]
-                       );
-
-                       return true; // no-op the write for being unsafe
-               }
-
-               $logicalTTL = null; // logical TTL override
-               // Check if there's a risk of writing stale data after the purge tombstone expired
-               if ( $lag === false || ( $lag + $age ) > self::MAX_READ_LAG ) {
-                       // Case A: any long-running transaction
-                       if ( $age > self::MAX_READ_LAG ) {
-                               if ( $lockTSE >= 0 ) {
-                                       // Store value as *almost* stale to avoid cache and mutex stampedes
-                                       $logicalTTL = self::TTL_SECOND;
-                                       $this->logger->info(
-                                               'Lowered set() TTL for {cachekey} due to snapshot lag.',
-                                               [ 'cachekey' => $key, 'lag' => $lag, 'age' => $age ]
-                                       );
-                               } else {
-                                       $this->logger->info(
-                                               'Rejected set() for {cachekey} due to snapshot lag.',
-                                               [ 'cachekey' => $key, 'lag' => $lag, 'age' => $age ]
-                                       );
-
-                                       return true; // no-op the write for being unsafe
-                               }
-                       // Case B: high replication lag; lower TTL instead of ignoring all set()s
-                       } elseif ( $lag === false || $lag > self::MAX_READ_LAG ) {
-                               if ( $lockTSE >= 0 ) {
-                                       $logicalTTL = min( $ttl ?: INF, self::TTL_LAGGED );
-                               } else {
-                                       $ttl = min( $ttl ?: INF, self::TTL_LAGGED );
-                               }
-                               $this->logger->warning(
-                                       'Lowered set() TTL for {cachekey} due to replication lag.',
-                                       [ 'cachekey' => $key, 'lag' => $lag, 'age' => $age ]
-                               );
-                       // Case C: medium length request with medium replication lag
-                       } elseif ( $lockTSE >= 0 ) {
-                               // Store value as *almost* stale to avoid cache and mutex stampedes
-                               $logicalTTL = self::TTL_SECOND;
-                               $this->logger->info(
-                                       'Lowered set() TTL for {cachekey} due to high read lag.',
-                                       [ 'cachekey' => $key, 'lag' => $lag, 'age' => $age ]
-                               );
-                       } else {
-                               $this->logger->info(
-                                       'Rejected set() for {cachekey} due to high read lag.',
-                                       [ 'cachekey' => $key, 'lag' => $lag, 'age' => $age ]
-                               );
-
-                               return true; // no-op the write for being unsafe
-                       }
-               }
-
-               // Wrap that value with time/TTL/version metadata
-               $wrapped = $this->wrap( $value, $logicalTTL ?: $ttl, $version, $now, $walltime );
-               $storeTTL = $ttl + $staleTTL;
-
-               if ( $creating ) {
-                       $ok = $this->cache->add( self::$VALUE_KEY_PREFIX . $key, $wrapped, $storeTTL );
-               } else {
-                       $ok = $this->cache->merge(
-                               self::$VALUE_KEY_PREFIX . $key,
-                               function ( $cache, $key, $cWrapped ) use ( $wrapped ) {
-                                       // A string value means that it is a tombstone; do nothing in that case
-                                       return ( is_string( $cWrapped ) ) ? false : $wrapped;
-                               },
-                               $storeTTL,
-                               1 // 1 attempt
-                       );
-               }
-
-               return $ok;
-       }
-
-       /**
-        * Purge a key from all datacenters
-        *
-        * This should only be called when the underlying data (being cached)
-        * changes in a significant way. This deletes the key and starts a hold-off
-        * period where the key cannot be written to for a few seconds (HOLDOFF_TTL).
-        * This is done to avoid the following race condition:
-        *   - a) Some DB data changes and delete() is called on a corresponding key
-        *   - b) A request refills the key with a stale value from a lagged DB
-        *   - c) The stale value is stuck there until the key is expired/evicted
-        *
-        * This is implemented by storing a special "tombstone" value at the cache
-        * key that this class recognizes; get() calls will return false for the key
-        * and any set() calls will refuse to replace tombstone values at the key.
-        * For this to always avoid stale value writes, the following must hold:
-        *   - a) Replication lag is bounded to being less than HOLDOFF_TTL; or
-        *   - b) If lag is higher, the DB will have gone into read-only mode already
-        *
-        * Note that set() can also be lag-aware and lower the TTL if it's high.
-        *
-        * Be aware that this does not clear the process cache. Even if it did, callbacks
-        * used by getWithSetCallback() might still return stale data in the case of either
-        * uncommitted or not-yet-replicated changes (callback generally use replica DBs).
-        *
-        * When using potentially long-running ACID transactions, a good pattern is
-        * to use a pre-commit hook to issue the delete. This means that immediately
-        * after commit, callers will see the tombstone in cache upon purge relay.
-        * It also avoids the following race condition:
-        *   - a) T1 begins, changes a row, and calls delete()
-        *   - b) The HOLDOFF_TTL passes, expiring the delete() tombstone
-        *   - c) T2 starts, reads the row and calls set() due to a cache miss
-        *   - d) T1 finally commits
-        *   - e) Stale value is stuck in cache
-        *
-        * Example usage:
-        * @code
-        *     $dbw->startAtomic( __METHOD__ ); // start of request
-        *     ... <execute some stuff> ...
-        *     // Update the row in the DB
-        *     $dbw->update( ... );
-        *     $key = $cache->makeKey( 'homes', $homeId );
-        *     // Purge the corresponding cache entry just before committing
-        *     $dbw->onTransactionPreCommitOrIdle( function() use ( $cache, $key ) {
-        *         $cache->delete( $key );
-        *     } );
-        *     ... <execute some stuff> ...
-        *     $dbw->endAtomic( __METHOD__ ); // end of request
-        * @endcode
-        *
-        * The $ttl parameter can be used when purging values that have not actually changed
-        * recently. For example, a cleanup script to purge cache entries does not really need
-        * a hold-off period, so it can use HOLDOFF_TTL_NONE. Likewise for user-requested purge.
-        * Note that $ttl limits the effective range of 'lockTSE' for getWithSetCallback().
-        *
-        * If called twice on the same key, then the last hold-off TTL takes precedence. For
-        * idempotence, the $ttl should not vary for different delete() calls on the same key.
-        *
-        * @param string $key Cache key
-        * @param int $ttl Tombstone TTL; Default: WANObjectCache::HOLDOFF_TTL
-        * @return bool True if the item was purged or not found, false on failure
-        */
-       final public function delete( $key, $ttl = self::HOLDOFF_TTL ) {
-               if ( $ttl <= 0 ) {
-                       // Publish the purge to all datacenters
-                       $ok = $this->relayDelete( self::$VALUE_KEY_PREFIX . $key );
-               } else {
-                       // Publish the purge to all datacenters
-                       $ok = $this->relayPurge( self::$VALUE_KEY_PREFIX . $key, $ttl, self::HOLDOFF_TTL_NONE );
-               }
-
-               $kClass = $this->determineKeyClassForStats( $key );
-               $this->stats->increment( "wanobjectcache.$kClass.delete." . ( $ok ? 'ok' : 'error' ) );
-
-               return $ok;
-       }
-
-       /**
-        * Fetch the value of a timestamp "check" key
-        *
-        * The key will be *initialized* to the current time if not set,
-        * so only call this method if this behavior is actually desired
-        *
-        * The timestamp can be used to check whether a cached value is valid.
-        * Callers should not assume that this returns the same timestamp in
-        * all datacenters due to relay delays.
-        *
-        * The level of staleness can roughly be estimated from this key, but
-        * if the key was evicted from cache, such calculations may show the
-        * time since expiry as ~0 seconds.
-        *
-        * Note that "check" keys won't collide with other regular keys.
-        *
-        * @param string $key
-        * @return float UNIX timestamp
-        */
-       final public function getCheckKeyTime( $key ) {
-               return $this->getMultiCheckKeyTime( [ $key ] )[$key];
-       }
-
-       /**
-        * Fetch the values of each timestamp "check" key
-        *
-        * This works like getCheckKeyTime() except it takes a list of keys
-        * and returns a map of timestamps instead of just that of one key
-        *
-        * This might be useful if both:
-        *   - a) a class of entities each depend on hundreds of other entities
-        *   - b) these other entities are depended upon by millions of entities
-        *
-        * The later entities can each use a "check" key to invalidate their dependee entities.
-        * However, it is expensive for the former entities to verify against all of the relevant
-        * "check" keys during each getWithSetCallback() call. A less expensive approach is to do
-        * these verifications only after a "time-till-verify" (TTV) has passed. This is a middle
-        * ground between using blind TTLs and using constant verification. The adaptiveTTL() method
-        * can be used to dynamically adjust the TTV. Also, the initial TTV can make use of the
-        * last-modified times of the dependant entities (either from the DB or the "check" keys).
-        *
-        * Example usage:
-        * @code
-        *     $value = $cache->getWithSetCallback(
-        *         $cache->makeGlobalKey( 'wikibase-item', $id ),
-        *         self::INITIAL_TTV, // initial time-till-verify
-        *         function ( $oldValue, &$ttv, &$setOpts, $oldAsOf ) use ( $checkKeys, $cache ) {
-        *             $now = microtime( true );
-        *             // Use $oldValue if it passes max ultimate age and "check" key comparisons
-        *             if ( $oldValue &&
-        *                 $oldAsOf > max( $cache->getMultiCheckKeyTime( $checkKeys ) ) &&
-        *                 ( $now - $oldValue['ctime'] ) <= self::MAX_CACHE_AGE
-        *             ) {
-        *                 // Increase time-till-verify by 50% of last time to reduce overhead
-        *                 $ttv = $cache->adaptiveTTL( $oldAsOf, self::MAX_TTV, self::MIN_TTV, 1.5 );
-        *                 // Unlike $oldAsOf, "ctime" is the ultimate age of the cached data
-        *                 return $oldValue;
-        *             }
-        *
-        *             $mtimes = []; // dependency last-modified times; passed by reference
-        *             $value = [ 'data' => $this->fetchEntityData( $mtimes ), 'ctime' => $now ];
-        *             // Guess time-till-change among the dependencies, e.g. 1/(total change rate)
-        *             $ttc = 1 / array_sum( array_map(
-        *                 function ( $mtime ) use ( $now ) {
-        *                     return 1 / ( $mtime ? ( $now - $mtime ) : 900 );
-        *                 },
-        *                 $mtimes
-        *             ) );
-        *             // The time-to-verify should not be overly pessimistic nor optimistic
-        *             $ttv = min( max( $ttc, self::MIN_TTV ), self::MAX_TTV );
-        *
-        *             return $value;
-        *         },
-        *         [ 'staleTTL' => $cache::TTL_DAY ] // keep around to verify and re-save
-        *     );
-        * @endcode
-        *
-        * @see WANObjectCache::getCheckKeyTime()
-        * @see WANObjectCache::getWithSetCallback()
-        *
-        * @param string[] $keys
-        * @return float[] Map of (key => UNIX timestamp)
-        * @since 1.31
-        */
-       final public function getMultiCheckKeyTime( array $keys ) {
-               $rawKeys = [];
-               foreach ( $keys as $key ) {
-                       $rawKeys[$key] = self::$TIME_KEY_PREFIX . $key;
-               }
-
-               $rawValues = $this->cache->getMulti( $rawKeys );
-               $rawValues += array_fill_keys( $rawKeys, false );
-
-               $times = [];
-               foreach ( $rawKeys as $key => $rawKey ) {
-                       $purge = $this->parsePurgeValue( $rawValues[$rawKey] );
-                       if ( $purge !== false ) {
-                               $time = $purge[self::$PURGE_TIME];
-                       } else {
-                               // Casting assures identical floats for the next getCheckKeyTime() calls
-                               $now = (string)$this->getCurrentTime();
-                               $this->cache->add(
-                                       $rawKey,
-                                       $this->makePurgeValue( $now, self::HOLDOFF_TTL ),
-                                       self::$CHECK_KEY_TTL
-                               );
-                               $time = (float)$now;
-                       }
-
-                       $times[$key] = $time;
-               }
-
-               return $times;
-       }
-
-       /**
-        * Purge a "check" key from all datacenters, invalidating keys that use it
-        *
-        * This should only be called when the underlying data (being cached)
-        * changes in a significant way, and it is impractical to call delete()
-        * on all keys that should be changed. When get() is called on those
-        * keys, the relevant "check" keys must be supplied for this to work.
-        *
-        * The "check" key essentially represents a last-modified time of an entity.
-        * When the key is touched, the timestamp will be updated to the current time.
-        * Keys using the "check" key via get(), getMulti(), or getWithSetCallback() will
-        * be invalidated. This approach is useful if many keys depend on a single entity.
-        *
-        * The timestamp of the "check" key is treated as being HOLDOFF_TTL seconds in the
-        * future by get*() methods in order to avoid race conditions where keys are updated
-        * with stale values (e.g. from a lagged replica DB). A high TTL is set on the "check"
-        * key, making it possible to know the timestamp of the last change to the corresponding
-        * entities in most cases. This might use more cache space than resetCheckKey().
-        *
-        * When a few important keys get a large number of hits, a high cache time is usually
-        * desired as well as "lockTSE" logic. The resetCheckKey() method is less appropriate
-        * in such cases since the "time since expiry" cannot be inferred, causing any get()
-        * after the reset to treat the key as being "hot", resulting in more stale value usage.
-        *
-        * Note that "check" keys won't collide with other regular keys.
-        *
-        * @see WANObjectCache::get()
-        * @see WANObjectCache::getWithSetCallback()
-        * @see WANObjectCache::resetCheckKey()
-        *
-        * @param string $key Cache key
-        * @param int $holdoff HOLDOFF_TTL or HOLDOFF_TTL_NONE constant
-        * @return bool True if the item was purged or not found, false on failure
-        */
-       final public function touchCheckKey( $key, $holdoff = self::HOLDOFF_TTL ) {
-               // Publish the purge to all datacenters
-               $ok = $this->relayPurge( self::$TIME_KEY_PREFIX . $key, self::$CHECK_KEY_TTL, $holdoff );
-
-               $kClass = $this->determineKeyClassForStats( $key );
-               $this->stats->increment( "wanobjectcache.$kClass.ck_touch." . ( $ok ? 'ok' : 'error' ) );
-
-               return $ok;
-       }
-
-       /**
-        * Delete a "check" key from all datacenters, invalidating keys that use it
-        *
-        * This is similar to touchCheckKey() in that keys using it via get(), getMulti(),
-        * or getWithSetCallback() will be invalidated. The differences are:
-        *   - a) The "check" key will be deleted from all caches and lazily
-        *        re-initialized when accessed (rather than set everywhere)
-        *   - b) Thus, dependent keys will be known to be stale, but not
-        *        for how long (they are treated as "just" purged), which
-        *        effects any lockTSE logic in getWithSetCallback()
-        *   - c) Since "check" keys are initialized only on the server the key hashes
-        *        to, any temporary ejection of that server will cause the value to be
-        *        seen as purged as a new server will initialize the "check" key.
-        *
-        * The advantage here is that the "check" keys, which have high TTLs, will only
-        * be created when a get*() method actually uses that key. This is better when
-        * a large number of "check" keys are invalided in a short period of time.
-        *
-        * Note that "check" keys won't collide with other regular keys.
-        *
-        * @see WANObjectCache::get()
-        * @see WANObjectCache::getWithSetCallback()
-        * @see WANObjectCache::touchCheckKey()
-        *
-        * @param string $key Cache key
-        * @return bool True if the item was purged or not found, false on failure
-        */
-       final public function resetCheckKey( $key ) {
-               // Publish the purge to all datacenters
-               $ok = $this->relayDelete( self::$TIME_KEY_PREFIX . $key );
-
-               $kClass = $this->determineKeyClassForStats( $key );
-               $this->stats->increment( "wanobjectcache.$kClass.ck_reset." . ( $ok ? 'ok' : 'error' ) );
-
-               return $ok;
-       }
-
-       /**
-        * Method to fetch/regenerate cache keys
-        *
-        * On cache miss, the key will be set to the callback result via set()
-        * (unless the callback returns false) and that result will be returned.
-        * The arguments supplied to the callback are:
-        *   - $oldValue : current cache value or false if not present
-        *   - &$ttl : a reference to the TTL which can be altered
-        *   - &$setOpts : a reference to options for set() which can be altered
-        *   - $oldAsOf : generation UNIX timestamp of $oldValue or null if not present (since 1.28)
-        *
-        * It is strongly recommended to set the 'lag' and 'since' fields to avoid race conditions
-        * that can cause stale values to get stuck at keys. Usually, callbacks ignore the current
-        * value, but it can be used to maintain "most recent X" values that come from time or
-        * sequence based source data, provided that the "as of" id/time is tracked. Note that
-        * preemptive regeneration and $checkKeys can result in a non-false current value.
-        *
-        * Usage of $checkKeys is similar to get() and getMulti(). However, rather than the caller
-        * having to inspect a "current time left" variable (e.g. $curTTL, $curTTLs), a cache
-        * regeneration will automatically be triggered using the callback.
-        *
-        * The $ttl argument and "hotTTR" option (in $opts) use time-dependant randomization
-        * to avoid stampedes. Keys that are slow to regenerate and either heavily used
-        * or subject to explicit (unpredictable) purges, may need additional mechanisms.
-        * The simplest way to avoid stampedes for such keys is to use 'lockTSE' (in $opts).
-        * If explicit purges are needed, also:
-        *   - a) Pass $key into $checkKeys
-        *   - b) Use touchCheckKey( $key ) instead of delete( $key )
-        *
-        * Example usage (typical key):
-        * @code
-        *     $catInfo = $cache->getWithSetCallback(
-        *         // Key to store the cached value under
-        *         $cache->makeKey( 'cat-attributes', $catId ),
-        *         // Time-to-live (in seconds)
-        *         $cache::TTL_MINUTE,
-        *         // Function that derives the new key value
-        *         function ( $oldValue, &$ttl, array &$setOpts ) {
-        *             $dbr = wfGetDB( DB_REPLICA );
-        *             // Account for any snapshot/replica DB lag
-        *             $setOpts += Database::getCacheSetOptions( $dbr );
-        *
-        *             return $dbr->selectRow( ... );
-        *        }
-        *     );
-        * @endcode
-        *
-        * Example usage (key that is expensive and hot):
-        * @code
-        *     $catConfig = $cache->getWithSetCallback(
-        *         // Key to store the cached value under
-        *         $cache->makeKey( 'site-cat-config' ),
-        *         // Time-to-live (in seconds)
-        *         $cache::TTL_DAY,
-        *         // Function that derives the new key value
-        *         function ( $oldValue, &$ttl, array &$setOpts ) {
-        *             $dbr = wfGetDB( DB_REPLICA );
-        *             // Account for any snapshot/replica DB lag
-        *             $setOpts += Database::getCacheSetOptions( $dbr );
-        *
-        *             return CatConfig::newFromRow( $dbr->selectRow( ... ) );
-        *         },
-        *         [
-        *             // Calling touchCheckKey() on this key invalidates the cache
-        *             'checkKeys' => [ $cache->makeKey( 'site-cat-config' ) ],
-        *             // Try to only let one datacenter thread manage cache updates at a time
-        *             'lockTSE' => 30,
-        *             // Avoid querying cache servers multiple times in a web request
-        *             'pcTTL' => $cache::TTL_PROC_LONG
-        *         ]
-        *     );
-        * @endcode
-        *
-        * Example usage (key with dynamic dependencies):
-        * @code
-        *     $catState = $cache->getWithSetCallback(
-        *         // Key to store the cached value under
-        *         $cache->makeKey( 'cat-state', $cat->getId() ),
-        *         // Time-to-live (seconds)
-        *         $cache::TTL_HOUR,
-        *         // Function that derives the new key value
-        *         function ( $oldValue, &$ttl, array &$setOpts ) {
-        *             // Determine new value from the DB
-        *             $dbr = wfGetDB( DB_REPLICA );
-        *             // Account for any snapshot/replica DB lag
-        *             $setOpts += Database::getCacheSetOptions( $dbr );
-        *
-        *             return CatState::newFromResults( $dbr->select( ... ) );
-        *         },
-        *         [
-        *              // The "check" keys that represent things the value depends on;
-        *              // Calling touchCheckKey() on any of them invalidates the cache
-        *             'checkKeys' => [
-        *                 $cache->makeKey( 'sustenance-bowls', $cat->getRoomId() ),
-        *                 $cache->makeKey( 'people-present', $cat->getHouseId() ),
-        *                 $cache->makeKey( 'cat-laws', $cat->getCityId() ),
-        *             ]
-        *         ]
-        *     );
-        * @endcode
-        *
-        * Example usage (key that is expensive with too many DB dependencies for "check keys"):
-        * @code
-        *     $catToys = $cache->getWithSetCallback(
-        *         // Key to store the cached value under
-        *         $cache->makeKey( 'cat-toys', $catId ),
-        *         // Time-to-live (seconds)
-        *         $cache::TTL_HOUR,
-        *         // Function that derives the new key value
-        *         function ( $oldValue, &$ttl, array &$setOpts ) {
-        *             // Determine new value from the DB
-        *             $dbr = wfGetDB( DB_REPLICA );
-        *             // Account for any snapshot/replica DB lag
-        *             $setOpts += Database::getCacheSetOptions( $dbr );
-        *
-        *             return CatToys::newFromResults( $dbr->select( ... ) );
-        *         },
-        *         [
-        *              // Get the highest timestamp of any of the cat's toys
-        *             'touchedCallback' => function ( $value ) use ( $catId ) {
-        *                 $dbr = wfGetDB( DB_REPLICA );
-        *                 $ts = $dbr->selectField( 'cat_toys', 'MAX(ct_touched)', ... );
-        *
-        *                 return wfTimestampOrNull( TS_UNIX, $ts );
-        *             },
-        *             // Avoid DB queries for repeated access
-        *             'pcTTL' => $cache::TTL_PROC_SHORT
-        *         ]
-        *     );
-        * @endcode
-        *
-        * Example usage (hot key holding most recent 100 events):
-        * @code
-        *     $lastCatActions = $cache->getWithSetCallback(
-        *         // Key to store the cached value under
-        *         $cache->makeKey( 'cat-last-actions', 100 ),
-        *         // Time-to-live (in seconds)
-        *         10,
-        *         // Function that derives the new key value
-        *         function ( $oldValue, &$ttl, array &$setOpts ) {
-        *             $dbr = wfGetDB( DB_REPLICA );
-        *             // Account for any snapshot/replica DB lag
-        *             $setOpts += Database::getCacheSetOptions( $dbr );
-        *
-        *             // Start off with the last cached list
-        *             $list = $oldValue ?: [];
-        *             // Fetch the last 100 relevant rows in descending order;
-        *             // only fetch rows newer than $list[0] to reduce scanning
-        *             $rows = iterator_to_array( $dbr->select( ... ) );
-        *             // Merge them and get the new "last 100" rows
-        *             return array_slice( array_merge( $new, $list ), 0, 100 );
-        *        },
-        *        [
-        *             // Try to only let one datacenter thread manage cache updates at a time
-        *             'lockTSE' => 30,
-        *             // Use a magic value when no cache value is ready rather than stampeding
-        *             'busyValue' => 'computing'
-        *        ]
-        *     );
-        * @endcode
-        *
-        * Example usage (key holding an LRU subkey:value map; this can avoid flooding cache with
-        * keys for an unlimited set of (constraint,situation) pairs, thereby avoiding elevated
-        * cache evictions and wasted memory):
-        * @code
-        *     $catSituationTolerabilityCache = $this->cache->getWithSetCallback(
-        *         // Group by constraint ID/hash, cat family ID/hash, or something else useful
-        *         $this->cache->makeKey( 'cat-situation-tolerability-checks', $groupKey ),
-        *         WANObjectCache::TTL_DAY, // rarely used groups should fade away
-        *         // The $scenarioKey format is $constraintId:<ID/hash of $situation>
-        *         function ( $cacheMap ) use ( $scenarioKey, $constraintId, $situation ) {
-        *             $lruCache = MapCacheLRU::newFromArray( $cacheMap ?: [], self::CACHE_SIZE );
-        *             $result = $lruCache->get( $scenarioKey ); // triggers LRU bump if present
-        *             if ( $result === null || $this->isScenarioResultExpired( $result ) ) {
-        *                 $result = $this->checkScenarioTolerability( $constraintId, $situation );
-        *                 $lruCache->set( $scenarioKey, $result, 3 / 8 );
-        *             }
-        *             // Save the new LRU cache map and reset the map's TTL
-        *             return $lruCache->toArray();
-        *         },
-        *         [
-        *             // Once map is > 1 sec old, consider refreshing
-        *             'ageNew' => 1,
-        *             // Update within 5 seconds after "ageNew" given a 1hz cache check rate
-        *             'hotTTR' => 5,
-        *             // Avoid querying cache servers multiple times in a request; this also means
-        *             // that a request can only alter the value of any given constraint key once
-        *             'pcTTL' => WANObjectCache::TTL_PROC_LONG
-        *         ]
-        *     );
-        *     $tolerability = isset( $catSituationTolerabilityCache[$scenarioKey] )
-        *         ? $catSituationTolerabilityCache[$scenarioKey]
-        *         : $this->checkScenarioTolerability( $constraintId, $situation );
-        * @endcode
-        *
-        * @see WANObjectCache::get()
-        * @see WANObjectCache::set()
-        *
-        * @param string $key Cache key made from makeKey() or makeGlobalKey()
-        * @param int $ttl Seconds to live for key updates. Special values are:
-        *   - WANObjectCache::TTL_INDEFINITE: Cache forever (subject to LRU-style evictions)
-        *   - WANObjectCache::TTL_UNCACHEABLE: Do not cache (if the key exists, it is not deleted)
-        * @param callable $callback Value generation function
-        * @param array $opts Options map:
-        *   - checkKeys: List of "check" keys. The key at $key will be seen as stale when either
-        *      touchCheckKey() or resetCheckKey() is called on any of the keys in this list. This
-        *      is useful if thousands or millions of keys depend on the same entity. The entity can
-        *      simply have its "check" key updated whenever the entity is modified.
-        *      Default: [].
-        *   - graceTTL: If the key is invalidated (by "checkKeys"/"touchedCallback") less than this
-        *      many seconds ago, consider reusing the stale value. The odds of a refresh becomes
-        *      more likely over time, becoming certain once the grace period is reached. This can
-        *      reduce traffic spikes when millions of keys are compared to the same "check" key and
-        *      touchCheckKey() or resetCheckKey() is called on that "check" key. This option is not
-        *      useful for avoiding traffic spikes in the case of the key simply expiring on account
-        *      of its TTL (use "lowTTL" instead).
-        *      Default: WANObjectCache::GRACE_TTL_NONE.
-        *   - lockTSE: If the key is tombstoned or invalidated (by "checkKeys"/"touchedCallback")
-        *      less than this many seconds ago, try to have a single thread handle cache regeneration
-        *      at any given time. Other threads will use stale values if possible. If, on miss,
-        *      the time since expiration is low, the assumption is that the key is hot and that a
-        *      stampede is worth avoiding. Note that if the key falls out of cache then concurrent
-        *      threads will all run the callback on cache miss until the value is saved in cache.
-        *      The only stampede protection in that case is from duplicate cache sets when the
-        *      callback takes longer than WANObjectCache::SET_DELAY_HIGH_MS milliseconds; consider
-        *      using "busyValue" if such stampedes are a problem. Note that the higher "lockTSE" is
-        *      set, the higher the worst-case staleness of returned values can be. Also note that
-        *      this option does not by itself handle the case of the key simply expiring on account
-        *      of its TTL, so make sure that "lowTTL" is not disabled when using this option. Avoid
-        *      combining this option with delete() as it can always cause a stampede due to their
-        *      being no stale value available until after a thread completes the callback.
-        *      Use WANObjectCache::TSE_NONE to disable this logic.
-        *      Default: WANObjectCache::TSE_NONE.
-        *   - busyValue: If no value exists and another thread is currently regenerating it, use this
-        *      as a fallback value (or a callback to generate such a value). This assures that cache
-        *      stampedes cannot happen if the value falls out of cache. This can be used as insurance
-        *      against cache regeneration becoming very slow for some reason (greater than the TTL).
-        *      Default: null.
-        *   - pcTTL: Process cache the value in this PHP instance for this many seconds. This avoids
-        *      network I/O when a key is read several times. This will not cache when the callback
-        *      returns false, however. Note that any purges will not be seen while process cached;
-        *      since the callback should use replica DBs and they may be lagged or have snapshot
-        *      isolation anyway, this should not typically matter.
-        *      Default: WANObjectCache::TTL_UNCACHEABLE.
-        *   - pcGroup: Process cache group to use instead of the primary one. If set, this must be
-        *      of the format ALPHANUMERIC_NAME:MAX_KEY_SIZE, e.g. "mydata:10". Use this for storing
-        *      large values, small yet numerous values, or some values with a high cost of eviction.
-        *      It is generally preferable to use a class constant when setting this value.
-        *      This has no effect unless pcTTL is used.
-        *      Default: WANObjectCache::PC_PRIMARY.
-        *   - version: Integer version number. This lets callers make breaking changes to the format
-        *      of cached values without causing problems for sites that use non-instantaneous code
-        *      deployments. Old and new code will recognize incompatible versions and purges from
-        *      both old and new code will been seen by each other. When this method encounters an
-        *      incompatibly versioned value at the provided key, a "variant key" will be used for
-        *      reading from and saving to cache. The variant key is specific to the key and version
-        *      number provided to this method. If the variant key value is older than that of the
-        *      provided key, or the provided key is non-existant, then the variant key will be seen
-        *      as non-existant. Therefore, delete() calls invalidate the provided key's variant keys.
-        *      The "checkKeys" and "touchedCallback" options still apply to variant keys as usual.
-        *      Avoid storing class objects, as this reduces compatibility (due to serialization).
-        *      Default: null.
-        *   - minAsOf: Reject values if they were generated before this UNIX timestamp.
-        *      This is useful if the source of a key is suspected of having possibly changed
-        *      recently, and the caller wants any such changes to be reflected.
-        *      Default: WANObjectCache::MIN_TIMESTAMP_NONE.
-        *   - hotTTR: Expected time-till-refresh (TTR) in seconds for keys that average ~1 hit per
-        *      second (e.g. 1Hz). Keys with a hit rate higher than 1Hz will refresh sooner than this
-        *      TTR and vise versa. Such refreshes won't happen until keys are "ageNew" seconds old.
-        *      This uses randomization to avoid triggering cache stampedes. The TTR is useful at
-        *      reducing the impact of missed cache purges, since the effect of a heavily referenced
-        *      key being stale is worse than that of a rarely referenced key. Unlike simply lowering
-        *      $ttl, seldomly used keys are largely unaffected by this option, which makes it
-        *      possible to have a high hit rate for the "long-tail" of less-used keys.
-        *      Default: WANObjectCache::HOT_TTR.
-        *   - lowTTL: Consider pre-emptive updates when the current TTL (seconds) of the key is less
-        *      than this. It becomes more likely over time, becoming certain once the key is expired.
-        *      This helps avoid cache stampedes that might be triggered due to the key expiring.
-        *      Default: WANObjectCache::LOW_TTL.
-        *   - ageNew: Consider popularity refreshes only once a key reaches this age in seconds.
-        *      Default: WANObjectCache::AGE_NEW.
-        *   - staleTTL: Seconds to keep the key around if it is stale. This means that on cache
-        *      miss the callback may get $oldValue/$oldAsOf values for keys that have already been
-        *      expired for this specified time. This is useful if adaptiveTTL() is used on the old
-        *      value's as-of time when it is verified as still being correct.
-        *      Default: WANObjectCache::STALE_TTL_NONE
-        *   - touchedCallback: A callback that takes the current value and returns a UNIX timestamp
-        *      indicating the last time a dynamic dependency changed. Null can be returned if there
-        *      are no relevant dependency changes to check. This can be used to check against things
-        *      like last-modified times of files or DB timestamp fields. This should generally not be
-        *      used for small and easily queried values in a DB if the callback itself ends up doing
-        *      a similarly expensive DB query to check a timestamp. Usages of this option makes the
-        *      most sense for values that are moderately to highly expensive to regenerate and easy
-        *      to query for dependency timestamps. The use of "pcTTL" reduces timestamp queries.
-        *      Default: null.
-        * @return mixed Value found or written to the key
-        * @note Options added in 1.28: version, busyValue, hotTTR, ageNew, pcGroup, minAsOf
-        * @note Options added in 1.31: staleTTL, graceTTL
-        * @note Options added in 1.33: touchedCallback
-        * @note Callable type hints are not used to avoid class-autoloading
-        */
-       final public function getWithSetCallback( $key, $ttl, $callback, array $opts = [] ) {
-               $version = $opts['version'] ?? null;
-               $pcTTL = $opts['pcTTL'] ?? self::TTL_UNCACHEABLE;
-               $pCache = ( $pcTTL >= 0 )
-                       ? $this->getProcessCache( $opts['pcGroup'] ?? self::PC_PRIMARY )
-                       : null;
-
-               // Use the process cache if requested as long as no outer cache callback is running.
-               // Nested callback process cache use is not lag-safe with regard to HOLDOFF_TTL since
-               // process cached values are more lagged than persistent ones as they are not purged.
-               if ( $pCache && $this->callbackDepth == 0 ) {
-                       $cached = $pCache->get( $this->getProcessCacheKey( $key, $version ), INF, false );
-                       if ( $cached !== false ) {
-                               return $cached;
-                       }
-               }
-
-               $res = $this->fetchOrRegenerate( $key, $ttl, $callback, $opts );
-               list( $value, $valueVersion, $curAsOf ) = $res;
-               if ( $valueVersion !== $version ) {
-                       // Current value has a different version; use the variant key for this version.
-                       // Regenerate the variant value if it is not newer than the main value at $key
-                       // so that purges to the main key propagate to the variant value.
-                       list( $value ) = $this->fetchOrRegenerate(
-                               $this->makeGlobalKey( 'WANCache-key-variant', md5( $key ), $version ),
-                               $ttl,
-                               $callback,
-                               [ 'version' => null, 'minAsOf' => $curAsOf ] + $opts
-                       );
-               }
-
-               // Update the process cache if enabled
-               if ( $pCache && $value !== false ) {
-                       $pCache->set( $this->getProcessCacheKey( $key, $version ), $value );
-               }
-
-               return $value;
-       }
-
-       /**
-        * Do the actual I/O for getWithSetCallback() when needed
-        *
-        * @see WANObjectCache::getWithSetCallback()
-        *
-        * @param string $key
-        * @param int $ttl
-        * @param callable $callback
-        * @param array $opts Options map for getWithSetCallback()
-        * @return array Ordered list of the following:
-        *   - Cached or regenerated value
-        *   - Cached or regenerated value version number or null if not versioned
-        *   - Timestamp of the cached value or null if there is no value
-        * @note Callable type hints are not used to avoid class-autoloading
-        */
-       private function fetchOrRegenerate( $key, $ttl, $callback, array $opts ) {
-               $checkKeys = $opts['checkKeys'] ?? [];
-               $graceTTL = $opts['graceTTL'] ?? self::GRACE_TTL_NONE;
-               $minAsOf = $opts['minAsOf'] ?? self::MIN_TIMESTAMP_NONE;
-               $hotTTR = $opts['hotTTR'] ?? self::HOT_TTR;
-               $lowTTL = $opts['lowTTL'] ?? min( self::LOW_TTL, $ttl );
-               $ageNew = $opts['ageNew'] ?? self::AGE_NEW;
-               $touchedCb = $opts['touchedCallback'] ?? null;
-               $initialTime = $this->getCurrentTime();
-
-               $kClass = $this->determineKeyClassForStats( $key );
-
-               // Get the current key value and its metadata
-               $curTTL = self::PASS_BY_REF;
-               $curInfo = self::PASS_BY_REF; /** @var array $curInfo */
-               $curValue = $this->get( $key, $curTTL, $checkKeys, $curInfo );
-               // Apply any $touchedCb invalidation timestamp to get the "last purge timestamp"
-               list( $curTTL, $LPT ) = $this->resolveCTL( $curValue, $curTTL, $curInfo, $touchedCb );
-               // Use the cached value if it exists and is not due for synchronous regeneration
-               if (
-                       $this->isValid( $curValue, $curInfo['asOf'], $minAsOf ) &&
-                       $this->isAliveOrInGracePeriod( $curTTL, $graceTTL )
-               ) {
-                       $preemptiveRefresh = (
-                               $this->worthRefreshExpiring( $curTTL, $lowTTL ) ||
-                               $this->worthRefreshPopular( $curInfo['asOf'], $ageNew, $hotTTR, $initialTime )
-                       );
-                       if ( !$preemptiveRefresh ) {
-                               $this->stats->increment( "wanobjectcache.$kClass.hit.good" );
-
-                               return [ $curValue, $curInfo['version'], $curInfo['asOf'] ];
-                       } elseif ( $this->scheduleAsyncRefresh( $key, $ttl, $callback, $opts ) ) {
-                               $this->stats->increment( "wanobjectcache.$kClass.hit.refresh" );
-
-                               return [ $curValue, $curInfo['version'], $curInfo['asOf'] ];
-                       }
-               }
-
-               // Determine if there is stale or volatile cached value that is still usable
-               $isKeyTombstoned = ( $curInfo['tombAsOf'] !== null );
-               if ( $isKeyTombstoned ) {
-                       // Key is write-holed; use the (volatile) interim key as an alternative
-                       list( $possValue, $possInfo ) = $this->getInterimValue( $key, $minAsOf );
-                       // Update the "last purge time" since the $touchedCb timestamp depends on $value
-                       $LPT = $this->resolveTouched( $possValue, $LPT, $touchedCb );
-               } else {
-                       $possValue = $curValue;
-                       $possInfo = $curInfo;
-               }
-
-               // Avoid overhead from callback runs, regeneration locks, and cache sets during
-               // hold-off periods for the key by reusing very recently generated cached values
-               if (
-                       $this->isValid( $possValue, $possInfo['asOf'], $minAsOf, $LPT ) &&
-                       $this->isVolatileValueAgeNegligible( $initialTime - $possInfo['asOf'] )
-               ) {
-                       $this->stats->increment( "wanobjectcache.$kClass.hit.volatile" );
-
-                       return [ $possValue, $possInfo['version'], $curInfo['asOf'] ];
-               }
-
-               $lockTSE = $opts['lockTSE'] ?? self::TSE_NONE;
-               $busyValue = $opts['busyValue'] ?? null;
-               $staleTTL = $opts['staleTTL'] ?? self::STALE_TTL_NONE;
-               $version = $opts['version'] ?? null;
-
-               // Determine whether one thread per datacenter should handle regeneration at a time
-               $useRegenerationLock =
-                       // Note that since tombstones no-op set(), $lockTSE and $curTTL cannot be used to
-                       // deduce the key hotness because |$curTTL| will always keep increasing until the
-                       // tombstone expires or is overwritten by a new tombstone. Also, even if $lockTSE
-                       // is not set, constant regeneration of a key for the tombstone lifetime might be
-                       // very expensive. Assume tombstoned keys are possibly hot in order to reduce
-                       // the risk of high regeneration load after the delete() method is called.
-                       $isKeyTombstoned ||
-                       // Assume a key is hot if requested soon ($lockTSE seconds) after invalidation.
-                       // This avoids stampedes when timestamps from $checkKeys/$touchedCb bump.
-                       ( $curTTL !== null && $curTTL <= 0 && abs( $curTTL ) <= $lockTSE ) ||
-                       // Assume a key is hot if there is no value and a busy fallback is given.
-                       // This avoids stampedes on eviction or preemptive regeneration taking too long.
-                       ( $busyValue !== null && $possValue === false );
-
-               // If a regeneration lock is required, threads that do not get the lock will try to use
-               // the stale value, the interim value, or the $busyValue placeholder, in that order. If
-               // none of those are set then all threads will bypass the lock and regenerate the value.
-               $hasLock = $useRegenerationLock && $this->claimStampedeLock( $key );
-               if ( $useRegenerationLock && !$hasLock ) {
-                       if ( $this->isValid( $possValue, $possInfo['asOf'], $minAsOf ) ) {
-                               $this->stats->increment( "wanobjectcache.$kClass.hit.stale" );
-
-                               return [ $possValue, $possInfo['version'], $curInfo['asOf'] ];
-                       } elseif ( $busyValue !== null ) {
-                               $miss = is_infinite( $minAsOf ) ? 'renew' : 'miss';
-                               $this->stats->increment( "wanobjectcache.$kClass.$miss.busy" );
-
-                               return [
-                                       is_callable( $busyValue ) ? $busyValue() : $busyValue,
-                                       $version,
-                                       $curInfo['asOf']
-                               ];
-                       }
-               }
-
-               // Generate the new value given any prior value with a matching version
-               $setOpts = [];
-               $preCallbackTime = $this->getCurrentTime();
-               ++$this->callbackDepth;
-               try {
-                       $value = $callback(
-                               ( $curInfo['version'] === $version ) ? $curValue : false,
-                               $ttl,
-                               $setOpts,
-                               ( $curInfo['version'] === $version ) ? $curInfo['asOf'] : null
-                       );
-               } finally {
-                       --$this->callbackDepth;
-               }
-               $postCallbackTime = $this->getCurrentTime();
-
-               // How long it took to fetch, validate, and generate the value
-               $elapsed = max( $postCallbackTime - $initialTime, 0.0 );
-
-               // Attempt to save the newly generated value if applicable
-               if (
-                       // Callback yielded a cacheable value
-                       ( $value !== false && $ttl >= 0 ) &&
-                       // Current thread was not raced out of a regeneration lock or key is tombstoned
-                       ( !$useRegenerationLock || $hasLock || $isKeyTombstoned ) &&
-                       // Key does not appear to be undergoing a set() stampede
-                       $this->checkAndSetCooloff( $key, $kClass, $elapsed, $lockTSE, $hasLock )
-               ) {
-                       // How long it took to generate the value
-                       $walltime = max( $postCallbackTime - $preCallbackTime, 0.0 );
-                       $this->stats->timing( "wanobjectcache.$kClass.regen_walltime", 1e3 * $walltime );
-                       // If the key is write-holed then use the (volatile) interim key as an alternative
-                       if ( $isKeyTombstoned ) {
-                               $this->setInterimValue( $key, $value, $lockTSE, $version, $walltime );
-                       } else {
-                               $finalSetOpts = [
-                                       'since' => $setOpts['since'] ?? $preCallbackTime,
-                                       'version' => $version,
-                                       'staleTTL' => $staleTTL,
-                                       'lockTSE' => $lockTSE, // informs lag vs performance trade-offs
-                                       'creating' => ( $curValue === false ), // optimization
-                                       'walltime' => $walltime
-                               ] + $setOpts;
-                               $this->set( $key, $value, $ttl, $finalSetOpts );
-                       }
-               }
-
-               $this->yieldStampedeLock( $key, $hasLock );
-
-               $miss = is_infinite( $minAsOf ) ? 'renew' : 'miss';
-               $this->stats->increment( "wanobjectcache.$kClass.$miss.compute" );
-
-               return [ $value, $version, $curInfo['asOf'] ];
-       }
-
-       /**
-        * @param string $key
-        * @return bool Success
-        */
-       private function claimStampedeLock( $key ) {
-               // Note that locking is not bypassed due to I/O errors; this avoids stampedes
-               return $this->cache->add( self::$MUTEX_KEY_PREFIX . $key, 1, self::$LOCK_TTL );
-       }
-
-       /**
-        * @param string $key
-        * @param bool $hasLock
-        */
-       private function yieldStampedeLock( $key, $hasLock ) {
-               if ( $hasLock ) {
-                       // The backend might be a mcrouter proxy set to broadcast DELETE to *all* the local
-                       // datacenter cache servers via OperationSelectorRoute (for increased consistency).
-                       // Since that would be excessive for these locks, use TOUCH to expire the key.
-                       $this->cache->changeTTL( self::$MUTEX_KEY_PREFIX . $key, $this->getCurrentTime() - 60 );
-               }
-       }
-
-       /**
-        * @param float $age Age of volatile/interim key in seconds
-        * @return bool Whether the age of a volatile value is negligible
-        */
-       private function isVolatileValueAgeNegligible( $age ) {
-               return ( $age < mt_rand( self::$RECENT_SET_LOW_MS, self::$RECENT_SET_HIGH_MS ) / 1e3 );
-       }
-
-       /**
-        * @param string $key
-        * @param string $kClass
-        * @param float $elapsed Seconds spent regenerating the value
-        * @param float $lockTSE
-        * @param bool $hasLock
-        * @return bool Whether it is OK to proceed with a key set operation
-        */
-       private function checkAndSetCooloff( $key, $kClass, $elapsed, $lockTSE, $hasLock ) {
-               $this->stats->timing( "wanobjectcache.$kClass.regen_set_delay", 1e3 * $elapsed );
-
-               // If $lockTSE is set, the lock was bypassed because there was no stale/interim value,
-               // and $elapsed indicates that regeration is slow, then there is a risk of set()
-               // stampedes with large blobs. With a typical scale-out infrastructure, CPU and query
-               // load from $callback invocations is distributed among appservers and replica DBs,
-               // but cache operations for a given key route to a single cache server (e.g. striped
-               // consistent hashing).
-               if ( $lockTSE < 0 || $hasLock ) {
-                       return true; // either not a priori hot or thread has the lock
-               } elseif ( $elapsed <= self::$SET_DELAY_HIGH_MS * 1e3 ) {
-                       return true; // not enough time for threads to pile up
-               }
-
-               $this->cache->clearLastError();
-               if (
-                       !$this->cache->add( self::$COOLOFF_KEY_PREFIX . $key, 1, self::$COOLOFF_TTL ) &&
-                       // Don't treat failures due to I/O errors as the key being in cooloff
-                       $this->cache->getLastError() === BagOStuff::ERR_NONE
-               ) {
-                       $this->stats->increment( "wanobjectcache.$kClass.cooloff_bounce" );
-
-                       return false;
-               }
-
-               return true;
-       }
-
-       /**
-        * @param mixed $value
-        * @param float|null $curTTL
-        * @param array $curInfo
-        * @param callable|null $touchedCallback
-        * @return array (current time left or null, UNIX timestamp of last purge or null)
-        * @note Callable type hints are not used to avoid class-autoloading
-        */
-       private function resolveCTL( $value, $curTTL, $curInfo, $touchedCallback ) {
-               if ( $touchedCallback === null || $value === false ) {
-                       return [ $curTTL, max( $curInfo['tombAsOf'], $curInfo['lastCKPurge'] ) ];
-               }
-
-               $touched = $touchedCallback( $value );
-               if ( $touched !== null && $touched >= $curInfo['asOf'] ) {
-                       $curTTL = min( $curTTL, self::$TINY_NEGATIVE, $curInfo['asOf'] - $touched );
-               }
-
-               return [ $curTTL, max( $curInfo['tombAsOf'], $curInfo['lastCKPurge'], $touched ) ];
-       }
-
-       /**
-        * @param mixed $value
-        * @param float|null $lastPurge
-        * @param callable|null $touchedCallback
-        * @return float|null UNIX timestamp of last purge or null
-        * @note Callable type hints are not used to avoid class-autoloading
-        */
-       private function resolveTouched( $value, $lastPurge, $touchedCallback ) {
-               return ( $touchedCallback === null || $value === false )
-                       ? $lastPurge // nothing to derive the "touched timestamp" from
-                       : max( $touchedCallback( $value ), $lastPurge );
-       }
-
-       /**
-        * @param string $key
-        * @param float $minAsOf Minimum acceptable "as of" timestamp
-        * @return array (cached value or false, cache key metadata map)
-        */
-       private function getInterimValue( $key, $minAsOf ) {
-               $now = $this->getCurrentTime();
-
-               if ( $this->useInterimHoldOffCaching ) {
-                       $wrapped = $this->cache->get( self::$INTERIM_KEY_PREFIX . $key );
-
-                       list( $value, $keyInfo ) = $this->unwrap( $wrapped, $now );
-                       if ( $this->isValid( $value, $keyInfo['asOf'], $minAsOf ) ) {
-                               return [ $value, $keyInfo ];
-                       }
-               }
-
-               return $this->unwrap( false, $now );
-       }
-
-       /**
-        * @param string $key
-        * @param mixed $value
-        * @param int $ttl
-        * @param int|null $version Value version number
-        * @param float $walltime How long it took to generate the value in seconds
-        */
-       private function setInterimValue( $key, $value, $ttl, $version, $walltime ) {
-               $ttl = max( self::$INTERIM_KEY_TTL, (int)$ttl );
-
-               $wrapped = $this->wrap( $value, $ttl, $version, $this->getCurrentTime(), $walltime );
-               $this->cache->merge(
-                       self::$INTERIM_KEY_PREFIX . $key,
-                       function () use ( $wrapped ) {
-                               return $wrapped;
-                       },
-                       $ttl,
-                       1
-               );
-       }
-
-       /**
-        * Method to fetch multiple cache keys at once with regeneration
-        *
-        * This works the same as getWithSetCallback() except:
-        *   - a) The $keys argument expects the result of WANObjectCache::makeMultiKeys()
-        *   - b) The $callback argument expects a callback taking the following arguments:
-        *         - $id: ID of an entity to query
-        *         - $oldValue : the prior cache value or false if none was present
-        *         - &$ttl : a reference to the new value TTL in seconds
-        *         - &$setOpts : a reference to options for set() which can be altered
-        *         - $oldAsOf : generation UNIX timestamp of $oldValue or null if not present
-        *        Aside from the additional $id argument, the other arguments function the same
-        *        way they do in getWithSetCallback().
-        *   - c) The return value is a map of (cache key => value) in the order of $keyedIds
-        *
-        * @see WANObjectCache::getWithSetCallback()
-        * @see WANObjectCache::getMultiWithUnionSetCallback()
-        *
-        * Example usage:
-        * @code
-        *     $rows = $cache->getMultiWithSetCallback(
-        *         // Map of cache keys to entity IDs
-        *         $cache->makeMultiKeys(
-        *             $this->fileVersionIds(),
-        *             function ( $id ) use ( $cache ) {
-        *                 return $cache->makeKey( 'file-version', $id );
-        *             }
-        *         ),
-        *         // Time-to-live (in seconds)
-        *         $cache::TTL_DAY,
-        *         // Function that derives the new key value
-        *         function ( $id, $oldValue, &$ttl, array &$setOpts ) {
-        *             $dbr = wfGetDB( DB_REPLICA );
-        *             // Account for any snapshot/replica DB lag
-        *             $setOpts += Database::getCacheSetOptions( $dbr );
-        *
-        *             // Load the row for this file
-        *             $queryInfo = File::getQueryInfo();
-        *             $row = $dbr->selectRow(
-        *                 $queryInfo['tables'],
-        *                 $queryInfo['fields'],
-        *                 [ 'id' => $id ],
-        *                 __METHOD__,
-        *                 [],
-        *                 $queryInfo['joins']
-        *             );
-        *
-        *             return $row ? (array)$row : false;
-        *         },
-        *         [
-        *             // Process cache for 30 seconds
-        *             'pcTTL' => 30,
-        *             // Use a dedicated 500 item cache (initialized on-the-fly)
-        *             'pcGroup' => 'file-versions:500'
-        *         ]
-        *     );
-        *     $files = array_map( [ __CLASS__, 'newFromRow' ], $rows );
-        * @endcode
-        *
-        * @param ArrayIterator $keyedIds Result of WANObjectCache::makeMultiKeys()
-        * @param int $ttl Seconds to live for key updates
-        * @param callable $callback Callback the yields entity regeneration callbacks
-        * @param array $opts Options map
-        * @return mixed[] Map of (cache key => value) in the same order as $keyedIds
-        * @since 1.28
-        */
-       final public function getMultiWithSetCallback(
-               ArrayIterator $keyedIds, $ttl, callable $callback, array $opts = []
-       ) {
-               // Load required keys into process cache in one go
-               $this->warmupCache = $this->getRawKeysForWarmup(
-                       $this->getNonProcessCachedMultiKeys( $keyedIds, $opts ),
-                       $opts['checkKeys'] ?? []
-               );
-               $this->warmupKeyMisses = 0;
-
-               // Wrap $callback to match the getWithSetCallback() format while passing $id to $callback
-               $id = null; // current entity ID
-               $func = function ( $oldValue, &$ttl, &$setOpts, $oldAsOf ) use ( $callback, &$id ) {
-                       return $callback( $id, $oldValue, $ttl, $setOpts, $oldAsOf );
-               };
-
-               $values = [];
-               foreach ( $keyedIds as $key => $id ) { // preserve order
-                       $values[$key] = $this->getWithSetCallback( $key, $ttl, $func, $opts );
-               }
-
-               $this->warmupCache = [];
-
-               return $values;
-       }
-
-       /**
-        * Method to fetch/regenerate multiple cache keys at once
-        *
-        * This works the same as getWithSetCallback() except:
-        *   - a) The $keys argument expects the result of WANObjectCache::makeMultiKeys()
-        *   - b) The $callback argument expects a callback returning a map of (ID => new value)
-        *        for all entity IDs in $ids and it takes the following arguments:
-        *          - $ids: a list of entity IDs that require cache regeneration
-        *          - &$ttls: a reference to the (entity ID => new TTL) map
-        *          - &$setOpts: a reference to options for set() which can be altered
-        *   - c) The return value is a map of (cache key => value) in the order of $keyedIds
-        *   - d) The "lockTSE" and "busyValue" options are ignored
-        *
-        * @see WANObjectCache::getWithSetCallback()
-        * @see WANObjectCache::getMultiWithSetCallback()
-        *
-        * Example usage:
-        * @code
-        *     $rows = $cache->getMultiWithUnionSetCallback(
-        *         // Map of cache keys to entity IDs
-        *         $cache->makeMultiKeys(
-        *             $this->fileVersionIds(),
-        *             function ( $id ) use ( $cache ) {
-        *                 return $cache->makeKey( 'file-version', $id );
-        *             }
-        *         ),
-        *         // Time-to-live (in seconds)
-        *         $cache::TTL_DAY,
-        *         // Function that derives the new key value
-        *         function ( array $ids, array &$ttls, array &$setOpts ) {
-        *             $dbr = wfGetDB( DB_REPLICA );
-        *             // Account for any snapshot/replica DB lag
-        *             $setOpts += Database::getCacheSetOptions( $dbr );
-        *
-        *             // Load the rows for these files
-        *             $rows = [];
-        *             $queryInfo = File::getQueryInfo();
-        *             $res = $dbr->select(
-        *                 $queryInfo['tables'],
-        *                 $queryInfo['fields'],
-        *                 [ 'id' => $ids ],
-        *                 __METHOD__,
-        *                 [],
-        *                 $queryInfo['joins']
-        *             );
-        *             foreach ( $res as $row ) {
-        *                 $rows[$row->id] = $row;
-        *                 $mtime = wfTimestamp( TS_UNIX, $row->timestamp );
-        *                 $ttls[$row->id] = $this->adaptiveTTL( $mtime, $ttls[$row->id] );
-        *             }
-        *
-        *             return $rows;
-        *         },
-        *         ]
-        *     );
-        *     $files = array_map( [ __CLASS__, 'newFromRow' ], $rows );
-        * @endcode
-        *
-        * @param ArrayIterator $keyedIds Result of WANObjectCache::makeMultiKeys()
-        * @param int $ttl Seconds to live for key updates
-        * @param callable $callback Callback the yields entity regeneration callbacks
-        * @param array $opts Options map
-        * @return mixed[] Map of (cache key => value) in the same order as $keyedIds
-        * @since 1.30
-        */
-       final public function getMultiWithUnionSetCallback(
-               ArrayIterator $keyedIds, $ttl, callable $callback, array $opts = []
-       ) {
-               $checkKeys = $opts['checkKeys'] ?? [];
-               unset( $opts['lockTSE'] ); // incompatible
-               unset( $opts['busyValue'] ); // incompatible
-
-               // Load required keys into process cache in one go
-               $keysByIdGet = $this->getNonProcessCachedMultiKeys( $keyedIds, $opts );
-               $this->warmupCache = $this->getRawKeysForWarmup( $keysByIdGet, $checkKeys );
-               $this->warmupKeyMisses = 0;
-
-               // IDs of entities known to be in need of regeneration
-               $idsRegen = [];
-
-               // Find out which keys are missing/deleted/stale
-               $curTTLs = [];
-               $asOfs = [];
-               $curByKey = $this->getMulti( $keysByIdGet, $curTTLs, $checkKeys, $asOfs );
-               foreach ( $keysByIdGet as $id => $key ) {
-                       if ( !array_key_exists( $key, $curByKey ) || $curTTLs[$key] < 0 ) {
-                               $idsRegen[] = $id;
-                       }
-               }
-
-               // Run the callback to populate the regeneration value map for all required IDs
-               $newSetOpts = [];
-               $newTTLsById = array_fill_keys( $idsRegen, $ttl );
-               $newValsById = $idsRegen ? $callback( $idsRegen, $newTTLsById, $newSetOpts ) : [];
-
-               // Wrap $callback to match the getWithSetCallback() format while passing $id to $callback
-               $id = null; // current entity ID
-               $func = function ( $oldValue, &$ttl, &$setOpts, $oldAsOf )
-                       use ( $callback, &$id, $newValsById, $newTTLsById, $newSetOpts )
-               {
-                       if ( array_key_exists( $id, $newValsById ) ) {
-                               // Value was already regerated as expected, so use the value in $newValsById
-                               $newValue = $newValsById[$id];
-                               $ttl = $newTTLsById[$id];
-                               $setOpts = $newSetOpts;
-                       } else {
-                               // Pre-emptive/popularity refresh and version mismatch cases are not detected
-                               // above and thus $newValsById has no entry. Run $callback on this single entity.
-                               $ttls = [ $id => $ttl ];
-                               $newValue = $callback( [ $id ], $ttls, $setOpts )[$id];
-                               $ttl = $ttls[$id];
-                       }
-
-                       return $newValue;
-               };
-
-               // Run the cache-aside logic using warmupCache instead of persistent cache queries
-               $values = [];
-               foreach ( $keyedIds as $key => $id ) { // preserve order
-                       $values[$key] = $this->getWithSetCallback( $key, $ttl, $func, $opts );
-               }
-
-               $this->warmupCache = [];
-
-               return $values;
-       }
-
-       /**
-        * Set a key to soon expire in the local cluster if it pre-dates $purgeTimestamp
-        *
-        * This sets stale keys' time-to-live at HOLDOFF_TTL seconds, which both avoids
-        * broadcasting in mcrouter setups and also avoids races with new tombstones.
-        *
-        * @param string $key Cache key
-        * @param int $purgeTimestamp UNIX timestamp of purge
-        * @param bool &$isStale Whether the key is stale
-        * @return bool Success
-        * @since 1.28
-        */
-       final public function reap( $key, $purgeTimestamp, &$isStale = false ) {
-               $minAsOf = $purgeTimestamp + self::HOLDOFF_TTL;
-               $wrapped = $this->cache->get( self::$VALUE_KEY_PREFIX . $key );
-               if ( is_array( $wrapped ) && $wrapped[self::$FLD_TIME] < $minAsOf ) {
-                       $isStale = true;
-                       $this->logger->warning( "Reaping stale value key '$key'." );
-                       $ttlReap = self::HOLDOFF_TTL; // avoids races with tombstone creation
-                       $ok = $this->cache->changeTTL( self::$VALUE_KEY_PREFIX . $key, $ttlReap );
-                       if ( !$ok ) {
-                               $this->logger->error( "Could not complete reap of key '$key'." );
-                       }
-
-                       return $ok;
-               }
-
-               $isStale = false;
-
-               return true;
-       }
-
-       /**
-        * Set a "check" key to soon expire in the local cluster if it pre-dates $purgeTimestamp
-        *
-        * @param string $key Cache key
-        * @param int $purgeTimestamp UNIX timestamp of purge
-        * @param bool &$isStale Whether the key is stale
-        * @return bool Success
-        * @since 1.28
-        */
-       final public function reapCheckKey( $key, $purgeTimestamp, &$isStale = false ) {
-               $purge = $this->parsePurgeValue( $this->cache->get( self::$TIME_KEY_PREFIX . $key ) );
-               if ( $purge && $purge[self::$PURGE_TIME] < $purgeTimestamp ) {
-                       $isStale = true;
-                       $this->logger->warning( "Reaping stale check key '$key'." );
-                       $ok = $this->cache->changeTTL( self::$TIME_KEY_PREFIX . $key, self::TTL_SECOND );
-                       if ( !$ok ) {
-                               $this->logger->error( "Could not complete reap of check key '$key'." );
-                       }
-
-                       return $ok;
-               }
-
-               $isStale = false;
-
-               return false;
-       }
-
-       /**
-        * @see BagOStuff::makeKey()
-        * @param string $class Key class
-        * @param string ...$components Key components (starting with a key collection name)
-        * @return string Colon-delimited list of $keyspace followed by escaped components
-        * @since 1.27
-        */
-       public function makeKey( $class, ...$components ) {
-               return $this->cache->makeKey( ...func_get_args() );
-       }
-
-       /**
-        * @see BagOStuff::makeGlobalKey()
-        * @param string $class Key class
-        * @param string ...$components Key components (starting with a key collection name)
-        * @return string Colon-delimited list of $keyspace followed by escaped components
-        * @since 1.27
-        */
-       public function makeGlobalKey( $class, ...$components ) {
-               return $this->cache->makeGlobalKey( ...func_get_args() );
-       }
-
-       /**
-        * Hash a possibly long string into a suitable component for makeKey()/makeGlobalKey()
-        *
-        * @param string $component A raw component used in building a cache key
-        * @return string 64 character HMAC using a stable secret for public collision resistance
-        * @since 1.34
-        */
-       public function hash256( $component ) {
-               return hash_hmac( 'sha256', $component, $this->secret );
-       }
-
-       /**
-        * Get an iterator of (cache key => entity ID) for a list of entity IDs
-        *
-        * The callback takes an ID string and returns a key via makeKey()/makeGlobalKey().
-        * There should be no network nor filesystem I/O used in the callback. The entity
-        * ID/key mapping must be 1:1 or an exception will be thrown. If hashing is needed,
-        * then use the hash256() method.
-        *
-        * Example usage for the default keyspace:
-        * @code
-        *     $keyedIds = $cache->makeMultiKeys(
-        *         $modules,
-        *         function ( $module ) use ( $cache ) {
-        *             return $cache->makeKey( 'module-info', $module );
-        *         }
-        *     );
-        * @endcode
-        *
-        * Example usage for mixed default and global keyspace:
-        * @code
-        *     $keyedIds = $cache->makeMultiKeys(
-        *         $filters,
-        *         function ( $filter ) use ( $cache ) {
-        *             return ( strpos( $filter, 'central:' ) === 0 )
-        *                 ? $cache->makeGlobalKey( 'regex-filter', $filter )
-        *                 : $cache->makeKey( 'regex-filter', $filter )
-        *         }
-        *     );
-        * @endcode
-        *
-        * Example usage with hashing:
-        * @code
-        *     $keyedIds = $cache->makeMultiKeys(
-        *         $urls,
-        *         function ( $url ) use ( $cache ) {
-        *             return $cache->makeKey( 'url-info', $cache->hash256( $url ) );
-        *         }
-        *     );
-        * @endcode
-        *
-        * @see WANObjectCache::makeKey()
-        * @see WANObjectCache::makeGlobalKey()
-        * @see WANObjectCache::hash256()
-        *
-        * @param string[]|int[] $ids List of entity IDs
-        * @param callable $keyCallback Function returning makeKey()/makeGlobalKey() on the input ID
-        * @return ArrayIterator Iterator of (cache key => ID); order of $ids is preserved
-        * @throws UnexpectedValueException
-        * @since 1.28
-        */
-       final public function makeMultiKeys( array $ids, $keyCallback ) {
-               $idByKey = [];
-               foreach ( $ids as $id ) {
-                       // Discourage triggering of automatic makeKey() hashing in some backends
-                       if ( strlen( $id ) > 64 ) {
-                               $this->logger->warning( __METHOD__ . ": long ID '$id'; use hash256()" );
-                       }
-                       $key = $keyCallback( $id, $this );
-                       // Edge case: ignore key collisions due to duplicate $ids like "42" and 42
-                       if ( !isset( $idByKey[$key] ) ) {
-                               $idByKey[$key] = $id;
-                       } elseif ( (string)$id !== (string)$idByKey[$key] ) {
-                               throw new UnexpectedValueException(
-                                       "Cache key collision; IDs ('$id','{$idByKey[$key]}') map to '$key'"
-                               );
-                       }
-               }
-
-               return new ArrayIterator( $idByKey );
-       }
-
-       /**
-        * Get an (ID => value) map from (i) a non-unique list of entity IDs, and (ii) the list
-        * of corresponding entity values by first appearance of each ID in the entity ID list
-        *
-        * For use with getMultiWithSetCallback() and getMultiWithUnionSetCallback().
-        *
-        * *Only* use this method if the entity ID/key mapping is trivially 1:1 without exception.
-        * Key generation method must utitilize the *full* entity ID in the key (not a hash of it).
-        *
-        * Example usage:
-        * @code
-        *     $poems = $cache->getMultiWithSetCallback(
-        *         $cache->makeMultiKeys(
-        *             $uuids,
-        *             function ( $uuid ) use ( $cache ) {
-        *                 return $cache->makeKey( 'poem', $uuid );
-        *             }
-        *         ),
-        *         $cache::TTL_DAY,
-        *         function ( $uuid ) use ( $url ) {
-        *             return $this->http->run( [ 'method' => 'GET', 'url' => "$url/$uuid" ] );
-        *         }
-        *     );
-        *     $poemsByUUID = $cache->multiRemap( $uuids, $poems );
-        * @endcode
-        *
-        * @see WANObjectCache::makeMultiKeys()
-        * @see WANObjectCache::getMultiWithSetCallback()
-        * @see WANObjectCache::getMultiWithUnionSetCallback()
-        *
-        * @param string[]|int[] $ids Entity ID list makeMultiKeys()
-        * @param mixed[] $res Result of getMultiWithSetCallback()/getMultiWithUnionSetCallback()
-        * @return mixed[] Map of (ID => value); order of $ids is preserved
-        * @since 1.34
-        */
-       final public function multiRemap( array $ids, array $res ) {
-               if ( count( $ids ) !== count( $res ) ) {
-                       // If makeMultiKeys() is called on a list of non-unique IDs, then the resulting
-                       // ArrayIterator will have less entries due to "first appearance" de-duplication
-                       $ids = array_keys( array_flip( $ids ) );
-                       if ( count( $ids ) !== count( $res ) ) {
-                               throw new UnexpectedValueException( "Multi-key result does not match ID list" );
-                       }
-               }
-
-               return array_combine( $ids, $res );
-       }
-
-       /**
-        * Get the "last error" registered; clearLastError() should be called manually
-        * @return int ERR_* class constant for the "last error" registry
-        */
-       final public function getLastError() {
-               $code = $this->cache->getLastError();
-               switch ( $code ) {
-                       case BagOStuff::ERR_NONE:
-                               return self::ERR_NONE;
-                       case BagOStuff::ERR_NO_RESPONSE:
-                               return self::ERR_NO_RESPONSE;
-                       case BagOStuff::ERR_UNREACHABLE:
-                               return self::ERR_UNREACHABLE;
-                       default:
-                               return self::ERR_UNEXPECTED;
-               }
-       }
-
-       /**
-        * Clear the "last error" registry
-        */
-       final public function clearLastError() {
-               $this->cache->clearLastError();
-       }
-
-       /**
-        * Clear the in-process caches; useful for testing
-        *
-        * @since 1.27
-        */
-       public function clearProcessCache() {
-               $this->processCaches = [];
-       }
-
-       /**
-        * Enable or disable the use of brief caching for tombstoned keys
-        *
-        * When a key is purged via delete(), there normally is a period where caching
-        * is hold-off limited to an extremely short time. This method will disable that
-        * caching, forcing the callback to run for any of:
-        *   - WANObjectCache::getWithSetCallback()
-        *   - WANObjectCache::getMultiWithSetCallback()
-        *   - WANObjectCache::getMultiWithUnionSetCallback()
-        *
-        * This is useful when both:
-        *   - a) the database used by the callback is known to be up-to-date enough
-        *        for some particular purpose (e.g. replica DB has applied transaction X)
-        *   - b) the caller needs to exploit that fact, and therefore needs to avoid the
-        *        use of inherently volatile and possibly stale interim keys
-        *
-        * @see WANObjectCache::delete()
-        * @param bool $enabled Whether to enable interim caching
-        * @since 1.31
-        */
-       final public function useInterimHoldOffCaching( $enabled ) {
-               $this->useInterimHoldOffCaching = $enabled;
-       }
-
-       /**
-        * @param int $flag ATTR_* class constant
-        * @return int QOS_* class constant
-        * @since 1.28
-        */
-       public function getQoS( $flag ) {
-               return $this->cache->getQoS( $flag );
-       }
-
-       /**
-        * Get a TTL that is higher for objects that have not changed recently
-        *
-        * This is useful for keys that get explicit purges and DB or purge relay
-        * lag is a potential concern (especially how it interacts with CDN cache)
-        *
-        * Example usage:
-        * @code
-        *     // Last-modified time of page
-        *     $mtime = wfTimestamp( TS_UNIX, $page->getTimestamp() );
-        *     // Get adjusted TTL. If $mtime is 3600 seconds ago and $minTTL/$factor left at
-        *     // defaults, then $ttl is 3600 * .2 = 720. If $minTTL was greater than 720, then
-        *     // $ttl would be $minTTL. If $maxTTL was smaller than 720, $ttl would be $maxTTL.
-        *     $ttl = $cache->adaptiveTTL( $mtime, $cache::TTL_DAY );
-        * @endcode
-        *
-        * Another use case is when there are no applicable "last modified" fields in the DB,
-        * and there are too many dependencies for explicit purges to be viable, and the rate of
-        * change to relevant content is unstable, and it is highly valued to have the cached value
-        * be as up-to-date as possible.
-        *
-        * Example usage:
-        * @code
-        *     $query = "<some complex query>";
-        *     $idListFromComplexQuery = $cache->getWithSetCallback(
-        *         $cache->makeKey( 'complex-graph-query', $hashOfQuery ),
-        *         GraphQueryClass::STARTING_TTL,
-        *         function ( $oldValue, &$ttl, array &$setOpts, $oldAsOf ) use ( $query, $cache ) {
-        *             $gdb = $this->getReplicaGraphDbConnection();
-        *             // Account for any snapshot/replica DB lag
-        *             $setOpts += GraphDatabase::getCacheSetOptions( $gdb );
-        *
-        *             $newList = iterator_to_array( $gdb->query( $query ) );
-        *             sort( $newList, SORT_NUMERIC ); // normalize
-        *
-        *             $minTTL = GraphQueryClass::MIN_TTL;
-        *             $maxTTL = GraphQueryClass::MAX_TTL;
-        *             if ( $oldValue !== false ) {
-        *                 // Note that $oldAsOf is the last time this callback ran
-        *                 $ttl = ( $newList === $oldValue )
-        *                     // No change: cache for 150% of the age of $oldValue
-        *                     ? $cache->adaptiveTTL( $oldAsOf, $maxTTL, $minTTL, 1.5 )
-        *                     // Changed: cache for 50% of the age of $oldValue
-        *                     : $cache->adaptiveTTL( $oldAsOf, $maxTTL, $minTTL, .5 );
-        *             }
-        *
-        *             return $newList;
-        *        },
-        *        [
-        *             // Keep stale values around for doing comparisons for TTL calculations.
-        *             // High values improve long-tail keys hit-rates, though might waste space.
-        *             'staleTTL' => GraphQueryClass::GRACE_TTL
-        *        ]
-        *     );
-        * @endcode
-        *
-        * @param int|float $mtime UNIX timestamp
-        * @param int $maxTTL Maximum TTL (seconds)
-        * @param int $minTTL Minimum TTL (seconds); Default: 30
-        * @param float $factor Value in the range (0,1); Default: .2
-        * @return int Adaptive TTL
-        * @since 1.28
-        */
-       public function adaptiveTTL( $mtime, $maxTTL, $minTTL = 30, $factor = 0.2 ) {
-               if ( is_float( $mtime ) || ctype_digit( $mtime ) ) {
-                       $mtime = (int)$mtime; // handle fractional seconds and string integers
-               }
-
-               if ( !is_int( $mtime ) || $mtime <= 0 ) {
-                       return $minTTL; // no last-modified time provided
-               }
-
-               $age = $this->getCurrentTime() - $mtime;
-
-               return (int)min( $maxTTL, max( $minTTL, $factor * $age ) );
-       }
-
-       /**
-        * @return int Number of warmup key cache misses last round
-        * @since 1.30
-        */
-       final public function getWarmupKeyMisses() {
-               return $this->warmupKeyMisses;
-       }
-
-       /**
-        * Do the actual async bus purge of a key
-        *
-        * This must set the key to "PURGED:<UNIX timestamp>:<holdoff>"
-        *
-        * @param string $key Cache key
-        * @param int $ttl Seconds to keep the tombstone around
-        * @param int $holdoff HOLDOFF_* constant controlling how long to ignore sets for this key
-        * @return bool Success
-        */
-       protected function relayPurge( $key, $ttl, $holdoff ) {
-               if ( $this->mcrouterAware ) {
-                       // See https://github.com/facebook/mcrouter/wiki/Multi-cluster-broadcast-setup
-                       // Wildcards select all matching routes, e.g. the WAN cluster on all DCs
-                       $ok = $this->cache->set(
-                               "/*/{$this->cluster}/{$key}",
-                               $this->makePurgeValue( $this->getCurrentTime(), self::HOLDOFF_TTL_NONE ),
-                               $ttl
-                       );
-               } else {
-                       // This handles the mcrouter and the single-DC case
-                       $ok = $this->cache->set(
-                               $key,
-                               $this->makePurgeValue( $this->getCurrentTime(), self::HOLDOFF_TTL_NONE ),
-                               $ttl
-                       );
-               }
-
-               return $ok;
-       }
-
-       /**
-        * Do the actual async bus delete of a key
-        *
-        * @param string $key Cache key
-        * @return bool Success
-        */
-       protected function relayDelete( $key ) {
-               if ( $this->mcrouterAware ) {
-                       // See https://github.com/facebook/mcrouter/wiki/Multi-cluster-broadcast-setup
-                       // Wildcards select all matching routes, e.g. the WAN cluster on all DCs
-                       $ok = $this->cache->delete( "/*/{$this->cluster}/{$key}" );
-               } else {
-                       // Some other proxy handles broadcasting or there is only one datacenter
-                       $ok = $this->cache->delete( $key );
-               }
-
-               return $ok;
-       }
-
-       /**
-        * @param string $key
-        * @param int $ttl Seconds to live
-        * @param callable $callback
-        * @param array $opts
-        * @return bool Success
-        * @note Callable type hints are not used to avoid class-autoloading
-        */
-       private function scheduleAsyncRefresh( $key, $ttl, $callback, $opts ) {
-               if ( !$this->asyncHandler ) {
-                       return false;
-               }
-               // Update the cache value later, such during post-send of an HTTP request
-               $func = $this->asyncHandler;
-               $func( function () use ( $key, $ttl, $callback, $opts ) {
-                       $opts['minAsOf'] = INF; // force a refresh
-                       $this->fetchOrRegenerate( $key, $ttl, $callback, $opts );
-               } );
-
-               return true;
-       }
-
-       /**
-        * Check if a key is fresh or in the grace window and thus due for randomized reuse
-        *
-        * If $curTTL > 0 (e.g. not expired) this returns true. Otherwise, the chance of returning
-        * true decrease steadily from 100% to 0% as the |$curTTL| moves from 0 to $graceTTL seconds.
-        * This handles widely varying levels of cache access traffic.
-        *
-        * If $curTTL <= -$graceTTL (e.g. already expired), then this returns false.
-        *
-        * @param float $curTTL Approximate TTL left on the key if present
-        * @param int $graceTTL Consider using stale values if $curTTL is greater than this
-        * @return bool
-        */
-       private function isAliveOrInGracePeriod( $curTTL, $graceTTL ) {
-               if ( $curTTL > 0 ) {
-                       return true;
-               } elseif ( $graceTTL <= 0 ) {
-                       return false;
-               }
-
-               $ageStale = abs( $curTTL ); // seconds of staleness
-               $curGTTL = ( $graceTTL - $ageStale ); // current grace-time-to-live
-               if ( $curGTTL <= 0 ) {
-                       return false; //  already out of grace period
-               }
-
-               // Chance of using a stale value is the complement of the chance of refreshing it
-               return !$this->worthRefreshExpiring( $curGTTL, $graceTTL );
-       }
-
-       /**
-        * Check if a key is nearing expiration and thus due for randomized regeneration
-        *
-        * This returns false if $curTTL >= $lowTTL. Otherwise, the chance of returning true
-        * increases steadily from 0% to 100% as the $curTTL moves from $lowTTL to 0 seconds.
-        * This handles widely varying levels of cache access traffic.
-        *
-        * If $curTTL <= 0 (e.g. already expired), then this returns false.
-        *
-        * @param float $curTTL Approximate TTL left on the key if present
-        * @param float $lowTTL Consider a refresh when $curTTL is less than this
-        * @return bool
-        */
-       protected function worthRefreshExpiring( $curTTL, $lowTTL ) {
-               if ( $lowTTL <= 0 ) {
-                       return false;
-               } elseif ( $curTTL >= $lowTTL ) {
-                       return false;
-               } elseif ( $curTTL <= 0 ) {
-                       return false;
-               }
-
-               $chance = ( 1 - $curTTL / $lowTTL );
-
-               return mt_rand( 1, 1e9 ) <= 1e9 * $chance;
-       }
-
-       /**
-        * Check if a key is due for randomized regeneration due to its popularity
-        *
-        * This is used so that popular keys can preemptively refresh themselves for higher
-        * consistency (especially in the case of purge loss/delay). Unpopular keys can remain
-        * in cache with their high nominal TTL. This means popular keys keep good consistency,
-        * whether the data changes frequently or not, and long-tail keys get to stay in cache
-        * and get hits too. Similar to worthRefreshExpiring(), randomization is used.
-        *
-        * @param float $asOf UNIX timestamp of the value
-        * @param int $ageNew Age of key when this might recommend refreshing (seconds)
-        * @param int $timeTillRefresh Age of key when it should be refreshed if popular (seconds)
-        * @param float $now The current UNIX timestamp
-        * @return bool
-        */
-       protected function worthRefreshPopular( $asOf, $ageNew, $timeTillRefresh, $now ) {
-               if ( $ageNew < 0 || $timeTillRefresh <= 0 ) {
-                       return false;
-               }
-
-               $age = $now - $asOf;
-               $timeOld = $age - $ageNew;
-               if ( $timeOld <= 0 ) {
-                       return false;
-               }
-
-               $popularHitsPerSec = 1;
-               // Lifecycle is: new, ramp-up refresh chance, full refresh chance.
-               // Note that the "expected # of refreshes" for the ramp-up time range is half
-               // of what it would be if P(refresh) was at its full value during that time range.
-               $refreshWindowSec = max( $timeTillRefresh - $ageNew - self::$RAMPUP_TTL / 2, 1 );
-               // P(refresh) * (# hits in $refreshWindowSec) = (expected # of refreshes)
-               // P(refresh) * ($refreshWindowSec * $popularHitsPerSec) = 1 (by definition)
-               // P(refresh) = 1/($refreshWindowSec * $popularHitsPerSec)
-               $chance = 1 / ( $popularHitsPerSec * $refreshWindowSec );
-
-               // Ramp up $chance from 0 to its nominal value over RAMPUP_TTL seconds to avoid stampedes
-               $chance *= ( $timeOld <= self::$RAMPUP_TTL ) ? $timeOld / self::$RAMPUP_TTL : 1;
-
-               return mt_rand( 1, 1e9 ) <= 1e9 * $chance;
-       }
-
-       /**
-        * Check if $value is not false, versioned (if needed), and not older than $minTime (if set)
-        *
-        * @param array|bool $value
-        * @param float $asOf The time $value was generated
-        * @param float $minAsOf Minimum acceptable "as of" timestamp
-        * @param float|null $purgeTime The last time the value was invalidated
-        * @return bool
-        */
-       protected function isValid( $value, $asOf, $minAsOf, $purgeTime = null ) {
-               // Avoid reading any key not generated after the latest delete() or touch
-               $safeMinAsOf = max( $minAsOf, $purgeTime + self::$TINY_POSTIVE );
-
-               if ( $value === false ) {
-                       return false;
-               } elseif ( $safeMinAsOf > 0 && $asOf < $minAsOf ) {
-                       return false;
-               }
-
-               return true;
-       }
-
-       /**
-        * @param mixed $value
-        * @param int $ttl Seconds to live or zero for "indefinite"
-        * @param int|null $version Value version number or null if not versioned
-        * @param float $now Unix Current timestamp just before calling set()
-        * @param float $walltime How long it took to generate the value in seconds
-        * @return array
-        */
-       private function wrap( $value, $ttl, $version, $now, $walltime ) {
-               // Returns keys in ascending integer order for PHP7 array packing:
-               // https://nikic.github.io/2014/12/22/PHPs-new-hashtable-implementation.html
-               $wrapped = [
-                       self::$FLD_FORMAT_VERSION => self::$VERSION,
-                       self::$FLD_VALUE => $value,
-                       self::$FLD_TTL => $ttl,
-                       self::$FLD_TIME => $now
-               ];
-               if ( $version !== null ) {
-                       $wrapped[self::$FLD_VALUE_VERSION] = $version;
-               }
-               if ( $walltime >= self::$GENERATION_SLOW_SEC ) {
-                       $wrapped[self::$FLD_GENERATION_TIME] = $walltime;
-               }
-
-               return $wrapped;
-       }
-
-       /**
-        * @param array|string|bool $wrapped The entry at a cache key
-        * @param float $now Unix Current timestamp (preferrably pre-query)
-        * @return array (value or false if absent/tombstoned/malformed, value metadata map).
-        * The cache key metadata includes the following metadata:
-        *   - asOf: UNIX timestamp of the value or null if there is no value
-        *   - curTTL: remaining time-to-live (negative if tombstoned) or null if there is no value
-        *   - version: value version number or null if the if there is no value
-        *   - tombAsOf: UNIX timestamp of the tombstone or null if there is no tombstone
-        */
-       private function unwrap( $wrapped, $now ) {
-               $value = false;
-               $info = [ 'asOf' => null, 'curTTL' => null, 'version' => null, 'tombAsOf' => null ];
-
-               if ( is_array( $wrapped ) ) {
-                       // Entry expected to be a cached value; validate it
-                       if (
-                               ( $wrapped[self::$FLD_FORMAT_VERSION] ?? null ) === self::$VERSION &&
-                               $wrapped[self::$FLD_TIME] >= $this->epoch
-                       ) {
-                               if ( $wrapped[self::$FLD_TTL] > 0 ) {
-                                       // Get the approximate time left on the key
-                                       $age = $now - $wrapped[self::$FLD_TIME];
-                                       $curTTL = max( $wrapped[self::$FLD_TTL] - $age, 0.0 );
-                               } else {
-                                       // Key had no TTL, so the time left is unbounded
-                                       $curTTL = INF;
-                               }
-                               $value = $wrapped[self::$FLD_VALUE];
-                               $info['version'] = $wrapped[self::$FLD_VALUE_VERSION] ?? null;
-                               $info['asOf'] = $wrapped[self::$FLD_TIME];
-                               $info['curTTL'] = $curTTL;
-                       }
-               } else {
-                       // Entry expected to be a tombstone; parse it
-                       $purge = $this->parsePurgeValue( $wrapped );
-                       if ( $purge !== false ) {
-                               // Tombstoned keys should always have a negative current $ttl
-                               $info['curTTL'] = min( $purge[self::$PURGE_TIME] - $now, self::$TINY_NEGATIVE );
-                               $info['tombAsOf'] = $purge[self::$PURGE_TIME];
-                       }
-               }
-
-               return [ $value, $info ];
-       }
-
-       /**
-        * @param string[] $keys
-        * @param string $prefix
-        * @return string[] Prefix keys; the order of $keys is preserved
-        */
-       protected static function prefixCacheKeys( array $keys, $prefix ) {
-               $res = [];
-               foreach ( $keys as $key ) {
-                       $res[] = $prefix . $key;
-               }
-
-               return $res;
-       }
-
-       /**
-        * @param string $key String of the format <scope>:<class>[:<class or variable>]...
-        * @return string A collection name to describe this class of key
-        */
-       private function determineKeyClassForStats( $key ) {
-               $parts = explode( ':', $key, 3 );
-
-               return $parts[1] ?? $parts[0]; // sanity
-       }
-
-       /**
-        * @param string|array|bool $value Possible string of the form "PURGED:<timestamp>:<holdoff>"
-        * @return array|bool Array containing a UNIX timestamp (float) and holdoff period (integer),
-        *  or false if value isn't a valid purge value
-        */
-       private function parsePurgeValue( $value ) {
-               if ( !is_string( $value ) ) {
-                       return false;
-               }
-
-               $segments = explode( ':', $value, 3 );
-               if (
-                       !isset( $segments[0] ) ||
-                       !isset( $segments[1] ) ||
-                       "{$segments[0]}:" !== self::$PURGE_VAL_PREFIX
-               ) {
-                       return false;
-               }
-
-               if ( !isset( $segments[2] ) ) {
-                       // Back-compat with old purge values without holdoff
-                       $segments[2] = self::HOLDOFF_TTL;
-               }
-
-               if ( $segments[1] < $this->epoch ) {
-                       // Values this old are ignored
-                       return false;
-               }
-
-               return [
-                       self::$PURGE_TIME => (float)$segments[1],
-                       self::$PURGE_HOLDOFF => (int)$segments[2],
-               ];
-       }
-
-       /**
-        * @param float $timestamp
-        * @param int $holdoff In seconds
-        * @return string Wrapped purge value
-        */
-       private function makePurgeValue( $timestamp, $holdoff ) {
-               return self::$PURGE_VAL_PREFIX . (float)$timestamp . ':' . (int)$holdoff;
-       }
-
-       /**
-        * @param string $group
-        * @return MapCacheLRU
-        */
-       private function getProcessCache( $group ) {
-               if ( !isset( $this->processCaches[$group] ) ) {
-                       list( , $size ) = explode( ':', $group );
-                       $this->processCaches[$group] = new MapCacheLRU( (int)$size );
-               }
-
-               return $this->processCaches[$group];
-       }
-
-       /**
-        * @param string $key
-        * @param int $version
-        * @return string
-        */
-       private function getProcessCacheKey( $key, $version ) {
-               return $key . ' ' . (int)$version;
-       }
-
-       /**
-        * @param ArrayIterator $keys
-        * @param array $opts
-        * @return string[] Map of (ID => cache key)
-        */
-       private function getNonProcessCachedMultiKeys( ArrayIterator $keys, array $opts ) {
-               $pcTTL = $opts['pcTTL'] ?? self::TTL_UNCACHEABLE;
-
-               $keysMissing = [];
-               if ( $pcTTL > 0 && $this->callbackDepth == 0 ) {
-                       $version = $opts['version'] ?? null;
-                       $pCache = $this->getProcessCache( $opts['pcGroup'] ?? self::PC_PRIMARY );
-                       foreach ( $keys as $key => $id ) {
-                               if ( !$pCache->has( $this->getProcessCacheKey( $key, $version ), $pcTTL ) ) {
-                                       $keysMissing[$id] = $key;
-                               }
-                       }
-               }
-
-               return $keysMissing;
-       }
-
-       /**
-        * @param string[] $keys
-        * @param string[]|string[][] $checkKeys
-        * @return string[] List of cache keys
-        */
-       private function getRawKeysForWarmup( array $keys, array $checkKeys ) {
-               if ( !$keys ) {
-                       return [];
-               }
-
-               $keysWarmUp = [];
-               // Get all the value keys to fetch...
-               foreach ( $keys as $key ) {
-                       $keysWarmUp[] = self::$VALUE_KEY_PREFIX . $key;
-               }
-               // Get all the check keys to fetch...
-               foreach ( $checkKeys as $i => $checkKeyOrKeys ) {
-                       if ( is_int( $i ) ) {
-                               // Single check key that applies to all value keys
-                               $keysWarmUp[] = self::$TIME_KEY_PREFIX . $checkKeyOrKeys;
-                       } else {
-                               // List of check keys that apply to value key $i
-                               $keysWarmUp = array_merge(
-                                       $keysWarmUp,
-                                       self::prefixCacheKeys( $checkKeyOrKeys, self::$TIME_KEY_PREFIX )
-                               );
-                       }
-               }
-
-               $warmupCache = $this->cache->getMulti( $keysWarmUp );
-               $warmupCache += array_fill_keys( $keysWarmUp, false );
-
-               return $warmupCache;
-       }
-
-       /**
-        * @return float UNIX timestamp
-        * @codeCoverageIgnore
-        */
-       protected function getCurrentTime() {
-               if ( $this->wallClockOverride ) {
-                       return $this->wallClockOverride;
-               }
-
-               $clockTime = (float)time(); // call this first
-               // microtime() uses an initial gettimeofday() call added to usage clocks.
-               // This can severely drift from time() and the microtime() value of other threads
-               // due to undercounting of the amount of time elapsed. Instead of seeing the current
-               // time as being in the past, use the value of time(). This avoids setting cache values
-               // that will immediately be seen as expired and possibly cause stampedes.
-               return max( microtime( true ), $clockTime );
-       }
-
-       /**
-        * @param float|null &$time Mock UNIX timestamp for testing
-        * @codeCoverageIgnore
-        */
-       public function setMockTime( &$time ) {
-               $this->wallClockOverride =& $time;
-               $this->cache->setMockTime( $time );
-       }
-}
diff --git a/includes/libs/objectcache/WANObjectCacheReaper.php b/includes/libs/objectcache/WANObjectCacheReaper.php
deleted file mode 100644 (file)
index fb8a754..0000000
+++ /dev/null
@@ -1,199 +0,0 @@
-<?php
-/**
- * This program is free software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License as published by
- * the Free Software Foundation; either version 2 of the License, or
- * (at your option) any later version.
- *
- * This program is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU General Public License for more details.
- *
- * You should have received a copy of the GNU General Public License along
- * with this program; if not, write to the Free Software Foundation, Inc.,
- * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
- * http://www.gnu.org/copyleft/gpl.html
- *
- * @file
- * @ingroup Cache
- */
-
-use Psr\Log\LoggerAwareInterface;
-use Psr\Log\LoggerInterface;
-use Psr\Log\NullLogger;
-use Wikimedia\ScopedCallback;
-
-/**
- * Class for scanning through chronological, log-structured data or change logs
- * and locally purging cache keys related to entities that appear in this data.
- *
- * This is useful for repairing cache when purges are missed by using a reliable
- * stream, such as Kafka or a replicated MySQL table. Purge loss between datacenters
- * is expected to be more common than within them.
- *
- * @since 1.28
- */
-class WANObjectCacheReaper implements LoggerAwareInterface {
-       /** @var WANObjectCache */
-       protected $cache;
-       /** @var BagOStuff */
-       protected $store;
-       /** @var callable */
-       protected $logChunkCallback;
-       /** @var callable */
-       protected $keyListCallback;
-       /** @var LoggerInterface */
-       protected $logger;
-
-       /** @var string */
-       protected $channel;
-       /** @var int */
-       protected $initialStartWindow;
-
-       /**
-        * @param WANObjectCache $cache Cache to reap bad keys from
-        * @param BagOStuff $store Cache to store positions use for locking
-        * @param callable $logCallback Callback taking arguments:
-        *          - The starting position as a UNIX timestamp
-        *          - The starting unique ID used for breaking timestamp collisions or null
-        *          - The ending position as a UNIX timestamp
-        *          - The maximum number of results to return
-        *        It returns a list of maps of (key: cache key, pos: UNIX timestamp, id: unique ID)
-        *        for each key affected, with the corrosponding event timestamp/ID information.
-        *        The events should be in ascending order, by (timestamp,id).
-        * @param callable $keyCallback Callback taking arguments:
-        *          - The WANObjectCache instance
-        *          - An object from the event log
-        *        It should return a list of WAN cache keys.
-        *        The callback must fully duck-type test the object, since can be any model class.
-        * @param array $params Additional options:
-        *          - channel: the name of the update event stream.
-        *          - initialStartWindow: seconds back in time to start if the position is lost.
-        *            Default: 1 hour.
-        *          - logger: an SPL monolog instance [optional]
-        */
-       public function __construct(
-               WANObjectCache $cache,
-               BagOStuff $store,
-               callable $logCallback,
-               callable $keyCallback,
-               array $params
-       ) {
-               $this->cache = $cache;
-               $this->store = $store;
-
-               $this->logChunkCallback = $logCallback;
-               $this->keyListCallback = $keyCallback;
-               if ( isset( $params['channel'] ) ) {
-                       $this->channel = $params['channel'];
-               } else {
-                       throw new UnexpectedValueException( "No channel specified." );
-               }
-
-               $this->initialStartWindow = $params['initialStartWindow'] ?? 3600;
-               $this->logger = $params['logger'] ?? new NullLogger();
-       }
-
-       public function setLogger( LoggerInterface $logger ) {
-               $this->logger = $logger;
-       }
-
-       /**
-        * Check and reap stale keys based on a chunk of events
-        *
-        * @param int $n Number of events
-        * @return int Number of keys checked
-        */
-       final public function invoke( $n = 100 ) {
-               $posKey = $this->store->makeGlobalKey( 'WANCache', 'reaper', $this->channel );
-               $scopeLock = $this->store->getScopedLock( "$posKey:busy", 0 );
-               if ( !$scopeLock ) {
-                       return 0;
-               }
-
-               $now = time();
-               $status = $this->store->get( $posKey );
-               if ( !$status ) {
-                       $status = [ 'pos' => $now - $this->initialStartWindow, 'id' => null ];
-               }
-
-               // Get events for entities who's keys tombstones/hold-off should have expired by now
-               $events = call_user_func_array(
-                       $this->logChunkCallback,
-                       [ $status['pos'], $status['id'], $now - WANObjectCache::HOLDOFF_TTL - 1, $n ]
-               );
-
-               $event = null;
-               $keyEvents = [];
-               foreach ( $events as $event ) {
-                       $keys = call_user_func_array(
-                               $this->keyListCallback,
-                               [ $this->cache, $event['item'] ]
-                       );
-                       foreach ( $keys as $key ) {
-                               unset( $keyEvents[$key] ); // use only the latest per key
-                               $keyEvents[$key] = [
-                                       'pos' => $event['pos'],
-                                       'id' => $event['id']
-                               ];
-                       }
-               }
-
-               $purgeCount = 0;
-               $lastOkEvent = null;
-               foreach ( $keyEvents as $key => $keyEvent ) {
-                       if ( !$this->cache->reap( $key, $keyEvent['pos'] ) ) {
-                               break;
-                       }
-                       ++$purgeCount;
-                       $lastOkEvent = $event;
-               }
-
-               if ( $lastOkEvent ) {
-                       $ok = $this->store->merge(
-                               $posKey,
-                               function ( $bag, $key, $curValue ) use ( $lastOkEvent ) {
-                                       if ( !$curValue ) {
-                                               // Use new position
-                                       } else {
-                                               $curCoord = [ $curValue['pos'], $curValue['id'] ];
-                                               $newCoord = [ $lastOkEvent['pos'], $lastOkEvent['id'] ];
-                                               if ( $newCoord < $curCoord ) {
-                                                       // Keep prior position instead of rolling it back
-                                                       return $curValue;
-                                               }
-                                       }
-
-                                       return [
-                                               'pos' => $lastOkEvent['pos'],
-                                               'id' => $lastOkEvent['id'],
-                                               'ctime' => $curValue ? $curValue['ctime'] : date( 'c' )
-                                       ];
-                               },
-                               IExpiringStore::TTL_INDEFINITE
-                       );
-
-                       $pos = $lastOkEvent['pos'];
-                       $id = $lastOkEvent['id'];
-                       if ( $ok ) {
-                               $this->logger->info( "Updated cache reap position ($pos, $id)." );
-                       } else {
-                               $this->logger->error( "Could not update cache reap position ($pos, $id)." );
-                       }
-               }
-
-               ScopedCallback::consume( $scopeLock );
-
-               return $purgeCount;
-       }
-
-       /**
-        * @return array|bool Returns (pos, id) map or false if not set
-        */
-       public function getState() {
-               $posKey = $this->store->makeGlobalKey( 'WANCache', 'reaper', $this->channel );
-
-               return $this->store->get( $posKey );
-       }
-}
diff --git a/includes/libs/objectcache/wancache/WANObjectCache.php b/includes/libs/objectcache/wancache/WANObjectCache.php
new file mode 100644 (file)
index 0000000..1852685
--- /dev/null
@@ -0,0 +1,2633 @@
+<?php
+/**
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License along
+ * with this program; if not, write to the Free Software Foundation, Inc.,
+ * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ * http://www.gnu.org/copyleft/gpl.html
+ *
+ * @file
+ * @ingroup Cache
+ */
+
+use Liuggio\StatsdClient\Factory\StatsdDataFactoryInterface;
+use Psr\Log\LoggerAwareInterface;
+use Psr\Log\LoggerInterface;
+use Psr\Log\NullLogger;
+
+/**
+ * Multi-datacenter aware caching interface
+ *
+ * ### Using WANObjectCache
+ *
+ * All operations go to the local datacenter cache, except for delete(),
+ * touchCheckKey(), and resetCheckKey(), which broadcast to all datacenters.
+ *
+ * This class is intended for caching data from primary stores.
+ * If the get() method does not return a value, then the caller
+ * should query the new value and backfill the cache using set().
+ * The preferred way to do this logic is through getWithSetCallback().
+ * When querying the store on cache miss, the closest DB replica
+ * should be used. Try to avoid heavyweight DB master or quorum reads.
+ *
+ * To ensure consumers of the cache see new values in a timely manner,
+ * you either need to follow either the validation strategy, or the
+ * purge strategy.
+ *
+ * The validation strategy refers to the natural avoidance of stale data
+ * by one of the following means:
+ *
+ *   - A) The cached value is immutable.
+ *        If the consumer has access to an identifier that uniquely describes a value,
+ *        cached value need not change. Instead, the key can change. This also allows
+ *        all servers to access their perceived current version. This is important
+ *        in context of multiple deployed versions of your application and/or cross-dc
+ *        database replication, to ensure deterministic values without oscillation.
+ *   - B) Validity is checked against the source after get().
+ *        This is the inverse of A. The unique identifier is embedded inside the value
+ *        and validated after on retreival. If outdated, the value is recomputed.
+ *   - C) The value is cached with a modest TTL (without validation).
+ *        If value recomputation is reasonably performant, and the value is allowed to
+ *        be stale, one should consider using TTL only – using the value's age as
+ *        method of validation.
+ *
+ * The purge strategy refers to the the approach whereby your application knows that
+ * source data has changed and can react by purging the relevant cache keys.
+ * As purges are expensive, this strategy should be avoided if possible.
+ * The simplest purge method is delete().
+ *
+ * No matter which strategy you choose, callers must not rely on updates or purges
+ * being immediately visible to other servers. It should be treated similarly as
+ * one would a database replica.
+ *
+ * The need for immediate updates should be avoided. If needed, solutions must be
+ * sought outside WANObjectCache.
+ *
+ * ### Deploying WANObjectCache
+ *
+ * There are two supported ways to set up broadcasted operations:
+ *
+ *   - A) Set up mcrouter as the underlying cache backend, using a memcached BagOStuff class
+ *        for the 'cache' parameter. The 'region' and 'cluster' parameters must be provided
+ *        and 'mcrouterAware' must be set to `true`.
+ *        Configure mcrouter as follows:
+ *          - 1) Use Route Prefixing based on region (datacenter) and cache cluster.
+ *               See https://github.com/facebook/mcrouter/wiki/Routing-Prefix and
+ *               https://github.com/facebook/mcrouter/wiki/Multi-cluster-broadcast-setup.
+ *          - 2) To increase the consistency of delete() and touchCheckKey() during cache
+ *               server membership changes, you can use the OperationSelectorRoute to
+ *               configure 'set' and 'delete' operations to go to all servers in the cache
+ *               cluster, instead of just one server determined by hashing.
+ *               See https://github.com/facebook/mcrouter/wiki/List-of-Route-Handles.
+ *   - B) Set up dynomite as a cache middleware between the web servers and either memcached
+ *        or redis and use it as the underlying cache backend, using a memcached BagOStuff
+ *        class for the 'cache' parameter. This will broadcast all key setting operations,
+ *        not just purges, which can be useful for cache warming. Writes are eventually
+ *        consistent via the Dynamo replication model. See https://github.com/Netflix/dynomite.
+ *
+ * Broadcasted operations like delete() and touchCheckKey() are intended to run
+ * immediately in the local datacenter and asynchronously in remote datacenters.
+ *
+ * This means that callers in all datacenters may see older values for however many
+ * milliseconds that the purge took to reach that datacenter. As with any cache, this
+ * should not be relied on for cases where reads are used to determine writes to source
+ * (e.g. non-cache) data stores, except when reading immutable data.
+ *
+ * All values are wrapped in metadata arrays. Keys use a "WANCache:" prefix
+ * to avoid collisions with keys that are not wrapped as metadata arrays. The
+ * prefixes are as follows:
+ *   - a) "WANCache:v" : used for regular value keys
+ *   - b) "WANCache:i" : used for temporarily storing values of tombstoned keys
+ *   - c) "WANCache:t" : used for storing timestamp "check" keys
+ *   - d) "WANCache:m" : used for temporary mutex keys to avoid cache stampedes
+ *
+ * @ingroup Cache
+ * @since 1.26
+ */
+class WANObjectCache implements IExpiringStore, IStoreKeyEncoder, LoggerAwareInterface {
+       /** @var BagOStuff The local datacenter cache */
+       protected $cache;
+       /** @var MapCacheLRU[] Map of group PHP instance caches */
+       protected $processCaches = [];
+       /** @var LoggerInterface */
+       protected $logger;
+       /** @var StatsdDataFactoryInterface */
+       protected $stats;
+       /** @var callable|null Function that takes a WAN cache callback and runs it later */
+       protected $asyncHandler;
+
+       /** @bar bool Whether to use mcrouter key prefixing for routing */
+       protected $mcrouterAware;
+       /** @var string Physical region for mcrouter use */
+       protected $region;
+       /** @var string Cache cluster name for mcrouter use */
+       protected $cluster;
+       /** @var bool Whether to use "interim" caching while keys are tombstoned */
+       protected $useInterimHoldOffCaching = true;
+       /** @var float Unix timestamp of the oldest possible valid values */
+       protected $epoch;
+       /** @var string Stable secret used for hasing long strings into key components */
+       protected $secret;
+
+       /** @var int Callback stack depth for getWithSetCallback() */
+       private $callbackDepth = 0;
+       /** @var mixed[] Temporary warm-up cache */
+       private $warmupCache = [];
+       /** @var int Key fetched */
+       private $warmupKeyMisses = 0;
+
+       /** @var float|null */
+       private $wallClockOverride;
+
+       /** @var int Max expected seconds to pass between delete() and DB commit finishing */
+       const MAX_COMMIT_DELAY = 3;
+       /** @var int Max expected seconds of combined lag from replication and view snapshots */
+       const MAX_READ_LAG = 7;
+       /** @var int Seconds to tombstone keys on delete() and treat as volatile after invalidation */
+       const HOLDOFF_TTL = self::MAX_COMMIT_DELAY + self::MAX_READ_LAG + 1;
+
+       /** @var int Idiom for getWithSetCallback() meaning "do not store the callback result" */
+       const TTL_UNCACHEABLE = -1;
+
+       /** @var int Consider regeneration if the key will expire within this many seconds */
+       const LOW_TTL = 30;
+       /** @var int Max TTL, in seconds, to store keys when a data sourced is lagged */
+       const TTL_LAGGED = 30;
+
+       /** @var int Expected time-till-refresh, in seconds, if the key is accessed once per second */
+       const HOT_TTR = 900;
+       /** @var int Minimum key age, in seconds, for expected time-till-refresh to be considered */
+       const AGE_NEW = 60;
+
+       /** @var int Idiom for getWithSetCallback() meaning "no cache stampede mutex required" */
+       const TSE_NONE = -1;
+
+       /** @var int Idiom for set()/getWithSetCallback() meaning "no post-expiration persistence" */
+       const STALE_TTL_NONE = 0;
+       /** @var int Idiom for set()/getWithSetCallback() meaning "no post-expiration grace period" */
+       const GRACE_TTL_NONE = 0;
+       /** @var int Idiom for delete()/touchCheckKey() meaning "no hold-off period" */
+       const HOLDOFF_TTL_NONE = 0;
+       /** @var int Alias for HOLDOFF_TTL_NONE (b/c) (deprecated since 1.34) */
+       const HOLDOFF_NONE = self::HOLDOFF_TTL_NONE;
+
+       /** @var float Idiom for getWithSetCallback() meaning "no minimum required as-of timestamp" */
+       const MIN_TIMESTAMP_NONE = 0.0;
+
+       /** @var string Default process cache name and max key count */
+       const PC_PRIMARY = 'primary:1000';
+
+       /** @var int Idion for get()/getMulti() to return extra information by reference */
+       const PASS_BY_REF = -1;
+
+       /** @var int Seconds to keep dependency purge keys around */
+       private static $CHECK_KEY_TTL = self::TTL_YEAR;
+       /** @var int Seconds to keep interim value keys for tombstoned keys around */
+       private static $INTERIM_KEY_TTL = 1;
+
+       /** @var int Seconds to keep lock keys around */
+       private static $LOCK_TTL = 10;
+       /** @var int Seconds to no-op key set() calls to avoid large blob I/O stampedes */
+       private static $COOLOFF_TTL = 1;
+       /** @var int Seconds to ramp up the chance of regeneration due to expected time-till-refresh */
+       private static $RAMPUP_TTL = 30;
+
+       /** @var float Tiny negative float to use when CTL comes up >= 0 due to clock skew */
+       private static $TINY_NEGATIVE = -0.000001;
+       /** @var float Tiny positive float to use when using "minTime" to assert an inequality */
+       private static $TINY_POSTIVE = 0.000001;
+
+       /** @var int Milliseconds of key fetch/validate/regenerate delay prone to set() stampedes */
+       private static $SET_DELAY_HIGH_MS = 50;
+       /** @var int Min millisecond set() backoff during hold-off (far less than INTERIM_KEY_TTL) */
+       private static $RECENT_SET_LOW_MS = 50;
+       /** @var int Max millisecond set() backoff during hold-off (far less than INTERIM_KEY_TTL) */
+       private static $RECENT_SET_HIGH_MS = 100;
+
+       /** @var int Consider value generation slow if it takes more than this many seconds */
+       private static $GENERATION_SLOW_SEC = 3;
+
+       /** @var int Key to the tombstone entry timestamp */
+       private static $PURGE_TIME = 0;
+       /** @var int Key to the tombstone entry hold-off TTL */
+       private static $PURGE_HOLDOFF = 1;
+
+       /** @var int Cache format version number */
+       private static $VERSION = 1;
+
+       /** @var int Key to WAN cache version number */
+       private static $FLD_FORMAT_VERSION = 0;
+       /** @var int Key to the cached value */
+       private static $FLD_VALUE = 1;
+       /** @var int Key to the original TTL */
+       private static $FLD_TTL = 2;
+       /** @var int Key to the cache timestamp */
+       private static $FLD_TIME = 3;
+       /** @var int Key to the flags bit field (reserved number) */
+       private static /** @noinspection PhpUnusedPrivateFieldInspection */ $FLD_FLAGS = 4;
+       /** @var int Key to collection cache version number */
+       private static $FLD_VALUE_VERSION = 5;
+       /** @var int Key to how long it took to generate the value */
+       private static $FLD_GENERATION_TIME = 6;
+
+       private static $VALUE_KEY_PREFIX = 'WANCache:v:';
+       private static $INTERIM_KEY_PREFIX = 'WANCache:i:';
+       private static $TIME_KEY_PREFIX = 'WANCache:t:';
+       private static $MUTEX_KEY_PREFIX = 'WANCache:m:';
+       private static $COOLOFF_KEY_PREFIX = 'WANCache:c:';
+
+       private static $PURGE_VAL_PREFIX = 'PURGED:';
+
+       /**
+        * @param array $params
+        *   - cache    : BagOStuff object for a persistent cache
+        *   - logger   : LoggerInterface object
+        *   - stats    : StatsdDataFactoryInterface object
+        *   - asyncHandler : A function that takes a callback and runs it later. If supplied,
+        *       whenever a preemptive refresh would be triggered in getWithSetCallback(), the
+        *       current cache value is still used instead. However, the async-handler function
+        *       receives a WAN cache callback that, when run, will execute the value generation
+        *       callback supplied by the getWithSetCallback() caller. The result will be saved
+        *       as normal. The handler is expected to call the WAN cache callback at an opportune
+        *       time (e.g. HTTP post-send), though generally within a few 100ms. [optional]
+        *   - region: the current physical region. This is required when using mcrouter as the
+        *       backing store proxy. [optional]
+        *   - cluster: name of the cache cluster used by this WAN cache. The name must be the
+        *       same in all datacenters; the ("region","cluster") tuple is what distinguishes
+        *       the counterpart cache clusters among all the datacenter. The contents of
+        *       https://github.com/facebook/mcrouter/wiki/Config-Files give background on this.
+        *       This is required when using mcrouter as the backing store proxy. [optional]
+        *   - mcrouterAware: set as true if mcrouter is the backing store proxy and mcrouter
+        *       is configured to interpret /<region>/<cluster>/ key prefixes as routes. This
+        *       requires that "region" and "cluster" are both set above. [optional]
+        *   - epoch: lowest UNIX timestamp a value/tombstone must have to be valid. [optional]
+        *   - secret: stable secret used for hashing long strings into key components. [optional]
+        */
+       public function __construct( array $params ) {
+               $this->cache = $params['cache'];
+               $this->region = $params['region'] ?? 'main';
+               $this->cluster = $params['cluster'] ?? 'wan-main';
+               $this->mcrouterAware = !empty( $params['mcrouterAware'] );
+               $this->epoch = $params['epoch'] ?? 0;
+               $this->secret = $params['secret'] ?? (string)$this->epoch;
+
+               $this->setLogger( $params['logger'] ?? new NullLogger() );
+               $this->stats = $params['stats'] ?? new NullStatsdDataFactory();
+               $this->asyncHandler = $params['asyncHandler'] ?? null;
+       }
+
+       /**
+        * @param LoggerInterface $logger
+        */
+       public function setLogger( LoggerInterface $logger ) {
+               $this->logger = $logger;
+       }
+
+       /**
+        * Get an instance that wraps EmptyBagOStuff
+        *
+        * @return WANObjectCache
+        */
+       public static function newEmpty() {
+               return new static( [ 'cache' => new EmptyBagOStuff() ] );
+       }
+
+       /**
+        * Fetch the value of a key from cache
+        *
+        * If supplied, $curTTL is set to the remaining TTL (current time left):
+        *   - a) INF; if $key exists, has no TTL, and is not invalidated by $checkKeys
+        *   - b) float (>=0); if $key exists, has a TTL, and is not invalidated by $checkKeys
+        *   - c) float (<0); if $key is tombstoned, stale, or existing but invalidated by $checkKeys
+        *   - d) null; if $key does not exist and is not tombstoned
+        *
+        * If a key is tombstoned, $curTTL will reflect the time since delete().
+        *
+        * The timestamp of $key will be checked against the last-purge timestamp
+        * of each of $checkKeys. Those $checkKeys not in cache will have the last-purge
+        * initialized to the current timestamp. If any of $checkKeys have a timestamp
+        * greater than that of $key, then $curTTL will reflect how long ago $key
+        * became invalid. Callers can use $curTTL to know when the value is stale.
+        * The $checkKeys parameter allow mass invalidations by updating a single key:
+        *   - a) Each "check" key represents "last purged" of some source data
+        *   - b) Callers pass in relevant "check" keys as $checkKeys in get()
+        *   - c) When the source data that "check" keys represent changes,
+        *        the touchCheckKey() method is called on them
+        *
+        * Source data entities might exists in a DB that uses snapshot isolation
+        * (e.g. the default REPEATABLE-READ in innoDB). Even for mutable data, that
+        * isolation can largely be maintained by doing the following:
+        *   - a) Calling delete() on entity change *and* creation, before DB commit
+        *   - b) Keeping transaction duration shorter than the delete() hold-off TTL
+        *   - c) Disabling interim key caching via useInterimHoldOffCaching() before get() calls
+        *
+        * However, pre-snapshot values might still be seen if an update was made
+        * in a remote datacenter but the purge from delete() didn't relay yet.
+        *
+        * Consider using getWithSetCallback() instead of get() and set() cycles.
+        * That method has cache slam avoiding features for hot/expensive keys.
+        *
+        * Pass $info as WANObjectCache::PASS_BY_REF to transform it into a cache key metadata map.
+        * This map includes the following metadata:
+        *   - asOf: UNIX timestamp of the value or null if the key is nonexistant
+        *   - tombAsOf: UNIX timestamp of the tombstone or null if the key is not tombstoned
+        *   - lastCKPurge: UNIX timestamp of the highest check key or null if none provided
+        *   - version: cached value version number or null if the key is nonexistant
+        *
+        * Otherwise, $info will transform into the cached value timestamp.
+        *
+        * @param string $key Cache key made from makeKey() or makeGlobalKey()
+        * @param mixed|null &$curTTL Approximate TTL left on the key if present/tombstoned [returned]
+        * @param string[] $checkKeys The "check" keys used to validate the value
+        * @param mixed|null &$info Key info if WANObjectCache::PASS_BY_REF [returned]
+        * @return mixed Value of cache key or false on failure
+        */
+       final public function get(
+               $key, &$curTTL = null, array $checkKeys = [], &$info = null
+       ) {
+               $curTTLs = self::PASS_BY_REF;
+               $infoByKey = self::PASS_BY_REF;
+               $values = $this->getMulti( [ $key ], $curTTLs, $checkKeys, $infoByKey );
+               $curTTL = $curTTLs[$key] ?? null;
+               if ( $info === self::PASS_BY_REF ) {
+                       $info = [
+                               'asOf' => $infoByKey[$key]['asOf'] ?? null,
+                               'tombAsOf' => $infoByKey[$key]['tombAsOf'] ?? null,
+                               'lastCKPurge' => $infoByKey[$key]['lastCKPurge'] ?? null,
+                               'version' => $infoByKey[$key]['version'] ?? null
+                       ];
+               } else {
+                       $info = $infoByKey[$key]['asOf'] ?? null; // b/c
+               }
+
+               return $values[$key] ?? false;
+       }
+
+       /**
+        * Fetch the value of several keys from cache
+        *
+        * Pass $info as WANObjectCache::PASS_BY_REF to transform it into a map of cache keys
+        * to cache key metadata maps, each having the same style as those of WANObjectCache::get().
+        * All the cache keys listed in $keys will have an entry.
+        *
+        * Othwerwise, $info will transform into a map of (cache key => cached value timestamp).
+        * Only the cache keys listed in $keys that exists or are tombstoned will have an entry.
+        *
+        * $checkKeys holds the "check" keys used to validate values of applicable keys. The integer
+        * indexes hold "check" keys that apply to all of $keys while the string indexes hold "check"
+        * keys that only apply to the cache key with that name.
+        *
+        * @see WANObjectCache::get()
+        *
+        * @param string[] $keys List of cache keys made from makeKey() or makeGlobalKey()
+        * @param mixed|null &$curTTLs Map of (key => TTL left) for existing/tombstoned keys [returned]
+        * @param string[]|string[][] $checkKeys Map of (integer or cache key => "check" key(s))
+        * @param mixed|null &$info Map of (key => info) if WANObjectCache::PASS_BY_REF [returned]
+        * @return mixed[] Map of (key => value) for existing values; order of $keys is preserved
+        */
+       final public function getMulti(
+               array $keys,
+               &$curTTLs = [],
+               array $checkKeys = [],
+               &$info = null
+       ) {
+               $result = [];
+               $curTTLs = [];
+               $infoByKey = [];
+
+               $vPrefixLen = strlen( self::$VALUE_KEY_PREFIX );
+               $valueKeys = self::prefixCacheKeys( $keys, self::$VALUE_KEY_PREFIX );
+
+               $checkKeysForAll = [];
+               $checkKeysByKey = [];
+               $checkKeysFlat = [];
+               foreach ( $checkKeys as $i => $checkKeyGroup ) {
+                       $prefixed = self::prefixCacheKeys( (array)$checkKeyGroup, self::$TIME_KEY_PREFIX );
+                       $checkKeysFlat = array_merge( $checkKeysFlat, $prefixed );
+                       // Are these check keys for a specific cache key, or for all keys being fetched?
+                       if ( is_int( $i ) ) {
+                               $checkKeysForAll = array_merge( $checkKeysForAll, $prefixed );
+                       } else {
+                               $checkKeysByKey[$i] = $prefixed;
+                       }
+               }
+
+               // Fetch all of the raw values
+               $keysGet = array_merge( $valueKeys, $checkKeysFlat );
+               if ( $this->warmupCache ) {
+                       $wrappedValues = array_intersect_key( $this->warmupCache, array_flip( $keysGet ) );
+                       $keysGet = array_diff( $keysGet, array_keys( $wrappedValues ) ); // keys left to fetch
+                       $this->warmupKeyMisses += count( $keysGet );
+               } else {
+                       $wrappedValues = [];
+               }
+               if ( $keysGet ) {
+                       $wrappedValues += $this->cache->getMulti( $keysGet );
+               }
+               // Time used to compare/init "check" keys (derived after getMulti() to be pessimistic)
+               $now = $this->getCurrentTime();
+
+               // Collect timestamps from all "check" keys
+               $purgeValuesForAll = $this->processCheckKeys( $checkKeysForAll, $wrappedValues, $now );
+               $purgeValuesByKey = [];
+               foreach ( $checkKeysByKey as $cacheKey => $checks ) {
+                       $purgeValuesByKey[$cacheKey] =
+                               $this->processCheckKeys( $checks, $wrappedValues, $now );
+               }
+
+               // Get the main cache value for each key and validate them
+               foreach ( $valueKeys as $vKey ) {
+                       $key = substr( $vKey, $vPrefixLen ); // unprefix
+                       list( $value, $keyInfo ) = $this->unwrap( $wrappedValues[$vKey] ?? false, $now );
+                       // Force dependent keys to be seen as stale for a while after purging
+                       // to reduce race conditions involving stale data getting cached
+                       $purgeValues = $purgeValuesForAll;
+                       if ( isset( $purgeValuesByKey[$key] ) ) {
+                               $purgeValues = array_merge( $purgeValues, $purgeValuesByKey[$key] );
+                       }
+
+                       $lastCKPurge = null; // timestamp of the highest check key
+                       foreach ( $purgeValues as $purge ) {
+                               $lastCKPurge = max( $purge[self::$PURGE_TIME], $lastCKPurge );
+                               $safeTimestamp = $purge[self::$PURGE_TIME] + $purge[self::$PURGE_HOLDOFF];
+                               if ( $value !== false && $safeTimestamp >= $keyInfo['asOf'] ) {
+                                       // How long ago this value was invalidated by *this* check key
+                                       $ago = min( $purge[self::$PURGE_TIME] - $now, self::$TINY_NEGATIVE );
+                                       // How long ago this value was invalidated by *any* known check key
+                                       $keyInfo['curTTL'] = min( $keyInfo['curTTL'], $ago );
+                               }
+                       }
+                       $keyInfo[ 'lastCKPurge'] = $lastCKPurge;
+
+                       if ( $value !== false ) {
+                               $result[$key] = $value;
+                       }
+                       if ( $keyInfo['curTTL'] !== null ) {
+                               $curTTLs[$key] = $keyInfo['curTTL'];
+                       }
+
+                       $infoByKey[$key] = ( $info === self::PASS_BY_REF )
+                               ? $keyInfo
+                               : $keyInfo['asOf']; // b/c
+               }
+
+               $info = $infoByKey;
+
+               return $result;
+       }
+
+       /**
+        * @since 1.27
+        * @param string[] $timeKeys List of prefixed time check keys
+        * @param mixed[] $wrappedValues
+        * @param float $now
+        * @return array[] List of purge value arrays
+        */
+       private function processCheckKeys( array $timeKeys, array $wrappedValues, $now ) {
+               $purgeValues = [];
+               foreach ( $timeKeys as $timeKey ) {
+                       $purge = isset( $wrappedValues[$timeKey] )
+                               ? $this->parsePurgeValue( $wrappedValues[$timeKey] )
+                               : false;
+                       if ( $purge === false ) {
+                               // Key is not set or malformed; regenerate
+                               $newVal = $this->makePurgeValue( $now, self::HOLDOFF_TTL );
+                               $this->cache->add( $timeKey, $newVal, self::$CHECK_KEY_TTL );
+                               $purge = $this->parsePurgeValue( $newVal );
+                       }
+                       $purgeValues[] = $purge;
+               }
+               return $purgeValues;
+       }
+
+       /**
+        * Set the value of a key in cache
+        *
+        * Simply calling this method when source data changes is not valid because
+        * the changes do not replicate to the other WAN sites. In that case, delete()
+        * should be used instead. This method is intended for use on cache misses.
+        *
+        * If the data was read from a snapshot-isolated transactions (e.g. the default
+        * REPEATABLE-READ in innoDB), use 'since' to avoid the following race condition:
+        *   - a) T1 starts
+        *   - b) T2 updates a row, calls delete(), and commits
+        *   - c) The HOLDOFF_TTL passes, expiring the delete() tombstone
+        *   - d) T1 reads the row and calls set() due to a cache miss
+        *   - e) Stale value is stuck in cache
+        *
+        * Setting 'lag' and 'since' help avoids keys getting stuck in stale states.
+        *
+        * Be aware that this does not update the process cache for getWithSetCallback()
+        * callers. Keys accessed via that method are not generally meant to also be set
+        * using this primitive method.
+        *
+        * Do not use this method on versioned keys accessed via getWithSetCallback().
+        *
+        * Example usage:
+        * @code
+        *     $dbr = wfGetDB( DB_REPLICA );
+        *     $setOpts = Database::getCacheSetOptions( $dbr );
+        *     // Fetch the row from the DB
+        *     $row = $dbr->selectRow( ... );
+        *     $key = $cache->makeKey( 'building', $buildingId );
+        *     $cache->set( $key, $row, $cache::TTL_DAY, $setOpts );
+        * @endcode
+        *
+        * @param string $key Cache key
+        * @param mixed $value
+        * @param int $ttl Seconds to live. Special values are:
+        *   - WANObjectCache::TTL_INDEFINITE: Cache forever (default)
+        *   - WANObjectCache::TTL_UNCACHEABLE: Do not cache (if the key exists, it is not deleted)
+        * @param array $opts Options map:
+        *   - lag: Seconds of replica DB lag. Typically, this is either the replica DB lag
+        *      before the data was read or, if applicable, the replica DB lag before
+        *      the snapshot-isolated transaction the data was read from started.
+        *      Use false to indicate that replication is not running.
+        *      Default: 0 seconds
+        *   - since: UNIX timestamp of the data in $value. Typically, this is either
+        *      the current time the data was read or (if applicable) the time when
+        *      the snapshot-isolated transaction the data was read from started.
+        *      Default: 0 seconds
+        *   - pending: Whether this data is possibly from an uncommitted write transaction.
+        *      Generally, other threads should not see values from the future and
+        *      they certainly should not see ones that ended up getting rolled back.
+        *      Default: false
+        *   - lockTSE: If excessive replication/snapshot lag is detected, then store the value
+        *      with this TTL and flag it as stale. This is only useful if the reads for this key
+        *      use getWithSetCallback() with "lockTSE" set. Note that if "staleTTL" is set
+        *      then it will still add on to this TTL in the excessive lag scenario.
+        *      Default: WANObjectCache::TSE_NONE
+        *   - staleTTL: Seconds to keep the key around if it is stale. The get()/getMulti()
+        *      methods return such stale values with a $curTTL of 0, and getWithSetCallback()
+        *      will call the regeneration callback in such cases, passing in the old value
+        *      and its as-of time to the callback. This is useful if adaptiveTTL() is used
+        *      on the old value's as-of time when it is verified as still being correct.
+        *      Default: WANObjectCache::STALE_TTL_NONE
+        *   - creating: Optimize for the case where the key does not already exist.
+        *      Default: false
+        *   - version: Integer version number signifiying the format of the value.
+        *      Default: null
+        *   - walltime: How long the value took to generate in seconds. Default: 0.0
+        * @note Options added in 1.28: staleTTL
+        * @note Options added in 1.33: creating
+        * @note Options added in 1.34: version, walltime
+        * @return bool Success
+        */
+       final public function set( $key, $value, $ttl = self::TTL_INDEFINITE, array $opts = [] ) {
+               $now = $this->getCurrentTime();
+               $lag = $opts['lag'] ?? 0;
+               $age = isset( $opts['since'] ) ? max( 0, $now - $opts['since'] ) : 0;
+               $pending = $opts['pending'] ?? false;
+               $lockTSE = $opts['lockTSE'] ?? self::TSE_NONE;
+               $staleTTL = $opts['staleTTL'] ?? self::STALE_TTL_NONE;
+               $creating = $opts['creating'] ?? false;
+               $version = $opts['version'] ?? null;
+               $walltime = $opts['walltime'] ?? 0.0;
+
+               if ( $ttl < 0 ) {
+                       return true;
+               }
+
+               // Do not cache potentially uncommitted data as it might get rolled back
+               if ( $pending ) {
+                       $this->logger->info(
+                               'Rejected set() for {cachekey} due to pending writes.',
+                               [ 'cachekey' => $key ]
+                       );
+
+                       return true; // no-op the write for being unsafe
+               }
+
+               $logicalTTL = null; // logical TTL override
+               // Check if there's a risk of writing stale data after the purge tombstone expired
+               if ( $lag === false || ( $lag + $age ) > self::MAX_READ_LAG ) {
+                       // Case A: any long-running transaction
+                       if ( $age > self::MAX_READ_LAG ) {
+                               if ( $lockTSE >= 0 ) {
+                                       // Store value as *almost* stale to avoid cache and mutex stampedes
+                                       $logicalTTL = self::TTL_SECOND;
+                                       $this->logger->info(
+                                               'Lowered set() TTL for {cachekey} due to snapshot lag.',
+                                               [ 'cachekey' => $key, 'lag' => $lag, 'age' => $age ]
+                                       );
+                               } else {
+                                       $this->logger->info(
+                                               'Rejected set() for {cachekey} due to snapshot lag.',
+                                               [ 'cachekey' => $key, 'lag' => $lag, 'age' => $age ]
+                                       );
+
+                                       return true; // no-op the write for being unsafe
+                               }
+                       // Case B: high replication lag; lower TTL instead of ignoring all set()s
+                       } elseif ( $lag === false || $lag > self::MAX_READ_LAG ) {
+                               if ( $lockTSE >= 0 ) {
+                                       $logicalTTL = min( $ttl ?: INF, self::TTL_LAGGED );
+                               } else {
+                                       $ttl = min( $ttl ?: INF, self::TTL_LAGGED );
+                               }
+                               $this->logger->warning(
+                                       'Lowered set() TTL for {cachekey} due to replication lag.',
+                                       [ 'cachekey' => $key, 'lag' => $lag, 'age' => $age ]
+                               );
+                       // Case C: medium length request with medium replication lag
+                       } elseif ( $lockTSE >= 0 ) {
+                               // Store value as *almost* stale to avoid cache and mutex stampedes
+                               $logicalTTL = self::TTL_SECOND;
+                               $this->logger->info(
+                                       'Lowered set() TTL for {cachekey} due to high read lag.',
+                                       [ 'cachekey' => $key, 'lag' => $lag, 'age' => $age ]
+                               );
+                       } else {
+                               $this->logger->info(
+                                       'Rejected set() for {cachekey} due to high read lag.',
+                                       [ 'cachekey' => $key, 'lag' => $lag, 'age' => $age ]
+                               );
+
+                               return true; // no-op the write for being unsafe
+                       }
+               }
+
+               // Wrap that value with time/TTL/version metadata
+               $wrapped = $this->wrap( $value, $logicalTTL ?: $ttl, $version, $now, $walltime );
+               $storeTTL = $ttl + $staleTTL;
+
+               if ( $creating ) {
+                       $ok = $this->cache->add( self::$VALUE_KEY_PREFIX . $key, $wrapped, $storeTTL );
+               } else {
+                       $ok = $this->cache->merge(
+                               self::$VALUE_KEY_PREFIX . $key,
+                               function ( $cache, $key, $cWrapped ) use ( $wrapped ) {
+                                       // A string value means that it is a tombstone; do nothing in that case
+                                       return ( is_string( $cWrapped ) ) ? false : $wrapped;
+                               },
+                               $storeTTL,
+                               1 // 1 attempt
+                       );
+               }
+
+               return $ok;
+       }
+
+       /**
+        * Purge a key from all datacenters
+        *
+        * This should only be called when the underlying data (being cached)
+        * changes in a significant way. This deletes the key and starts a hold-off
+        * period where the key cannot be written to for a few seconds (HOLDOFF_TTL).
+        * This is done to avoid the following race condition:
+        *   - a) Some DB data changes and delete() is called on a corresponding key
+        *   - b) A request refills the key with a stale value from a lagged DB
+        *   - c) The stale value is stuck there until the key is expired/evicted
+        *
+        * This is implemented by storing a special "tombstone" value at the cache
+        * key that this class recognizes; get() calls will return false for the key
+        * and any set() calls will refuse to replace tombstone values at the key.
+        * For this to always avoid stale value writes, the following must hold:
+        *   - a) Replication lag is bounded to being less than HOLDOFF_TTL; or
+        *   - b) If lag is higher, the DB will have gone into read-only mode already
+        *
+        * Note that set() can also be lag-aware and lower the TTL if it's high.
+        *
+        * Be aware that this does not clear the process cache. Even if it did, callbacks
+        * used by getWithSetCallback() might still return stale data in the case of either
+        * uncommitted or not-yet-replicated changes (callback generally use replica DBs).
+        *
+        * When using potentially long-running ACID transactions, a good pattern is
+        * to use a pre-commit hook to issue the delete. This means that immediately
+        * after commit, callers will see the tombstone in cache upon purge relay.
+        * It also avoids the following race condition:
+        *   - a) T1 begins, changes a row, and calls delete()
+        *   - b) The HOLDOFF_TTL passes, expiring the delete() tombstone
+        *   - c) T2 starts, reads the row and calls set() due to a cache miss
+        *   - d) T1 finally commits
+        *   - e) Stale value is stuck in cache
+        *
+        * Example usage:
+        * @code
+        *     $dbw->startAtomic( __METHOD__ ); // start of request
+        *     ... <execute some stuff> ...
+        *     // Update the row in the DB
+        *     $dbw->update( ... );
+        *     $key = $cache->makeKey( 'homes', $homeId );
+        *     // Purge the corresponding cache entry just before committing
+        *     $dbw->onTransactionPreCommitOrIdle( function() use ( $cache, $key ) {
+        *         $cache->delete( $key );
+        *     } );
+        *     ... <execute some stuff> ...
+        *     $dbw->endAtomic( __METHOD__ ); // end of request
+        * @endcode
+        *
+        * The $ttl parameter can be used when purging values that have not actually changed
+        * recently. For example, a cleanup script to purge cache entries does not really need
+        * a hold-off period, so it can use HOLDOFF_TTL_NONE. Likewise for user-requested purge.
+        * Note that $ttl limits the effective range of 'lockTSE' for getWithSetCallback().
+        *
+        * If called twice on the same key, then the last hold-off TTL takes precedence. For
+        * idempotence, the $ttl should not vary for different delete() calls on the same key.
+        *
+        * @param string $key Cache key
+        * @param int $ttl Tombstone TTL; Default: WANObjectCache::HOLDOFF_TTL
+        * @return bool True if the item was purged or not found, false on failure
+        */
+       final public function delete( $key, $ttl = self::HOLDOFF_TTL ) {
+               if ( $ttl <= 0 ) {
+                       // Publish the purge to all datacenters
+                       $ok = $this->relayDelete( self::$VALUE_KEY_PREFIX . $key );
+               } else {
+                       // Publish the purge to all datacenters
+                       $ok = $this->relayPurge( self::$VALUE_KEY_PREFIX . $key, $ttl, self::HOLDOFF_TTL_NONE );
+               }
+
+               $kClass = $this->determineKeyClassForStats( $key );
+               $this->stats->increment( "wanobjectcache.$kClass.delete." . ( $ok ? 'ok' : 'error' ) );
+
+               return $ok;
+       }
+
+       /**
+        * Fetch the value of a timestamp "check" key
+        *
+        * The key will be *initialized* to the current time if not set,
+        * so only call this method if this behavior is actually desired
+        *
+        * The timestamp can be used to check whether a cached value is valid.
+        * Callers should not assume that this returns the same timestamp in
+        * all datacenters due to relay delays.
+        *
+        * The level of staleness can roughly be estimated from this key, but
+        * if the key was evicted from cache, such calculations may show the
+        * time since expiry as ~0 seconds.
+        *
+        * Note that "check" keys won't collide with other regular keys.
+        *
+        * @param string $key
+        * @return float UNIX timestamp
+        */
+       final public function getCheckKeyTime( $key ) {
+               return $this->getMultiCheckKeyTime( [ $key ] )[$key];
+       }
+
+       /**
+        * Fetch the values of each timestamp "check" key
+        *
+        * This works like getCheckKeyTime() except it takes a list of keys
+        * and returns a map of timestamps instead of just that of one key
+        *
+        * This might be useful if both:
+        *   - a) a class of entities each depend on hundreds of other entities
+        *   - b) these other entities are depended upon by millions of entities
+        *
+        * The later entities can each use a "check" key to invalidate their dependee entities.
+        * However, it is expensive for the former entities to verify against all of the relevant
+        * "check" keys during each getWithSetCallback() call. A less expensive approach is to do
+        * these verifications only after a "time-till-verify" (TTV) has passed. This is a middle
+        * ground between using blind TTLs and using constant verification. The adaptiveTTL() method
+        * can be used to dynamically adjust the TTV. Also, the initial TTV can make use of the
+        * last-modified times of the dependant entities (either from the DB or the "check" keys).
+        *
+        * Example usage:
+        * @code
+        *     $value = $cache->getWithSetCallback(
+        *         $cache->makeGlobalKey( 'wikibase-item', $id ),
+        *         self::INITIAL_TTV, // initial time-till-verify
+        *         function ( $oldValue, &$ttv, &$setOpts, $oldAsOf ) use ( $checkKeys, $cache ) {
+        *             $now = microtime( true );
+        *             // Use $oldValue if it passes max ultimate age and "check" key comparisons
+        *             if ( $oldValue &&
+        *                 $oldAsOf > max( $cache->getMultiCheckKeyTime( $checkKeys ) ) &&
+        *                 ( $now - $oldValue['ctime'] ) <= self::MAX_CACHE_AGE
+        *             ) {
+        *                 // Increase time-till-verify by 50% of last time to reduce overhead
+        *                 $ttv = $cache->adaptiveTTL( $oldAsOf, self::MAX_TTV, self::MIN_TTV, 1.5 );
+        *                 // Unlike $oldAsOf, "ctime" is the ultimate age of the cached data
+        *                 return $oldValue;
+        *             }
+        *
+        *             $mtimes = []; // dependency last-modified times; passed by reference
+        *             $value = [ 'data' => $this->fetchEntityData( $mtimes ), 'ctime' => $now ];
+        *             // Guess time-till-change among the dependencies, e.g. 1/(total change rate)
+        *             $ttc = 1 / array_sum( array_map(
+        *                 function ( $mtime ) use ( $now ) {
+        *                     return 1 / ( $mtime ? ( $now - $mtime ) : 900 );
+        *                 },
+        *                 $mtimes
+        *             ) );
+        *             // The time-to-verify should not be overly pessimistic nor optimistic
+        *             $ttv = min( max( $ttc, self::MIN_TTV ), self::MAX_TTV );
+        *
+        *             return $value;
+        *         },
+        *         [ 'staleTTL' => $cache::TTL_DAY ] // keep around to verify and re-save
+        *     );
+        * @endcode
+        *
+        * @see WANObjectCache::getCheckKeyTime()
+        * @see WANObjectCache::getWithSetCallback()
+        *
+        * @param string[] $keys
+        * @return float[] Map of (key => UNIX timestamp)
+        * @since 1.31
+        */
+       final public function getMultiCheckKeyTime( array $keys ) {
+               $rawKeys = [];
+               foreach ( $keys as $key ) {
+                       $rawKeys[$key] = self::$TIME_KEY_PREFIX . $key;
+               }
+
+               $rawValues = $this->cache->getMulti( $rawKeys );
+               $rawValues += array_fill_keys( $rawKeys, false );
+
+               $times = [];
+               foreach ( $rawKeys as $key => $rawKey ) {
+                       $purge = $this->parsePurgeValue( $rawValues[$rawKey] );
+                       if ( $purge !== false ) {
+                               $time = $purge[self::$PURGE_TIME];
+                       } else {
+                               // Casting assures identical floats for the next getCheckKeyTime() calls
+                               $now = (string)$this->getCurrentTime();
+                               $this->cache->add(
+                                       $rawKey,
+                                       $this->makePurgeValue( $now, self::HOLDOFF_TTL ),
+                                       self::$CHECK_KEY_TTL
+                               );
+                               $time = (float)$now;
+                       }
+
+                       $times[$key] = $time;
+               }
+
+               return $times;
+       }
+
+       /**
+        * Purge a "check" key from all datacenters, invalidating keys that use it
+        *
+        * This should only be called when the underlying data (being cached)
+        * changes in a significant way, and it is impractical to call delete()
+        * on all keys that should be changed. When get() is called on those
+        * keys, the relevant "check" keys must be supplied for this to work.
+        *
+        * The "check" key essentially represents a last-modified time of an entity.
+        * When the key is touched, the timestamp will be updated to the current time.
+        * Keys using the "check" key via get(), getMulti(), or getWithSetCallback() will
+        * be invalidated. This approach is useful if many keys depend on a single entity.
+        *
+        * The timestamp of the "check" key is treated as being HOLDOFF_TTL seconds in the
+        * future by get*() methods in order to avoid race conditions where keys are updated
+        * with stale values (e.g. from a lagged replica DB). A high TTL is set on the "check"
+        * key, making it possible to know the timestamp of the last change to the corresponding
+        * entities in most cases. This might use more cache space than resetCheckKey().
+        *
+        * When a few important keys get a large number of hits, a high cache time is usually
+        * desired as well as "lockTSE" logic. The resetCheckKey() method is less appropriate
+        * in such cases since the "time since expiry" cannot be inferred, causing any get()
+        * after the reset to treat the key as being "hot", resulting in more stale value usage.
+        *
+        * Note that "check" keys won't collide with other regular keys.
+        *
+        * @see WANObjectCache::get()
+        * @see WANObjectCache::getWithSetCallback()
+        * @see WANObjectCache::resetCheckKey()
+        *
+        * @param string $key Cache key
+        * @param int $holdoff HOLDOFF_TTL or HOLDOFF_TTL_NONE constant
+        * @return bool True if the item was purged or not found, false on failure
+        */
+       final public function touchCheckKey( $key, $holdoff = self::HOLDOFF_TTL ) {
+               // Publish the purge to all datacenters
+               $ok = $this->relayPurge( self::$TIME_KEY_PREFIX . $key, self::$CHECK_KEY_TTL, $holdoff );
+
+               $kClass = $this->determineKeyClassForStats( $key );
+               $this->stats->increment( "wanobjectcache.$kClass.ck_touch." . ( $ok ? 'ok' : 'error' ) );
+
+               return $ok;
+       }
+
+       /**
+        * Delete a "check" key from all datacenters, invalidating keys that use it
+        *
+        * This is similar to touchCheckKey() in that keys using it via get(), getMulti(),
+        * or getWithSetCallback() will be invalidated. The differences are:
+        *   - a) The "check" key will be deleted from all caches and lazily
+        *        re-initialized when accessed (rather than set everywhere)
+        *   - b) Thus, dependent keys will be known to be stale, but not
+        *        for how long (they are treated as "just" purged), which
+        *        effects any lockTSE logic in getWithSetCallback()
+        *   - c) Since "check" keys are initialized only on the server the key hashes
+        *        to, any temporary ejection of that server will cause the value to be
+        *        seen as purged as a new server will initialize the "check" key.
+        *
+        * The advantage here is that the "check" keys, which have high TTLs, will only
+        * be created when a get*() method actually uses that key. This is better when
+        * a large number of "check" keys are invalided in a short period of time.
+        *
+        * Note that "check" keys won't collide with other regular keys.
+        *
+        * @see WANObjectCache::get()
+        * @see WANObjectCache::getWithSetCallback()
+        * @see WANObjectCache::touchCheckKey()
+        *
+        * @param string $key Cache key
+        * @return bool True if the item was purged or not found, false on failure
+        */
+       final public function resetCheckKey( $key ) {
+               // Publish the purge to all datacenters
+               $ok = $this->relayDelete( self::$TIME_KEY_PREFIX . $key );
+
+               $kClass = $this->determineKeyClassForStats( $key );
+               $this->stats->increment( "wanobjectcache.$kClass.ck_reset." . ( $ok ? 'ok' : 'error' ) );
+
+               return $ok;
+       }
+
+       /**
+        * Method to fetch/regenerate cache keys
+        *
+        * On cache miss, the key will be set to the callback result via set()
+        * (unless the callback returns false) and that result will be returned.
+        * The arguments supplied to the callback are:
+        *   - $oldValue : current cache value or false if not present
+        *   - &$ttl : a reference to the TTL which can be altered
+        *   - &$setOpts : a reference to options for set() which can be altered
+        *   - $oldAsOf : generation UNIX timestamp of $oldValue or null if not present (since 1.28)
+        *
+        * It is strongly recommended to set the 'lag' and 'since' fields to avoid race conditions
+        * that can cause stale values to get stuck at keys. Usually, callbacks ignore the current
+        * value, but it can be used to maintain "most recent X" values that come from time or
+        * sequence based source data, provided that the "as of" id/time is tracked. Note that
+        * preemptive regeneration and $checkKeys can result in a non-false current value.
+        *
+        * Usage of $checkKeys is similar to get() and getMulti(). However, rather than the caller
+        * having to inspect a "current time left" variable (e.g. $curTTL, $curTTLs), a cache
+        * regeneration will automatically be triggered using the callback.
+        *
+        * The $ttl argument and "hotTTR" option (in $opts) use time-dependant randomization
+        * to avoid stampedes. Keys that are slow to regenerate and either heavily used
+        * or subject to explicit (unpredictable) purges, may need additional mechanisms.
+        * The simplest way to avoid stampedes for such keys is to use 'lockTSE' (in $opts).
+        * If explicit purges are needed, also:
+        *   - a) Pass $key into $checkKeys
+        *   - b) Use touchCheckKey( $key ) instead of delete( $key )
+        *
+        * Example usage (typical key):
+        * @code
+        *     $catInfo = $cache->getWithSetCallback(
+        *         // Key to store the cached value under
+        *         $cache->makeKey( 'cat-attributes', $catId ),
+        *         // Time-to-live (in seconds)
+        *         $cache::TTL_MINUTE,
+        *         // Function that derives the new key value
+        *         function ( $oldValue, &$ttl, array &$setOpts ) {
+        *             $dbr = wfGetDB( DB_REPLICA );
+        *             // Account for any snapshot/replica DB lag
+        *             $setOpts += Database::getCacheSetOptions( $dbr );
+        *
+        *             return $dbr->selectRow( ... );
+        *        }
+        *     );
+        * @endcode
+        *
+        * Example usage (key that is expensive and hot):
+        * @code
+        *     $catConfig = $cache->getWithSetCallback(
+        *         // Key to store the cached value under
+        *         $cache->makeKey( 'site-cat-config' ),
+        *         // Time-to-live (in seconds)
+        *         $cache::TTL_DAY,
+        *         // Function that derives the new key value
+        *         function ( $oldValue, &$ttl, array &$setOpts ) {
+        *             $dbr = wfGetDB( DB_REPLICA );
+        *             // Account for any snapshot/replica DB lag
+        *             $setOpts += Database::getCacheSetOptions( $dbr );
+        *
+        *             return CatConfig::newFromRow( $dbr->selectRow( ... ) );
+        *         },
+        *         [
+        *             // Calling touchCheckKey() on this key invalidates the cache
+        *             'checkKeys' => [ $cache->makeKey( 'site-cat-config' ) ],
+        *             // Try to only let one datacenter thread manage cache updates at a time
+        *             'lockTSE' => 30,
+        *             // Avoid querying cache servers multiple times in a web request
+        *             'pcTTL' => $cache::TTL_PROC_LONG
+        *         ]
+        *     );
+        * @endcode
+        *
+        * Example usage (key with dynamic dependencies):
+        * @code
+        *     $catState = $cache->getWithSetCallback(
+        *         // Key to store the cached value under
+        *         $cache->makeKey( 'cat-state', $cat->getId() ),
+        *         // Time-to-live (seconds)
+        *         $cache::TTL_HOUR,
+        *         // Function that derives the new key value
+        *         function ( $oldValue, &$ttl, array &$setOpts ) {
+        *             // Determine new value from the DB
+        *             $dbr = wfGetDB( DB_REPLICA );
+        *             // Account for any snapshot/replica DB lag
+        *             $setOpts += Database::getCacheSetOptions( $dbr );
+        *
+        *             return CatState::newFromResults( $dbr->select( ... ) );
+        *         },
+        *         [
+        *              // The "check" keys that represent things the value depends on;
+        *              // Calling touchCheckKey() on any of them invalidates the cache
+        *             'checkKeys' => [
+        *                 $cache->makeKey( 'sustenance-bowls', $cat->getRoomId() ),
+        *                 $cache->makeKey( 'people-present', $cat->getHouseId() ),
+        *                 $cache->makeKey( 'cat-laws', $cat->getCityId() ),
+        *             ]
+        *         ]
+        *     );
+        * @endcode
+        *
+        * Example usage (key that is expensive with too many DB dependencies for "check keys"):
+        * @code
+        *     $catToys = $cache->getWithSetCallback(
+        *         // Key to store the cached value under
+        *         $cache->makeKey( 'cat-toys', $catId ),
+        *         // Time-to-live (seconds)
+        *         $cache::TTL_HOUR,
+        *         // Function that derives the new key value
+        *         function ( $oldValue, &$ttl, array &$setOpts ) {
+        *             // Determine new value from the DB
+        *             $dbr = wfGetDB( DB_REPLICA );
+        *             // Account for any snapshot/replica DB lag
+        *             $setOpts += Database::getCacheSetOptions( $dbr );
+        *
+        *             return CatToys::newFromResults( $dbr->select( ... ) );
+        *         },
+        *         [
+        *              // Get the highest timestamp of any of the cat's toys
+        *             'touchedCallback' => function ( $value ) use ( $catId ) {
+        *                 $dbr = wfGetDB( DB_REPLICA );
+        *                 $ts = $dbr->selectField( 'cat_toys', 'MAX(ct_touched)', ... );
+        *
+        *                 return wfTimestampOrNull( TS_UNIX, $ts );
+        *             },
+        *             // Avoid DB queries for repeated access
+        *             'pcTTL' => $cache::TTL_PROC_SHORT
+        *         ]
+        *     );
+        * @endcode
+        *
+        * Example usage (hot key holding most recent 100 events):
+        * @code
+        *     $lastCatActions = $cache->getWithSetCallback(
+        *         // Key to store the cached value under
+        *         $cache->makeKey( 'cat-last-actions', 100 ),
+        *         // Time-to-live (in seconds)
+        *         10,
+        *         // Function that derives the new key value
+        *         function ( $oldValue, &$ttl, array &$setOpts ) {
+        *             $dbr = wfGetDB( DB_REPLICA );
+        *             // Account for any snapshot/replica DB lag
+        *             $setOpts += Database::getCacheSetOptions( $dbr );
+        *
+        *             // Start off with the last cached list
+        *             $list = $oldValue ?: [];
+        *             // Fetch the last 100 relevant rows in descending order;
+        *             // only fetch rows newer than $list[0] to reduce scanning
+        *             $rows = iterator_to_array( $dbr->select( ... ) );
+        *             // Merge them and get the new "last 100" rows
+        *             return array_slice( array_merge( $new, $list ), 0, 100 );
+        *        },
+        *        [
+        *             // Try to only let one datacenter thread manage cache updates at a time
+        *             'lockTSE' => 30,
+        *             // Use a magic value when no cache value is ready rather than stampeding
+        *             'busyValue' => 'computing'
+        *        ]
+        *     );
+        * @endcode
+        *
+        * Example usage (key holding an LRU subkey:value map; this can avoid flooding cache with
+        * keys for an unlimited set of (constraint,situation) pairs, thereby avoiding elevated
+        * cache evictions and wasted memory):
+        * @code
+        *     $catSituationTolerabilityCache = $this->cache->getWithSetCallback(
+        *         // Group by constraint ID/hash, cat family ID/hash, or something else useful
+        *         $this->cache->makeKey( 'cat-situation-tolerability-checks', $groupKey ),
+        *         WANObjectCache::TTL_DAY, // rarely used groups should fade away
+        *         // The $scenarioKey format is $constraintId:<ID/hash of $situation>
+        *         function ( $cacheMap ) use ( $scenarioKey, $constraintId, $situation ) {
+        *             $lruCache = MapCacheLRU::newFromArray( $cacheMap ?: [], self::CACHE_SIZE );
+        *             $result = $lruCache->get( $scenarioKey ); // triggers LRU bump if present
+        *             if ( $result === null || $this->isScenarioResultExpired( $result ) ) {
+        *                 $result = $this->checkScenarioTolerability( $constraintId, $situation );
+        *                 $lruCache->set( $scenarioKey, $result, 3 / 8 );
+        *             }
+        *             // Save the new LRU cache map and reset the map's TTL
+        *             return $lruCache->toArray();
+        *         },
+        *         [
+        *             // Once map is > 1 sec old, consider refreshing
+        *             'ageNew' => 1,
+        *             // Update within 5 seconds after "ageNew" given a 1hz cache check rate
+        *             'hotTTR' => 5,
+        *             // Avoid querying cache servers multiple times in a request; this also means
+        *             // that a request can only alter the value of any given constraint key once
+        *             'pcTTL' => WANObjectCache::TTL_PROC_LONG
+        *         ]
+        *     );
+        *     $tolerability = isset( $catSituationTolerabilityCache[$scenarioKey] )
+        *         ? $catSituationTolerabilityCache[$scenarioKey]
+        *         : $this->checkScenarioTolerability( $constraintId, $situation );
+        * @endcode
+        *
+        * @see WANObjectCache::get()
+        * @see WANObjectCache::set()
+        *
+        * @param string $key Cache key made from makeKey() or makeGlobalKey()
+        * @param int $ttl Seconds to live for key updates. Special values are:
+        *   - WANObjectCache::TTL_INDEFINITE: Cache forever (subject to LRU-style evictions)
+        *   - WANObjectCache::TTL_UNCACHEABLE: Do not cache (if the key exists, it is not deleted)
+        * @param callable $callback Value generation function
+        * @param array $opts Options map:
+        *   - checkKeys: List of "check" keys. The key at $key will be seen as stale when either
+        *      touchCheckKey() or resetCheckKey() is called on any of the keys in this list. This
+        *      is useful if thousands or millions of keys depend on the same entity. The entity can
+        *      simply have its "check" key updated whenever the entity is modified.
+        *      Default: [].
+        *   - graceTTL: If the key is invalidated (by "checkKeys"/"touchedCallback") less than this
+        *      many seconds ago, consider reusing the stale value. The odds of a refresh becomes
+        *      more likely over time, becoming certain once the grace period is reached. This can
+        *      reduce traffic spikes when millions of keys are compared to the same "check" key and
+        *      touchCheckKey() or resetCheckKey() is called on that "check" key. This option is not
+        *      useful for avoiding traffic spikes in the case of the key simply expiring on account
+        *      of its TTL (use "lowTTL" instead).
+        *      Default: WANObjectCache::GRACE_TTL_NONE.
+        *   - lockTSE: If the key is tombstoned or invalidated (by "checkKeys"/"touchedCallback")
+        *      less than this many seconds ago, try to have a single thread handle cache regeneration
+        *      at any given time. Other threads will use stale values if possible. If, on miss,
+        *      the time since expiration is low, the assumption is that the key is hot and that a
+        *      stampede is worth avoiding. Note that if the key falls out of cache then concurrent
+        *      threads will all run the callback on cache miss until the value is saved in cache.
+        *      The only stampede protection in that case is from duplicate cache sets when the
+        *      callback takes longer than WANObjectCache::SET_DELAY_HIGH_MS milliseconds; consider
+        *      using "busyValue" if such stampedes are a problem. Note that the higher "lockTSE" is
+        *      set, the higher the worst-case staleness of returned values can be. Also note that
+        *      this option does not by itself handle the case of the key simply expiring on account
+        *      of its TTL, so make sure that "lowTTL" is not disabled when using this option. Avoid
+        *      combining this option with delete() as it can always cause a stampede due to their
+        *      being no stale value available until after a thread completes the callback.
+        *      Use WANObjectCache::TSE_NONE to disable this logic.
+        *      Default: WANObjectCache::TSE_NONE.
+        *   - busyValue: Specify a placeholder value to use when no value exists and another thread
+        *      is currently regenerating it. This assures that cache stampedes cannot happen if the
+        *      value falls out of cache. This also mitigates stampedes when value regeneration
+        *      becomes very slow (greater than $ttl/"lowTTL"). If this is a closure, then it will
+        *      be invoked to get the placeholder when needed.
+        *      Default: null.
+        *   - pcTTL: Process cache the value in this PHP instance for this many seconds. This avoids
+        *      network I/O when a key is read several times. This will not cache when the callback
+        *      returns false, however. Note that any purges will not be seen while process cached;
+        *      since the callback should use replica DBs and they may be lagged or have snapshot
+        *      isolation anyway, this should not typically matter.
+        *      Default: WANObjectCache::TTL_UNCACHEABLE.
+        *   - pcGroup: Process cache group to use instead of the primary one. If set, this must be
+        *      of the format ALPHANUMERIC_NAME:MAX_KEY_SIZE, e.g. "mydata:10". Use this for storing
+        *      large values, small yet numerous values, or some values with a high cost of eviction.
+        *      It is generally preferable to use a class constant when setting this value.
+        *      This has no effect unless pcTTL is used.
+        *      Default: WANObjectCache::PC_PRIMARY.
+        *   - version: Integer version number. This lets callers make breaking changes to the format
+        *      of cached values without causing problems for sites that use non-instantaneous code
+        *      deployments. Old and new code will recognize incompatible versions and purges from
+        *      both old and new code will been seen by each other. When this method encounters an
+        *      incompatibly versioned value at the provided key, a "variant key" will be used for
+        *      reading from and saving to cache. The variant key is specific to the key and version
+        *      number provided to this method. If the variant key value is older than that of the
+        *      provided key, or the provided key is non-existant, then the variant key will be seen
+        *      as non-existant. Therefore, delete() calls invalidate the provided key's variant keys.
+        *      The "checkKeys" and "touchedCallback" options still apply to variant keys as usual.
+        *      Avoid storing class objects, as this reduces compatibility (due to serialization).
+        *      Default: null.
+        *   - minAsOf: Reject values if they were generated before this UNIX timestamp.
+        *      This is useful if the source of a key is suspected of having possibly changed
+        *      recently, and the caller wants any such changes to be reflected.
+        *      Default: WANObjectCache::MIN_TIMESTAMP_NONE.
+        *   - hotTTR: Expected time-till-refresh (TTR) in seconds for keys that average ~1 hit per
+        *      second (e.g. 1Hz). Keys with a hit rate higher than 1Hz will refresh sooner than this
+        *      TTR and vise versa. Such refreshes won't happen until keys are "ageNew" seconds old.
+        *      This uses randomization to avoid triggering cache stampedes. The TTR is useful at
+        *      reducing the impact of missed cache purges, since the effect of a heavily referenced
+        *      key being stale is worse than that of a rarely referenced key. Unlike simply lowering
+        *      $ttl, seldomly used keys are largely unaffected by this option, which makes it
+        *      possible to have a high hit rate for the "long-tail" of less-used keys.
+        *      Default: WANObjectCache::HOT_TTR.
+        *   - lowTTL: Consider pre-emptive updates when the current TTL (seconds) of the key is less
+        *      than this. It becomes more likely over time, becoming certain once the key is expired.
+        *      This helps avoid cache stampedes that might be triggered due to the key expiring.
+        *      Default: WANObjectCache::LOW_TTL.
+        *   - ageNew: Consider popularity refreshes only once a key reaches this age in seconds.
+        *      Default: WANObjectCache::AGE_NEW.
+        *   - staleTTL: Seconds to keep the key around if it is stale. This means that on cache
+        *      miss the callback may get $oldValue/$oldAsOf values for keys that have already been
+        *      expired for this specified time. This is useful if adaptiveTTL() is used on the old
+        *      value's as-of time when it is verified as still being correct.
+        *      Default: WANObjectCache::STALE_TTL_NONE
+        *   - touchedCallback: A callback that takes the current value and returns a UNIX timestamp
+        *      indicating the last time a dynamic dependency changed. Null can be returned if there
+        *      are no relevant dependency changes to check. This can be used to check against things
+        *      like last-modified times of files or DB timestamp fields. This should generally not be
+        *      used for small and easily queried values in a DB if the callback itself ends up doing
+        *      a similarly expensive DB query to check a timestamp. Usages of this option makes the
+        *      most sense for values that are moderately to highly expensive to regenerate and easy
+        *      to query for dependency timestamps. The use of "pcTTL" reduces timestamp queries.
+        *      Default: null.
+        * @return mixed Value found or written to the key
+        * @note Options added in 1.28: version, busyValue, hotTTR, ageNew, pcGroup, minAsOf
+        * @note Options added in 1.31: staleTTL, graceTTL
+        * @note Options added in 1.33: touchedCallback
+        * @note Callable type hints are not used to avoid class-autoloading
+        */
+       final public function getWithSetCallback( $key, $ttl, $callback, array $opts = [] ) {
+               $version = $opts['version'] ?? null;
+               $pcTTL = $opts['pcTTL'] ?? self::TTL_UNCACHEABLE;
+               $pCache = ( $pcTTL >= 0 )
+                       ? $this->getProcessCache( $opts['pcGroup'] ?? self::PC_PRIMARY )
+                       : null;
+
+               // Use the process cache if requested as long as no outer cache callback is running.
+               // Nested callback process cache use is not lag-safe with regard to HOLDOFF_TTL since
+               // process cached values are more lagged than persistent ones as they are not purged.
+               if ( $pCache && $this->callbackDepth == 0 ) {
+                       $cached = $pCache->get( $this->getProcessCacheKey( $key, $version ), INF, false );
+                       if ( $cached !== false ) {
+                               return $cached;
+                       }
+               }
+
+               $res = $this->fetchOrRegenerate( $key, $ttl, $callback, $opts );
+               list( $value, $valueVersion, $curAsOf ) = $res;
+               if ( $valueVersion !== $version ) {
+                       // Current value has a different version; use the variant key for this version.
+                       // Regenerate the variant value if it is not newer than the main value at $key
+                       // so that purges to the main key propagate to the variant value.
+                       list( $value ) = $this->fetchOrRegenerate(
+                               $this->makeGlobalKey( 'WANCache-key-variant', md5( $key ), $version ),
+                               $ttl,
+                               $callback,
+                               [ 'version' => null, 'minAsOf' => $curAsOf ] + $opts
+                       );
+               }
+
+               // Update the process cache if enabled
+               if ( $pCache && $value !== false ) {
+                       $pCache->set( $this->getProcessCacheKey( $key, $version ), $value );
+               }
+
+               return $value;
+       }
+
+       /**
+        * Do the actual I/O for getWithSetCallback() when needed
+        *
+        * @see WANObjectCache::getWithSetCallback()
+        *
+        * @param string $key
+        * @param int $ttl
+        * @param callable $callback
+        * @param array $opts Options map for getWithSetCallback()
+        * @return array Ordered list of the following:
+        *   - Cached or regenerated value
+        *   - Cached or regenerated value version number or null if not versioned
+        *   - Timestamp of the current cached value at the key or null if there is no value
+        * @note Callable type hints are not used to avoid class-autoloading
+        */
+       private function fetchOrRegenerate( $key, $ttl, $callback, array $opts ) {
+               $checkKeys = $opts['checkKeys'] ?? [];
+               $graceTTL = $opts['graceTTL'] ?? self::GRACE_TTL_NONE;
+               $minAsOf = $opts['minAsOf'] ?? self::MIN_TIMESTAMP_NONE;
+               $hotTTR = $opts['hotTTR'] ?? self::HOT_TTR;
+               $lowTTL = $opts['lowTTL'] ?? min( self::LOW_TTL, $ttl );
+               $ageNew = $opts['ageNew'] ?? self::AGE_NEW;
+               $touchedCb = $opts['touchedCallback'] ?? null;
+               $initialTime = $this->getCurrentTime();
+
+               $kClass = $this->determineKeyClassForStats( $key );
+
+               // Get the current key value and its metadata
+               $curTTL = self::PASS_BY_REF;
+               $curInfo = self::PASS_BY_REF; /** @var array $curInfo */
+               $curValue = $this->get( $key, $curTTL, $checkKeys, $curInfo );
+               // Apply any $touchedCb invalidation timestamp to get the "last purge timestamp"
+               list( $curTTL, $LPT ) = $this->resolveCTL( $curValue, $curTTL, $curInfo, $touchedCb );
+               // Use the cached value if it exists and is not due for synchronous regeneration
+               if (
+                       $this->isValid( $curValue, $curInfo['asOf'], $minAsOf ) &&
+                       $this->isAliveOrInGracePeriod( $curTTL, $graceTTL )
+               ) {
+                       $preemptiveRefresh = (
+                               $this->worthRefreshExpiring( $curTTL, $lowTTL ) ||
+                               $this->worthRefreshPopular( $curInfo['asOf'], $ageNew, $hotTTR, $initialTime )
+                       );
+                       if ( !$preemptiveRefresh ) {
+                               $this->stats->increment( "wanobjectcache.$kClass.hit.good" );
+
+                               return [ $curValue, $curInfo['version'], $curInfo['asOf'] ];
+                       } elseif ( $this->scheduleAsyncRefresh( $key, $ttl, $callback, $opts ) ) {
+                               $this->stats->increment( "wanobjectcache.$kClass.hit.refresh" );
+
+                               return [ $curValue, $curInfo['version'], $curInfo['asOf'] ];
+                       }
+               }
+
+               // Determine if there is stale or volatile cached value that is still usable
+               $isKeyTombstoned = ( $curInfo['tombAsOf'] !== null );
+               if ( $isKeyTombstoned ) {
+                       // Key is write-holed; use the (volatile) interim key as an alternative
+                       list( $possValue, $possInfo ) = $this->getInterimValue( $key, $minAsOf );
+                       // Update the "last purge time" since the $touchedCb timestamp depends on $value
+                       $LPT = $this->resolveTouched( $possValue, $LPT, $touchedCb );
+               } else {
+                       $possValue = $curValue;
+                       $possInfo = $curInfo;
+               }
+
+               // Avoid overhead from callback runs, regeneration locks, and cache sets during
+               // hold-off periods for the key by reusing very recently generated cached values
+               if (
+                       $this->isValid( $possValue, $possInfo['asOf'], $minAsOf, $LPT ) &&
+                       $this->isVolatileValueAgeNegligible( $initialTime - $possInfo['asOf'] )
+               ) {
+                       $this->stats->increment( "wanobjectcache.$kClass.hit.volatile" );
+
+                       return [ $possValue, $possInfo['version'], $curInfo['asOf'] ];
+               }
+
+               $lockTSE = $opts['lockTSE'] ?? self::TSE_NONE;
+               $busyValue = $opts['busyValue'] ?? null;
+               $staleTTL = $opts['staleTTL'] ?? self::STALE_TTL_NONE;
+               $version = $opts['version'] ?? null;
+
+               // Determine whether one thread per datacenter should handle regeneration at a time
+               $useRegenerationLock =
+                       // Note that since tombstones no-op set(), $lockTSE and $curTTL cannot be used to
+                       // deduce the key hotness because |$curTTL| will always keep increasing until the
+                       // tombstone expires or is overwritten by a new tombstone. Also, even if $lockTSE
+                       // is not set, constant regeneration of a key for the tombstone lifetime might be
+                       // very expensive. Assume tombstoned keys are possibly hot in order to reduce
+                       // the risk of high regeneration load after the delete() method is called.
+                       $isKeyTombstoned ||
+                       // Assume a key is hot if requested soon ($lockTSE seconds) after invalidation.
+                       // This avoids stampedes when timestamps from $checkKeys/$touchedCb bump.
+                       ( $curTTL !== null && $curTTL <= 0 && abs( $curTTL ) <= $lockTSE ) ||
+                       // Assume a key is hot if there is no value and a busy fallback is given.
+                       // This avoids stampedes on eviction or preemptive regeneration taking too long.
+                       ( $busyValue !== null && $possValue === false );
+
+               // If a regeneration lock is required, threads that do not get the lock will try to use
+               // the stale value, the interim value, or the $busyValue placeholder, in that order. If
+               // none of those are set then all threads will bypass the lock and regenerate the value.
+               $hasLock = $useRegenerationLock && $this->claimStampedeLock( $key );
+               if ( $useRegenerationLock && !$hasLock ) {
+                       if ( $this->isValid( $possValue, $possInfo['asOf'], $minAsOf ) ) {
+                               $this->stats->increment( "wanobjectcache.$kClass.hit.stale" );
+
+                               return [ $possValue, $possInfo['version'], $curInfo['asOf'] ];
+                       } elseif ( $busyValue !== null ) {
+                               $miss = is_infinite( $minAsOf ) ? 'renew' : 'miss';
+                               $this->stats->increment( "wanobjectcache.$kClass.$miss.busy" );
+
+                               return [ $this->resolveBusyValue( $busyValue ), $version, $curInfo['asOf'] ];
+                       }
+               }
+
+               // Generate the new value given any prior value with a matching version
+               $setOpts = [];
+               $preCallbackTime = $this->getCurrentTime();
+               ++$this->callbackDepth;
+               try {
+                       $value = $callback(
+                               ( $curInfo['version'] === $version ) ? $curValue : false,
+                               $ttl,
+                               $setOpts,
+                               ( $curInfo['version'] === $version ) ? $curInfo['asOf'] : null
+                       );
+               } finally {
+                       --$this->callbackDepth;
+               }
+               $postCallbackTime = $this->getCurrentTime();
+
+               // How long it took to fetch, validate, and generate the value
+               $elapsed = max( $postCallbackTime - $initialTime, 0.0 );
+
+               // Attempt to save the newly generated value if applicable
+               if (
+                       // Callback yielded a cacheable value
+                       ( $value !== false && $ttl >= 0 ) &&
+                       // Current thread was not raced out of a regeneration lock or key is tombstoned
+                       ( !$useRegenerationLock || $hasLock || $isKeyTombstoned ) &&
+                       // Key does not appear to be undergoing a set() stampede
+                       $this->checkAndSetCooloff( $key, $kClass, $elapsed, $lockTSE, $hasLock )
+               ) {
+                       // How long it took to generate the value
+                       $walltime = max( $postCallbackTime - $preCallbackTime, 0.0 );
+                       $this->stats->timing( "wanobjectcache.$kClass.regen_walltime", 1e3 * $walltime );
+                       // If the key is write-holed then use the (volatile) interim key as an alternative
+                       if ( $isKeyTombstoned ) {
+                               $this->setInterimValue( $key, $value, $lockTSE, $version, $walltime );
+                       } else {
+                               $finalSetOpts = [
+                                       'since' => $setOpts['since'] ?? $preCallbackTime,
+                                       'version' => $version,
+                                       'staleTTL' => $staleTTL,
+                                       'lockTSE' => $lockTSE, // informs lag vs performance trade-offs
+                                       'creating' => ( $curValue === false ), // optimization
+                                       'walltime' => $walltime
+                               ] + $setOpts;
+                               $this->set( $key, $value, $ttl, $finalSetOpts );
+                       }
+               }
+
+               $this->yieldStampedeLock( $key, $hasLock );
+
+               $miss = is_infinite( $minAsOf ) ? 'renew' : 'miss';
+               $this->stats->increment( "wanobjectcache.$kClass.$miss.compute" );
+
+               return [ $value, $version, $curInfo['asOf'] ];
+       }
+
+       /**
+        * @param string $key
+        * @return bool Success
+        */
+       private function claimStampedeLock( $key ) {
+               // Note that locking is not bypassed due to I/O errors; this avoids stampedes
+               return $this->cache->add( self::$MUTEX_KEY_PREFIX . $key, 1, self::$LOCK_TTL );
+       }
+
+       /**
+        * @param string $key
+        * @param bool $hasLock
+        */
+       private function yieldStampedeLock( $key, $hasLock ) {
+               if ( $hasLock ) {
+                       // The backend might be a mcrouter proxy set to broadcast DELETE to *all* the local
+                       // datacenter cache servers via OperationSelectorRoute (for increased consistency).
+                       // Since that would be excessive for these locks, use TOUCH to expire the key.
+                       $this->cache->changeTTL( self::$MUTEX_KEY_PREFIX . $key, $this->getCurrentTime() - 60 );
+               }
+       }
+
+       /**
+        * @param float $age Age of volatile/interim key in seconds
+        * @return bool Whether the age of a volatile value is negligible
+        */
+       private function isVolatileValueAgeNegligible( $age ) {
+               return ( $age < mt_rand( self::$RECENT_SET_LOW_MS, self::$RECENT_SET_HIGH_MS ) / 1e3 );
+       }
+
+       /**
+        * @param string $key
+        * @param string $kClass
+        * @param float $elapsed Seconds spent regenerating the value
+        * @param float $lockTSE
+        * @param bool $hasLock
+        * @return bool Whether it is OK to proceed with a key set operation
+        */
+       private function checkAndSetCooloff( $key, $kClass, $elapsed, $lockTSE, $hasLock ) {
+               $this->stats->timing( "wanobjectcache.$kClass.regen_set_delay", 1e3 * $elapsed );
+
+               // If $lockTSE is set, the lock was bypassed because there was no stale/interim value,
+               // and $elapsed indicates that regeration is slow, then there is a risk of set()
+               // stampedes with large blobs. With a typical scale-out infrastructure, CPU and query
+               // load from $callback invocations is distributed among appservers and replica DBs,
+               // but cache operations for a given key route to a single cache server (e.g. striped
+               // consistent hashing).
+               if ( $lockTSE < 0 || $hasLock ) {
+                       return true; // either not a priori hot or thread has the lock
+               } elseif ( $elapsed <= self::$SET_DELAY_HIGH_MS * 1e3 ) {
+                       return true; // not enough time for threads to pile up
+               }
+
+               $this->cache->clearLastError();
+               if (
+                       !$this->cache->add( self::$COOLOFF_KEY_PREFIX . $key, 1, self::$COOLOFF_TTL ) &&
+                       // Don't treat failures due to I/O errors as the key being in cooloff
+                       $this->cache->getLastError() === BagOStuff::ERR_NONE
+               ) {
+                       $this->stats->increment( "wanobjectcache.$kClass.cooloff_bounce" );
+
+                       return false;
+               }
+
+               return true;
+       }
+
+       /**
+        * @param mixed $value
+        * @param float|null $curTTL
+        * @param array $curInfo
+        * @param callable|null $touchedCallback
+        * @return array (current time left or null, UNIX timestamp of last purge or null)
+        * @note Callable type hints are not used to avoid class-autoloading
+        */
+       private function resolveCTL( $value, $curTTL, $curInfo, $touchedCallback ) {
+               if ( $touchedCallback === null || $value === false ) {
+                       return [ $curTTL, max( $curInfo['tombAsOf'], $curInfo['lastCKPurge'] ) ];
+               }
+
+               $touched = $touchedCallback( $value );
+               if ( $touched !== null && $touched >= $curInfo['asOf'] ) {
+                       $curTTL = min( $curTTL, self::$TINY_NEGATIVE, $curInfo['asOf'] - $touched );
+               }
+
+               return [ $curTTL, max( $curInfo['tombAsOf'], $curInfo['lastCKPurge'], $touched ) ];
+       }
+
+       /**
+        * @param mixed $value
+        * @param float|null $lastPurge
+        * @param callable|null $touchedCallback
+        * @return float|null UNIX timestamp of last purge or null
+        * @note Callable type hints are not used to avoid class-autoloading
+        */
+       private function resolveTouched( $value, $lastPurge, $touchedCallback ) {
+               return ( $touchedCallback === null || $value === false )
+                       ? $lastPurge // nothing to derive the "touched timestamp" from
+                       : max( $touchedCallback( $value ), $lastPurge );
+       }
+
+       /**
+        * @param string $key
+        * @param float $minAsOf Minimum acceptable "as of" timestamp
+        * @return array (cached value or false, cache key metadata map)
+        */
+       private function getInterimValue( $key, $minAsOf ) {
+               $now = $this->getCurrentTime();
+
+               if ( $this->useInterimHoldOffCaching ) {
+                       $wrapped = $this->cache->get( self::$INTERIM_KEY_PREFIX . $key );
+
+                       list( $value, $keyInfo ) = $this->unwrap( $wrapped, $now );
+                       if ( $this->isValid( $value, $keyInfo['asOf'], $minAsOf ) ) {
+                               return [ $value, $keyInfo ];
+                       }
+               }
+
+               return $this->unwrap( false, $now );
+       }
+
+       /**
+        * @param string $key
+        * @param mixed $value
+        * @param int $ttl
+        * @param int|null $version Value version number
+        * @param float $walltime How long it took to generate the value in seconds
+        */
+       private function setInterimValue( $key, $value, $ttl, $version, $walltime ) {
+               $ttl = max( self::$INTERIM_KEY_TTL, (int)$ttl );
+
+               $wrapped = $this->wrap( $value, $ttl, $version, $this->getCurrentTime(), $walltime );
+               $this->cache->merge(
+                       self::$INTERIM_KEY_PREFIX . $key,
+                       function () use ( $wrapped ) {
+                               return $wrapped;
+                       },
+                       $ttl,
+                       1
+               );
+       }
+
+       /**
+        * @param mixed $busyValue
+        * @return mixed
+        */
+       private function resolveBusyValue( $busyValue ) {
+               return ( $busyValue instanceof Closure ) ? $busyValue() : $busyValue;
+       }
+
+       /**
+        * Method to fetch multiple cache keys at once with regeneration
+        *
+        * This works the same as getWithSetCallback() except:
+        *   - a) The $keys argument expects the result of WANObjectCache::makeMultiKeys()
+        *   - b) The $callback argument expects a callback taking the following arguments:
+        *         - $id: ID of an entity to query
+        *         - $oldValue : the prior cache value or false if none was present
+        *         - &$ttl : a reference to the new value TTL in seconds
+        *         - &$setOpts : a reference to options for set() which can be altered
+        *         - $oldAsOf : generation UNIX timestamp of $oldValue or null if not present
+        *        Aside from the additional $id argument, the other arguments function the same
+        *        way they do in getWithSetCallback().
+        *   - c) The return value is a map of (cache key => value) in the order of $keyedIds
+        *
+        * @see WANObjectCache::getWithSetCallback()
+        * @see WANObjectCache::getMultiWithUnionSetCallback()
+        *
+        * Example usage:
+        * @code
+        *     $rows = $cache->getMultiWithSetCallback(
+        *         // Map of cache keys to entity IDs
+        *         $cache->makeMultiKeys(
+        *             $this->fileVersionIds(),
+        *             function ( $id ) use ( $cache ) {
+        *                 return $cache->makeKey( 'file-version', $id );
+        *             }
+        *         ),
+        *         // Time-to-live (in seconds)
+        *         $cache::TTL_DAY,
+        *         // Function that derives the new key value
+        *         function ( $id, $oldValue, &$ttl, array &$setOpts ) {
+        *             $dbr = wfGetDB( DB_REPLICA );
+        *             // Account for any snapshot/replica DB lag
+        *             $setOpts += Database::getCacheSetOptions( $dbr );
+        *
+        *             // Load the row for this file
+        *             $queryInfo = File::getQueryInfo();
+        *             $row = $dbr->selectRow(
+        *                 $queryInfo['tables'],
+        *                 $queryInfo['fields'],
+        *                 [ 'id' => $id ],
+        *                 __METHOD__,
+        *                 [],
+        *                 $queryInfo['joins']
+        *             );
+        *
+        *             return $row ? (array)$row : false;
+        *         },
+        *         [
+        *             // Process cache for 30 seconds
+        *             'pcTTL' => 30,
+        *             // Use a dedicated 500 item cache (initialized on-the-fly)
+        *             'pcGroup' => 'file-versions:500'
+        *         ]
+        *     );
+        *     $files = array_map( [ __CLASS__, 'newFromRow' ], $rows );
+        * @endcode
+        *
+        * @param ArrayIterator $keyedIds Result of WANObjectCache::makeMultiKeys()
+        * @param int $ttl Seconds to live for key updates
+        * @param callable $callback Callback the yields entity regeneration callbacks
+        * @param array $opts Options map
+        * @return mixed[] Map of (cache key => value) in the same order as $keyedIds
+        * @since 1.28
+        */
+       final public function getMultiWithSetCallback(
+               ArrayIterator $keyedIds, $ttl, callable $callback, array $opts = []
+       ) {
+               // Load required keys into process cache in one go
+               $this->warmupCache = $this->getRawKeysForWarmup(
+                       $this->getNonProcessCachedMultiKeys( $keyedIds, $opts ),
+                       $opts['checkKeys'] ?? []
+               );
+               $this->warmupKeyMisses = 0;
+
+               // Wrap $callback to match the getWithSetCallback() format while passing $id to $callback
+               $id = null; // current entity ID
+               $func = function ( $oldValue, &$ttl, &$setOpts, $oldAsOf ) use ( $callback, &$id ) {
+                       return $callback( $id, $oldValue, $ttl, $setOpts, $oldAsOf );
+               };
+
+               $values = [];
+               foreach ( $keyedIds as $key => $id ) { // preserve order
+                       $values[$key] = $this->getWithSetCallback( $key, $ttl, $func, $opts );
+               }
+
+               $this->warmupCache = [];
+
+               return $values;
+       }
+
+       /**
+        * Method to fetch/regenerate multiple cache keys at once
+        *
+        * This works the same as getWithSetCallback() except:
+        *   - a) The $keys argument expects the result of WANObjectCache::makeMultiKeys()
+        *   - b) The $callback argument expects a callback returning a map of (ID => new value)
+        *        for all entity IDs in $ids and it takes the following arguments:
+        *          - $ids: a list of entity IDs that require cache regeneration
+        *          - &$ttls: a reference to the (entity ID => new TTL) map
+        *          - &$setOpts: a reference to options for set() which can be altered
+        *   - c) The return value is a map of (cache key => value) in the order of $keyedIds
+        *   - d) The "lockTSE" and "busyValue" options are ignored
+        *
+        * @see WANObjectCache::getWithSetCallback()
+        * @see WANObjectCache::getMultiWithSetCallback()
+        *
+        * Example usage:
+        * @code
+        *     $rows = $cache->getMultiWithUnionSetCallback(
+        *         // Map of cache keys to entity IDs
+        *         $cache->makeMultiKeys(
+        *             $this->fileVersionIds(),
+        *             function ( $id ) use ( $cache ) {
+        *                 return $cache->makeKey( 'file-version', $id );
+        *             }
+        *         ),
+        *         // Time-to-live (in seconds)
+        *         $cache::TTL_DAY,
+        *         // Function that derives the new key value
+        *         function ( array $ids, array &$ttls, array &$setOpts ) {
+        *             $dbr = wfGetDB( DB_REPLICA );
+        *             // Account for any snapshot/replica DB lag
+        *             $setOpts += Database::getCacheSetOptions( $dbr );
+        *
+        *             // Load the rows for these files
+        *             $rows = [];
+        *             $queryInfo = File::getQueryInfo();
+        *             $res = $dbr->select(
+        *                 $queryInfo['tables'],
+        *                 $queryInfo['fields'],
+        *                 [ 'id' => $ids ],
+        *                 __METHOD__,
+        *                 [],
+        *                 $queryInfo['joins']
+        *             );
+        *             foreach ( $res as $row ) {
+        *                 $rows[$row->id] = $row;
+        *                 $mtime = wfTimestamp( TS_UNIX, $row->timestamp );
+        *                 $ttls[$row->id] = $this->adaptiveTTL( $mtime, $ttls[$row->id] );
+        *             }
+        *
+        *             return $rows;
+        *         },
+        *         ]
+        *     );
+        *     $files = array_map( [ __CLASS__, 'newFromRow' ], $rows );
+        * @endcode
+        *
+        * @param ArrayIterator $keyedIds Result of WANObjectCache::makeMultiKeys()
+        * @param int $ttl Seconds to live for key updates
+        * @param callable $callback Callback the yields entity regeneration callbacks
+        * @param array $opts Options map
+        * @return mixed[] Map of (cache key => value) in the same order as $keyedIds
+        * @since 1.30
+        */
+       final public function getMultiWithUnionSetCallback(
+               ArrayIterator $keyedIds, $ttl, callable $callback, array $opts = []
+       ) {
+               $checkKeys = $opts['checkKeys'] ?? [];
+               unset( $opts['lockTSE'] ); // incompatible
+               unset( $opts['busyValue'] ); // incompatible
+
+               // Load required keys into process cache in one go
+               $keysByIdGet = $this->getNonProcessCachedMultiKeys( $keyedIds, $opts );
+               $this->warmupCache = $this->getRawKeysForWarmup( $keysByIdGet, $checkKeys );
+               $this->warmupKeyMisses = 0;
+
+               // IDs of entities known to be in need of regeneration
+               $idsRegen = [];
+
+               // Find out which keys are missing/deleted/stale
+               $curTTLs = [];
+               $asOfs = [];
+               $curByKey = $this->getMulti( $keysByIdGet, $curTTLs, $checkKeys, $asOfs );
+               foreach ( $keysByIdGet as $id => $key ) {
+                       if ( !array_key_exists( $key, $curByKey ) || $curTTLs[$key] < 0 ) {
+                               $idsRegen[] = $id;
+                       }
+               }
+
+               // Run the callback to populate the regeneration value map for all required IDs
+               $newSetOpts = [];
+               $newTTLsById = array_fill_keys( $idsRegen, $ttl );
+               $newValsById = $idsRegen ? $callback( $idsRegen, $newTTLsById, $newSetOpts ) : [];
+
+               // Wrap $callback to match the getWithSetCallback() format while passing $id to $callback
+               $id = null; // current entity ID
+               $func = function ( $oldValue, &$ttl, &$setOpts, $oldAsOf )
+                       use ( $callback, &$id, $newValsById, $newTTLsById, $newSetOpts )
+               {
+                       if ( array_key_exists( $id, $newValsById ) ) {
+                               // Value was already regerated as expected, so use the value in $newValsById
+                               $newValue = $newValsById[$id];
+                               $ttl = $newTTLsById[$id];
+                               $setOpts = $newSetOpts;
+                       } else {
+                               // Pre-emptive/popularity refresh and version mismatch cases are not detected
+                               // above and thus $newValsById has no entry. Run $callback on this single entity.
+                               $ttls = [ $id => $ttl ];
+                               $newValue = $callback( [ $id ], $ttls, $setOpts )[$id];
+                               $ttl = $ttls[$id];
+                       }
+
+                       return $newValue;
+               };
+
+               // Run the cache-aside logic using warmupCache instead of persistent cache queries
+               $values = [];
+               foreach ( $keyedIds as $key => $id ) { // preserve order
+                       $values[$key] = $this->getWithSetCallback( $key, $ttl, $func, $opts );
+               }
+
+               $this->warmupCache = [];
+
+               return $values;
+       }
+
+       /**
+        * Set a key to soon expire in the local cluster if it pre-dates $purgeTimestamp
+        *
+        * This sets stale keys' time-to-live at HOLDOFF_TTL seconds, which both avoids
+        * broadcasting in mcrouter setups and also avoids races with new tombstones.
+        *
+        * @param string $key Cache key
+        * @param int $purgeTimestamp UNIX timestamp of purge
+        * @param bool &$isStale Whether the key is stale
+        * @return bool Success
+        * @since 1.28
+        */
+       final public function reap( $key, $purgeTimestamp, &$isStale = false ) {
+               $minAsOf = $purgeTimestamp + self::HOLDOFF_TTL;
+               $wrapped = $this->cache->get( self::$VALUE_KEY_PREFIX . $key );
+               if ( is_array( $wrapped ) && $wrapped[self::$FLD_TIME] < $minAsOf ) {
+                       $isStale = true;
+                       $this->logger->warning( "Reaping stale value key '$key'." );
+                       $ttlReap = self::HOLDOFF_TTL; // avoids races with tombstone creation
+                       $ok = $this->cache->changeTTL( self::$VALUE_KEY_PREFIX . $key, $ttlReap );
+                       if ( !$ok ) {
+                               $this->logger->error( "Could not complete reap of key '$key'." );
+                       }
+
+                       return $ok;
+               }
+
+               $isStale = false;
+
+               return true;
+       }
+
+       /**
+        * Set a "check" key to soon expire in the local cluster if it pre-dates $purgeTimestamp
+        *
+        * @param string $key Cache key
+        * @param int $purgeTimestamp UNIX timestamp of purge
+        * @param bool &$isStale Whether the key is stale
+        * @return bool Success
+        * @since 1.28
+        */
+       final public function reapCheckKey( $key, $purgeTimestamp, &$isStale = false ) {
+               $purge = $this->parsePurgeValue( $this->cache->get( self::$TIME_KEY_PREFIX . $key ) );
+               if ( $purge && $purge[self::$PURGE_TIME] < $purgeTimestamp ) {
+                       $isStale = true;
+                       $this->logger->warning( "Reaping stale check key '$key'." );
+                       $ok = $this->cache->changeTTL( self::$TIME_KEY_PREFIX . $key, self::TTL_SECOND );
+                       if ( !$ok ) {
+                               $this->logger->error( "Could not complete reap of check key '$key'." );
+                       }
+
+                       return $ok;
+               }
+
+               $isStale = false;
+
+               return false;
+       }
+
+       /**
+        * @see BagOStuff::makeKey()
+        * @param string $class Key class
+        * @param string ...$components Key components (starting with a key collection name)
+        * @return string Colon-delimited list of $keyspace followed by escaped components
+        * @since 1.27
+        */
+       public function makeKey( $class, ...$components ) {
+               return $this->cache->makeKey( ...func_get_args() );
+       }
+
+       /**
+        * @see BagOStuff::makeGlobalKey()
+        * @param string $class Key class
+        * @param string ...$components Key components (starting with a key collection name)
+        * @return string Colon-delimited list of $keyspace followed by escaped components
+        * @since 1.27
+        */
+       public function makeGlobalKey( $class, ...$components ) {
+               return $this->cache->makeGlobalKey( ...func_get_args() );
+       }
+
+       /**
+        * Hash a possibly long string into a suitable component for makeKey()/makeGlobalKey()
+        *
+        * @param string $component A raw component used in building a cache key
+        * @return string 64 character HMAC using a stable secret for public collision resistance
+        * @since 1.34
+        */
+       public function hash256( $component ) {
+               return hash_hmac( 'sha256', $component, $this->secret );
+       }
+
+       /**
+        * Get an iterator of (cache key => entity ID) for a list of entity IDs
+        *
+        * The callback takes an ID string and returns a key via makeKey()/makeGlobalKey().
+        * There should be no network nor filesystem I/O used in the callback. The entity
+        * ID/key mapping must be 1:1 or an exception will be thrown. If hashing is needed,
+        * then use the hash256() method.
+        *
+        * Example usage for the default keyspace:
+        * @code
+        *     $keyedIds = $cache->makeMultiKeys(
+        *         $modules,
+        *         function ( $module ) use ( $cache ) {
+        *             return $cache->makeKey( 'module-info', $module );
+        *         }
+        *     );
+        * @endcode
+        *
+        * Example usage for mixed default and global keyspace:
+        * @code
+        *     $keyedIds = $cache->makeMultiKeys(
+        *         $filters,
+        *         function ( $filter ) use ( $cache ) {
+        *             return ( strpos( $filter, 'central:' ) === 0 )
+        *                 ? $cache->makeGlobalKey( 'regex-filter', $filter )
+        *                 : $cache->makeKey( 'regex-filter', $filter )
+        *         }
+        *     );
+        * @endcode
+        *
+        * Example usage with hashing:
+        * @code
+        *     $keyedIds = $cache->makeMultiKeys(
+        *         $urls,
+        *         function ( $url ) use ( $cache ) {
+        *             return $cache->makeKey( 'url-info', $cache->hash256( $url ) );
+        *         }
+        *     );
+        * @endcode
+        *
+        * @see WANObjectCache::makeKey()
+        * @see WANObjectCache::makeGlobalKey()
+        * @see WANObjectCache::hash256()
+        *
+        * @param string[]|int[] $ids List of entity IDs
+        * @param callable $keyCallback Function returning makeKey()/makeGlobalKey() on the input ID
+        * @return ArrayIterator Iterator of (cache key => ID); order of $ids is preserved
+        * @throws UnexpectedValueException
+        * @since 1.28
+        */
+       final public function makeMultiKeys( array $ids, $keyCallback ) {
+               $idByKey = [];
+               foreach ( $ids as $id ) {
+                       // Discourage triggering of automatic makeKey() hashing in some backends
+                       if ( strlen( $id ) > 64 ) {
+                               $this->logger->warning( __METHOD__ . ": long ID '$id'; use hash256()" );
+                       }
+                       $key = $keyCallback( $id, $this );
+                       // Edge case: ignore key collisions due to duplicate $ids like "42" and 42
+                       if ( !isset( $idByKey[$key] ) ) {
+                               $idByKey[$key] = $id;
+                       } elseif ( (string)$id !== (string)$idByKey[$key] ) {
+                               throw new UnexpectedValueException(
+                                       "Cache key collision; IDs ('$id','{$idByKey[$key]}') map to '$key'"
+                               );
+                       }
+               }
+
+               return new ArrayIterator( $idByKey );
+       }
+
+       /**
+        * Get an (ID => value) map from (i) a non-unique list of entity IDs, and (ii) the list
+        * of corresponding entity values by first appearance of each ID in the entity ID list
+        *
+        * For use with getMultiWithSetCallback() and getMultiWithUnionSetCallback().
+        *
+        * *Only* use this method if the entity ID/key mapping is trivially 1:1 without exception.
+        * Key generation method must utitilize the *full* entity ID in the key (not a hash of it).
+        *
+        * Example usage:
+        * @code
+        *     $poems = $cache->getMultiWithSetCallback(
+        *         $cache->makeMultiKeys(
+        *             $uuids,
+        *             function ( $uuid ) use ( $cache ) {
+        *                 return $cache->makeKey( 'poem', $uuid );
+        *             }
+        *         ),
+        *         $cache::TTL_DAY,
+        *         function ( $uuid ) use ( $url ) {
+        *             return $this->http->run( [ 'method' => 'GET', 'url' => "$url/$uuid" ] );
+        *         }
+        *     );
+        *     $poemsByUUID = $cache->multiRemap( $uuids, $poems );
+        * @endcode
+        *
+        * @see WANObjectCache::makeMultiKeys()
+        * @see WANObjectCache::getMultiWithSetCallback()
+        * @see WANObjectCache::getMultiWithUnionSetCallback()
+        *
+        * @param string[]|int[] $ids Entity ID list makeMultiKeys()
+        * @param mixed[] $res Result of getMultiWithSetCallback()/getMultiWithUnionSetCallback()
+        * @return mixed[] Map of (ID => value); order of $ids is preserved
+        * @since 1.34
+        */
+       final public function multiRemap( array $ids, array $res ) {
+               if ( count( $ids ) !== count( $res ) ) {
+                       // If makeMultiKeys() is called on a list of non-unique IDs, then the resulting
+                       // ArrayIterator will have less entries due to "first appearance" de-duplication
+                       $ids = array_keys( array_flip( $ids ) );
+                       if ( count( $ids ) !== count( $res ) ) {
+                               throw new UnexpectedValueException( "Multi-key result does not match ID list" );
+                       }
+               }
+
+               return array_combine( $ids, $res );
+       }
+
+       /**
+        * Get the "last error" registered; clearLastError() should be called manually
+        * @return int ERR_* class constant for the "last error" registry
+        */
+       final public function getLastError() {
+               $code = $this->cache->getLastError();
+               switch ( $code ) {
+                       case BagOStuff::ERR_NONE:
+                               return self::ERR_NONE;
+                       case BagOStuff::ERR_NO_RESPONSE:
+                               return self::ERR_NO_RESPONSE;
+                       case BagOStuff::ERR_UNREACHABLE:
+                               return self::ERR_UNREACHABLE;
+                       default:
+                               return self::ERR_UNEXPECTED;
+               }
+       }
+
+       /**
+        * Clear the "last error" registry
+        */
+       final public function clearLastError() {
+               $this->cache->clearLastError();
+       }
+
+       /**
+        * Clear the in-process caches; useful for testing
+        *
+        * @since 1.27
+        */
+       public function clearProcessCache() {
+               $this->processCaches = [];
+       }
+
+       /**
+        * Enable or disable the use of brief caching for tombstoned keys
+        *
+        * When a key is purged via delete(), there normally is a period where caching
+        * is hold-off limited to an extremely short time. This method will disable that
+        * caching, forcing the callback to run for any of:
+        *   - WANObjectCache::getWithSetCallback()
+        *   - WANObjectCache::getMultiWithSetCallback()
+        *   - WANObjectCache::getMultiWithUnionSetCallback()
+        *
+        * This is useful when both:
+        *   - a) the database used by the callback is known to be up-to-date enough
+        *        for some particular purpose (e.g. replica DB has applied transaction X)
+        *   - b) the caller needs to exploit that fact, and therefore needs to avoid the
+        *        use of inherently volatile and possibly stale interim keys
+        *
+        * @see WANObjectCache::delete()
+        * @param bool $enabled Whether to enable interim caching
+        * @since 1.31
+        */
+       final public function useInterimHoldOffCaching( $enabled ) {
+               $this->useInterimHoldOffCaching = $enabled;
+       }
+
+       /**
+        * @param int $flag ATTR_* class constant
+        * @return int QOS_* class constant
+        * @since 1.28
+        */
+       public function getQoS( $flag ) {
+               return $this->cache->getQoS( $flag );
+       }
+
+       /**
+        * Get a TTL that is higher for objects that have not changed recently
+        *
+        * This is useful for keys that get explicit purges and DB or purge relay
+        * lag is a potential concern (especially how it interacts with CDN cache)
+        *
+        * Example usage:
+        * @code
+        *     // Last-modified time of page
+        *     $mtime = wfTimestamp( TS_UNIX, $page->getTimestamp() );
+        *     // Get adjusted TTL. If $mtime is 3600 seconds ago and $minTTL/$factor left at
+        *     // defaults, then $ttl is 3600 * .2 = 720. If $minTTL was greater than 720, then
+        *     // $ttl would be $minTTL. If $maxTTL was smaller than 720, $ttl would be $maxTTL.
+        *     $ttl = $cache->adaptiveTTL( $mtime, $cache::TTL_DAY );
+        * @endcode
+        *
+        * Another use case is when there are no applicable "last modified" fields in the DB,
+        * and there are too many dependencies for explicit purges to be viable, and the rate of
+        * change to relevant content is unstable, and it is highly valued to have the cached value
+        * be as up-to-date as possible.
+        *
+        * Example usage:
+        * @code
+        *     $query = "<some complex query>";
+        *     $idListFromComplexQuery = $cache->getWithSetCallback(
+        *         $cache->makeKey( 'complex-graph-query', $hashOfQuery ),
+        *         GraphQueryClass::STARTING_TTL,
+        *         function ( $oldValue, &$ttl, array &$setOpts, $oldAsOf ) use ( $query, $cache ) {
+        *             $gdb = $this->getReplicaGraphDbConnection();
+        *             // Account for any snapshot/replica DB lag
+        *             $setOpts += GraphDatabase::getCacheSetOptions( $gdb );
+        *
+        *             $newList = iterator_to_array( $gdb->query( $query ) );
+        *             sort( $newList, SORT_NUMERIC ); // normalize
+        *
+        *             $minTTL = GraphQueryClass::MIN_TTL;
+        *             $maxTTL = GraphQueryClass::MAX_TTL;
+        *             if ( $oldValue !== false ) {
+        *                 // Note that $oldAsOf is the last time this callback ran
+        *                 $ttl = ( $newList === $oldValue )
+        *                     // No change: cache for 150% of the age of $oldValue
+        *                     ? $cache->adaptiveTTL( $oldAsOf, $maxTTL, $minTTL, 1.5 )
+        *                     // Changed: cache for 50% of the age of $oldValue
+        *                     : $cache->adaptiveTTL( $oldAsOf, $maxTTL, $minTTL, .5 );
+        *             }
+        *
+        *             return $newList;
+        *        },
+        *        [
+        *             // Keep stale values around for doing comparisons for TTL calculations.
+        *             // High values improve long-tail keys hit-rates, though might waste space.
+        *             'staleTTL' => GraphQueryClass::GRACE_TTL
+        *        ]
+        *     );
+        * @endcode
+        *
+        * @param int|float $mtime UNIX timestamp
+        * @param int $maxTTL Maximum TTL (seconds)
+        * @param int $minTTL Minimum TTL (seconds); Default: 30
+        * @param float $factor Value in the range (0,1); Default: .2
+        * @return int Adaptive TTL
+        * @since 1.28
+        */
+       public function adaptiveTTL( $mtime, $maxTTL, $minTTL = 30, $factor = 0.2 ) {
+               if ( is_float( $mtime ) || ctype_digit( $mtime ) ) {
+                       $mtime = (int)$mtime; // handle fractional seconds and string integers
+               }
+
+               if ( !is_int( $mtime ) || $mtime <= 0 ) {
+                       return $minTTL; // no last-modified time provided
+               }
+
+               $age = $this->getCurrentTime() - $mtime;
+
+               return (int)min( $maxTTL, max( $minTTL, $factor * $age ) );
+       }
+
+       /**
+        * @return int Number of warmup key cache misses last round
+        * @since 1.30
+        */
+       final public function getWarmupKeyMisses() {
+               return $this->warmupKeyMisses;
+       }
+
+       /**
+        * Do the actual async bus purge of a key
+        *
+        * This must set the key to "PURGED:<UNIX timestamp>:<holdoff>"
+        *
+        * @param string $key Cache key
+        * @param int $ttl Seconds to keep the tombstone around
+        * @param int $holdoff HOLDOFF_* constant controlling how long to ignore sets for this key
+        * @return bool Success
+        */
+       protected function relayPurge( $key, $ttl, $holdoff ) {
+               if ( $this->mcrouterAware ) {
+                       // See https://github.com/facebook/mcrouter/wiki/Multi-cluster-broadcast-setup
+                       // Wildcards select all matching routes, e.g. the WAN cluster on all DCs
+                       $ok = $this->cache->set(
+                               "/*/{$this->cluster}/{$key}",
+                               $this->makePurgeValue( $this->getCurrentTime(), self::HOLDOFF_TTL_NONE ),
+                               $ttl
+                       );
+               } else {
+                       // This handles the mcrouter and the single-DC case
+                       $ok = $this->cache->set(
+                               $key,
+                               $this->makePurgeValue( $this->getCurrentTime(), self::HOLDOFF_TTL_NONE ),
+                               $ttl
+                       );
+               }
+
+               return $ok;
+       }
+
+       /**
+        * Do the actual async bus delete of a key
+        *
+        * @param string $key Cache key
+        * @return bool Success
+        */
+       protected function relayDelete( $key ) {
+               if ( $this->mcrouterAware ) {
+                       // See https://github.com/facebook/mcrouter/wiki/Multi-cluster-broadcast-setup
+                       // Wildcards select all matching routes, e.g. the WAN cluster on all DCs
+                       $ok = $this->cache->delete( "/*/{$this->cluster}/{$key}" );
+               } else {
+                       // Some other proxy handles broadcasting or there is only one datacenter
+                       $ok = $this->cache->delete( $key );
+               }
+
+               return $ok;
+       }
+
+       /**
+        * @param string $key
+        * @param int $ttl Seconds to live
+        * @param callable $callback
+        * @param array $opts
+        * @return bool Success
+        * @note Callable type hints are not used to avoid class-autoloading
+        */
+       private function scheduleAsyncRefresh( $key, $ttl, $callback, $opts ) {
+               if ( !$this->asyncHandler ) {
+                       return false;
+               }
+               // Update the cache value later, such during post-send of an HTTP request
+               $func = $this->asyncHandler;
+               $func( function () use ( $key, $ttl, $callback, $opts ) {
+                       $opts['minAsOf'] = INF; // force a refresh
+                       $this->fetchOrRegenerate( $key, $ttl, $callback, $opts );
+               } );
+
+               return true;
+       }
+
+       /**
+        * Check if a key is fresh or in the grace window and thus due for randomized reuse
+        *
+        * If $curTTL > 0 (e.g. not expired) this returns true. Otherwise, the chance of returning
+        * true decrease steadily from 100% to 0% as the |$curTTL| moves from 0 to $graceTTL seconds.
+        * This handles widely varying levels of cache access traffic.
+        *
+        * If $curTTL <= -$graceTTL (e.g. already expired), then this returns false.
+        *
+        * @param float $curTTL Approximate TTL left on the key if present
+        * @param int $graceTTL Consider using stale values if $curTTL is greater than this
+        * @return bool
+        */
+       private function isAliveOrInGracePeriod( $curTTL, $graceTTL ) {
+               if ( $curTTL > 0 ) {
+                       return true;
+               } elseif ( $graceTTL <= 0 ) {
+                       return false;
+               }
+
+               $ageStale = abs( $curTTL ); // seconds of staleness
+               $curGTTL = ( $graceTTL - $ageStale ); // current grace-time-to-live
+               if ( $curGTTL <= 0 ) {
+                       return false; //  already out of grace period
+               }
+
+               // Chance of using a stale value is the complement of the chance of refreshing it
+               return !$this->worthRefreshExpiring( $curGTTL, $graceTTL );
+       }
+
+       /**
+        * Check if a key is nearing expiration and thus due for randomized regeneration
+        *
+        * This returns false if $curTTL >= $lowTTL. Otherwise, the chance of returning true
+        * increases steadily from 0% to 100% as the $curTTL moves from $lowTTL to 0 seconds.
+        * This handles widely varying levels of cache access traffic.
+        *
+        * If $curTTL <= 0 (e.g. already expired), then this returns false.
+        *
+        * @param float $curTTL Approximate TTL left on the key if present
+        * @param float $lowTTL Consider a refresh when $curTTL is less than this
+        * @return bool
+        */
+       protected function worthRefreshExpiring( $curTTL, $lowTTL ) {
+               if ( $lowTTL <= 0 ) {
+                       return false;
+               } elseif ( $curTTL >= $lowTTL ) {
+                       return false;
+               } elseif ( $curTTL <= 0 ) {
+                       return false;
+               }
+
+               $chance = ( 1 - $curTTL / $lowTTL );
+
+               return mt_rand( 1, 1e9 ) <= 1e9 * $chance;
+       }
+
+       /**
+        * Check if a key is due for randomized regeneration due to its popularity
+        *
+        * This is used so that popular keys can preemptively refresh themselves for higher
+        * consistency (especially in the case of purge loss/delay). Unpopular keys can remain
+        * in cache with their high nominal TTL. This means popular keys keep good consistency,
+        * whether the data changes frequently or not, and long-tail keys get to stay in cache
+        * and get hits too. Similar to worthRefreshExpiring(), randomization is used.
+        *
+        * @param float $asOf UNIX timestamp of the value
+        * @param int $ageNew Age of key when this might recommend refreshing (seconds)
+        * @param int $timeTillRefresh Age of key when it should be refreshed if popular (seconds)
+        * @param float $now The current UNIX timestamp
+        * @return bool
+        */
+       protected function worthRefreshPopular( $asOf, $ageNew, $timeTillRefresh, $now ) {
+               if ( $ageNew < 0 || $timeTillRefresh <= 0 ) {
+                       return false;
+               }
+
+               $age = $now - $asOf;
+               $timeOld = $age - $ageNew;
+               if ( $timeOld <= 0 ) {
+                       return false;
+               }
+
+               $popularHitsPerSec = 1;
+               // Lifecycle is: new, ramp-up refresh chance, full refresh chance.
+               // Note that the "expected # of refreshes" for the ramp-up time range is half
+               // of what it would be if P(refresh) was at its full value during that time range.
+               $refreshWindowSec = max( $timeTillRefresh - $ageNew - self::$RAMPUP_TTL / 2, 1 );
+               // P(refresh) * (# hits in $refreshWindowSec) = (expected # of refreshes)
+               // P(refresh) * ($refreshWindowSec * $popularHitsPerSec) = 1 (by definition)
+               // P(refresh) = 1/($refreshWindowSec * $popularHitsPerSec)
+               $chance = 1 / ( $popularHitsPerSec * $refreshWindowSec );
+
+               // Ramp up $chance from 0 to its nominal value over RAMPUP_TTL seconds to avoid stampedes
+               $chance *= ( $timeOld <= self::$RAMPUP_TTL ) ? $timeOld / self::$RAMPUP_TTL : 1;
+
+               return mt_rand( 1, 1e9 ) <= 1e9 * $chance;
+       }
+
+       /**
+        * Check if $value is not false, versioned (if needed), and not older than $minTime (if set)
+        *
+        * @param array|bool $value
+        * @param float $asOf The time $value was generated
+        * @param float $minAsOf Minimum acceptable "as of" timestamp
+        * @param float|null $purgeTime The last time the value was invalidated
+        * @return bool
+        */
+       protected function isValid( $value, $asOf, $minAsOf, $purgeTime = null ) {
+               // Avoid reading any key not generated after the latest delete() or touch
+               $safeMinAsOf = max( $minAsOf, $purgeTime + self::$TINY_POSTIVE );
+
+               if ( $value === false ) {
+                       return false;
+               } elseif ( $safeMinAsOf > 0 && $asOf < $minAsOf ) {
+                       return false;
+               }
+
+               return true;
+       }
+
+       /**
+        * @param mixed $value
+        * @param int $ttl Seconds to live or zero for "indefinite"
+        * @param int|null $version Value version number or null if not versioned
+        * @param float $now Unix Current timestamp just before calling set()
+        * @param float $walltime How long it took to generate the value in seconds
+        * @return array
+        */
+       private function wrap( $value, $ttl, $version, $now, $walltime ) {
+               // Returns keys in ascending integer order for PHP7 array packing:
+               // https://nikic.github.io/2014/12/22/PHPs-new-hashtable-implementation.html
+               $wrapped = [
+                       self::$FLD_FORMAT_VERSION => self::$VERSION,
+                       self::$FLD_VALUE => $value,
+                       self::$FLD_TTL => $ttl,
+                       self::$FLD_TIME => $now
+               ];
+               if ( $version !== null ) {
+                       $wrapped[self::$FLD_VALUE_VERSION] = $version;
+               }
+               if ( $walltime >= self::$GENERATION_SLOW_SEC ) {
+                       $wrapped[self::$FLD_GENERATION_TIME] = $walltime;
+               }
+
+               return $wrapped;
+       }
+
+       /**
+        * @param array|string|bool $wrapped The entry at a cache key
+        * @param float $now Unix Current timestamp (preferrably pre-query)
+        * @return array (value or false if absent/tombstoned/malformed, value metadata map).
+        * The cache key metadata includes the following metadata:
+        *   - asOf: UNIX timestamp of the value or null if there is no value
+        *   - curTTL: remaining time-to-live (negative if tombstoned) or null if there is no value
+        *   - version: value version number or null if the if there is no value
+        *   - tombAsOf: UNIX timestamp of the tombstone or null if there is no tombstone
+        */
+       private function unwrap( $wrapped, $now ) {
+               $value = false;
+               $info = [ 'asOf' => null, 'curTTL' => null, 'version' => null, 'tombAsOf' => null ];
+
+               if ( is_array( $wrapped ) ) {
+                       // Entry expected to be a cached value; validate it
+                       if (
+                               ( $wrapped[self::$FLD_FORMAT_VERSION] ?? null ) === self::$VERSION &&
+                               $wrapped[self::$FLD_TIME] >= $this->epoch
+                       ) {
+                               if ( $wrapped[self::$FLD_TTL] > 0 ) {
+                                       // Get the approximate time left on the key
+                                       $age = $now - $wrapped[self::$FLD_TIME];
+                                       $curTTL = max( $wrapped[self::$FLD_TTL] - $age, 0.0 );
+                               } else {
+                                       // Key had no TTL, so the time left is unbounded
+                                       $curTTL = INF;
+                               }
+                               $value = $wrapped[self::$FLD_VALUE];
+                               $info['version'] = $wrapped[self::$FLD_VALUE_VERSION] ?? null;
+                               $info['asOf'] = $wrapped[self::$FLD_TIME];
+                               $info['curTTL'] = $curTTL;
+                       }
+               } else {
+                       // Entry expected to be a tombstone; parse it
+                       $purge = $this->parsePurgeValue( $wrapped );
+                       if ( $purge !== false ) {
+                               // Tombstoned keys should always have a negative current $ttl
+                               $info['curTTL'] = min( $purge[self::$PURGE_TIME] - $now, self::$TINY_NEGATIVE );
+                               $info['tombAsOf'] = $purge[self::$PURGE_TIME];
+                       }
+               }
+
+               return [ $value, $info ];
+       }
+
+       /**
+        * @param string[] $keys
+        * @param string $prefix
+        * @return string[] Prefix keys; the order of $keys is preserved
+        */
+       protected static function prefixCacheKeys( array $keys, $prefix ) {
+               $res = [];
+               foreach ( $keys as $key ) {
+                       $res[] = $prefix . $key;
+               }
+
+               return $res;
+       }
+
+       /**
+        * @param string $key String of the format <scope>:<class>[:<class or variable>]...
+        * @return string A collection name to describe this class of key
+        */
+       private function determineKeyClassForStats( $key ) {
+               $parts = explode( ':', $key, 3 );
+
+               return $parts[1] ?? $parts[0]; // sanity
+       }
+
+       /**
+        * @param string|array|bool $value Possible string of the form "PURGED:<timestamp>:<holdoff>"
+        * @return array|bool Array containing a UNIX timestamp (float) and holdoff period (integer),
+        *  or false if value isn't a valid purge value
+        */
+       private function parsePurgeValue( $value ) {
+               if ( !is_string( $value ) ) {
+                       return false;
+               }
+
+               $segments = explode( ':', $value, 3 );
+               if (
+                       !isset( $segments[0] ) ||
+                       !isset( $segments[1] ) ||
+                       "{$segments[0]}:" !== self::$PURGE_VAL_PREFIX
+               ) {
+                       return false;
+               }
+
+               if ( !isset( $segments[2] ) ) {
+                       // Back-compat with old purge values without holdoff
+                       $segments[2] = self::HOLDOFF_TTL;
+               }
+
+               if ( $segments[1] < $this->epoch ) {
+                       // Values this old are ignored
+                       return false;
+               }
+
+               return [
+                       self::$PURGE_TIME => (float)$segments[1],
+                       self::$PURGE_HOLDOFF => (int)$segments[2],
+               ];
+       }
+
+       /**
+        * @param float $timestamp
+        * @param int $holdoff In seconds
+        * @return string Wrapped purge value
+        */
+       private function makePurgeValue( $timestamp, $holdoff ) {
+               return self::$PURGE_VAL_PREFIX . (float)$timestamp . ':' . (int)$holdoff;
+       }
+
+       /**
+        * @param string $group
+        * @return MapCacheLRU
+        */
+       private function getProcessCache( $group ) {
+               if ( !isset( $this->processCaches[$group] ) ) {
+                       list( , $size ) = explode( ':', $group );
+                       $this->processCaches[$group] = new MapCacheLRU( (int)$size );
+               }
+
+               return $this->processCaches[$group];
+       }
+
+       /**
+        * @param string $key
+        * @param int $version
+        * @return string
+        */
+       private function getProcessCacheKey( $key, $version ) {
+               return $key . ' ' . (int)$version;
+       }
+
+       /**
+        * @param ArrayIterator $keys
+        * @param array $opts
+        * @return string[] Map of (ID => cache key)
+        */
+       private function getNonProcessCachedMultiKeys( ArrayIterator $keys, array $opts ) {
+               $pcTTL = $opts['pcTTL'] ?? self::TTL_UNCACHEABLE;
+
+               $keysMissing = [];
+               if ( $pcTTL > 0 && $this->callbackDepth == 0 ) {
+                       $version = $opts['version'] ?? null;
+                       $pCache = $this->getProcessCache( $opts['pcGroup'] ?? self::PC_PRIMARY );
+                       foreach ( $keys as $key => $id ) {
+                               if ( !$pCache->has( $this->getProcessCacheKey( $key, $version ), $pcTTL ) ) {
+                                       $keysMissing[$id] = $key;
+                               }
+                       }
+               }
+
+               return $keysMissing;
+       }
+
+       /**
+        * @param string[] $keys
+        * @param string[]|string[][] $checkKeys
+        * @return string[] List of cache keys
+        */
+       private function getRawKeysForWarmup( array $keys, array $checkKeys ) {
+               if ( !$keys ) {
+                       return [];
+               }
+
+               $keysWarmUp = [];
+               // Get all the value keys to fetch...
+               foreach ( $keys as $key ) {
+                       $keysWarmUp[] = self::$VALUE_KEY_PREFIX . $key;
+               }
+               // Get all the check keys to fetch...
+               foreach ( $checkKeys as $i => $checkKeyOrKeys ) {
+                       if ( is_int( $i ) ) {
+                               // Single check key that applies to all value keys
+                               $keysWarmUp[] = self::$TIME_KEY_PREFIX . $checkKeyOrKeys;
+                       } else {
+                               // List of check keys that apply to value key $i
+                               $keysWarmUp = array_merge(
+                                       $keysWarmUp,
+                                       self::prefixCacheKeys( $checkKeyOrKeys, self::$TIME_KEY_PREFIX )
+                               );
+                       }
+               }
+
+               $warmupCache = $this->cache->getMulti( $keysWarmUp );
+               $warmupCache += array_fill_keys( $keysWarmUp, false );
+
+               return $warmupCache;
+       }
+
+       /**
+        * @return float UNIX timestamp
+        * @codeCoverageIgnore
+        */
+       protected function getCurrentTime() {
+               if ( $this->wallClockOverride ) {
+                       return $this->wallClockOverride;
+               }
+
+               $clockTime = (float)time(); // call this first
+               // microtime() uses an initial gettimeofday() call added to usage clocks.
+               // This can severely drift from time() and the microtime() value of other threads
+               // due to undercounting of the amount of time elapsed. Instead of seeing the current
+               // time as being in the past, use the value of time(). This avoids setting cache values
+               // that will immediately be seen as expired and possibly cause stampedes.
+               return max( microtime( true ), $clockTime );
+       }
+
+       /**
+        * @param float|null &$time Mock UNIX timestamp for testing
+        * @codeCoverageIgnore
+        */
+       public function setMockTime( &$time ) {
+               $this->wallClockOverride =& $time;
+               $this->cache->setMockTime( $time );
+       }
+}
diff --git a/includes/libs/objectcache/wancache/WANObjectCacheReaper.php b/includes/libs/objectcache/wancache/WANObjectCacheReaper.php
new file mode 100644 (file)
index 0000000..fb8a754
--- /dev/null
@@ -0,0 +1,199 @@
+<?php
+/**
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License along
+ * with this program; if not, write to the Free Software Foundation, Inc.,
+ * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ * http://www.gnu.org/copyleft/gpl.html
+ *
+ * @file
+ * @ingroup Cache
+ */
+
+use Psr\Log\LoggerAwareInterface;
+use Psr\Log\LoggerInterface;
+use Psr\Log\NullLogger;
+use Wikimedia\ScopedCallback;
+
+/**
+ * Class for scanning through chronological, log-structured data or change logs
+ * and locally purging cache keys related to entities that appear in this data.
+ *
+ * This is useful for repairing cache when purges are missed by using a reliable
+ * stream, such as Kafka or a replicated MySQL table. Purge loss between datacenters
+ * is expected to be more common than within them.
+ *
+ * @since 1.28
+ */
+class WANObjectCacheReaper implements LoggerAwareInterface {
+       /** @var WANObjectCache */
+       protected $cache;
+       /** @var BagOStuff */
+       protected $store;
+       /** @var callable */
+       protected $logChunkCallback;
+       /** @var callable */
+       protected $keyListCallback;
+       /** @var LoggerInterface */
+       protected $logger;
+
+       /** @var string */
+       protected $channel;
+       /** @var int */
+       protected $initialStartWindow;
+
+       /**
+        * @param WANObjectCache $cache Cache to reap bad keys from
+        * @param BagOStuff $store Cache to store positions use for locking
+        * @param callable $logCallback Callback taking arguments:
+        *          - The starting position as a UNIX timestamp
+        *          - The starting unique ID used for breaking timestamp collisions or null
+        *          - The ending position as a UNIX timestamp
+        *          - The maximum number of results to return
+        *        It returns a list of maps of (key: cache key, pos: UNIX timestamp, id: unique ID)
+        *        for each key affected, with the corrosponding event timestamp/ID information.
+        *        The events should be in ascending order, by (timestamp,id).
+        * @param callable $keyCallback Callback taking arguments:
+        *          - The WANObjectCache instance
+        *          - An object from the event log
+        *        It should return a list of WAN cache keys.
+        *        The callback must fully duck-type test the object, since can be any model class.
+        * @param array $params Additional options:
+        *          - channel: the name of the update event stream.
+        *          - initialStartWindow: seconds back in time to start if the position is lost.
+        *            Default: 1 hour.
+        *          - logger: an SPL monolog instance [optional]
+        */
+       public function __construct(
+               WANObjectCache $cache,
+               BagOStuff $store,
+               callable $logCallback,
+               callable $keyCallback,
+               array $params
+       ) {
+               $this->cache = $cache;
+               $this->store = $store;
+
+               $this->logChunkCallback = $logCallback;
+               $this->keyListCallback = $keyCallback;
+               if ( isset( $params['channel'] ) ) {
+                       $this->channel = $params['channel'];
+               } else {
+                       throw new UnexpectedValueException( "No channel specified." );
+               }
+
+               $this->initialStartWindow = $params['initialStartWindow'] ?? 3600;
+               $this->logger = $params['logger'] ?? new NullLogger();
+       }
+
+       public function setLogger( LoggerInterface $logger ) {
+               $this->logger = $logger;
+       }
+
+       /**
+        * Check and reap stale keys based on a chunk of events
+        *
+        * @param int $n Number of events
+        * @return int Number of keys checked
+        */
+       final public function invoke( $n = 100 ) {
+               $posKey = $this->store->makeGlobalKey( 'WANCache', 'reaper', $this->channel );
+               $scopeLock = $this->store->getScopedLock( "$posKey:busy", 0 );
+               if ( !$scopeLock ) {
+                       return 0;
+               }
+
+               $now = time();
+               $status = $this->store->get( $posKey );
+               if ( !$status ) {
+                       $status = [ 'pos' => $now - $this->initialStartWindow, 'id' => null ];
+               }
+
+               // Get events for entities who's keys tombstones/hold-off should have expired by now
+               $events = call_user_func_array(
+                       $this->logChunkCallback,
+                       [ $status['pos'], $status['id'], $now - WANObjectCache::HOLDOFF_TTL - 1, $n ]
+               );
+
+               $event = null;
+               $keyEvents = [];
+               foreach ( $events as $event ) {
+                       $keys = call_user_func_array(
+                               $this->keyListCallback,
+                               [ $this->cache, $event['item'] ]
+                       );
+                       foreach ( $keys as $key ) {
+                               unset( $keyEvents[$key] ); // use only the latest per key
+                               $keyEvents[$key] = [
+                                       'pos' => $event['pos'],
+                                       'id' => $event['id']
+                               ];
+                       }
+               }
+
+               $purgeCount = 0;
+               $lastOkEvent = null;
+               foreach ( $keyEvents as $key => $keyEvent ) {
+                       if ( !$this->cache->reap( $key, $keyEvent['pos'] ) ) {
+                               break;
+                       }
+                       ++$purgeCount;
+                       $lastOkEvent = $event;
+               }
+
+               if ( $lastOkEvent ) {
+                       $ok = $this->store->merge(
+                               $posKey,
+                               function ( $bag, $key, $curValue ) use ( $lastOkEvent ) {
+                                       if ( !$curValue ) {
+                                               // Use new position
+                                       } else {
+                                               $curCoord = [ $curValue['pos'], $curValue['id'] ];
+                                               $newCoord = [ $lastOkEvent['pos'], $lastOkEvent['id'] ];
+                                               if ( $newCoord < $curCoord ) {
+                                                       // Keep prior position instead of rolling it back
+                                                       return $curValue;
+                                               }
+                                       }
+
+                                       return [
+                                               'pos' => $lastOkEvent['pos'],
+                                               'id' => $lastOkEvent['id'],
+                                               'ctime' => $curValue ? $curValue['ctime'] : date( 'c' )
+                                       ];
+                               },
+                               IExpiringStore::TTL_INDEFINITE
+                       );
+
+                       $pos = $lastOkEvent['pos'];
+                       $id = $lastOkEvent['id'];
+                       if ( $ok ) {
+                               $this->logger->info( "Updated cache reap position ($pos, $id)." );
+                       } else {
+                               $this->logger->error( "Could not update cache reap position ($pos, $id)." );
+                       }
+               }
+
+               ScopedCallback::consume( $scopeLock );
+
+               return $purgeCount;
+       }
+
+       /**
+        * @return array|bool Returns (pos, id) map or false if not set
+        */
+       public function getState() {
+               $posKey = $this->store->makeGlobalKey( 'WANCache', 'reaper', $this->channel );
+
+               return $this->store->get( $posKey );
+       }
+}
index 2c9858a..f27d042 100644 (file)
@@ -140,10 +140,6 @@ class DBConnRef implements IDatabase {
                throw new DBUnexpectedError( $this, "Database injection is disallowed to enable reuse." );
        }
 
-       public function implicitGroupby() {
-               return $this->__call( __FUNCTION__, func_get_args() );
-       }
-
        public function implicitOrderby() {
                return $this->__call( __FUNCTION__, func_get_args() );
        }
@@ -152,10 +148,6 @@ class DBConnRef implements IDatabase {
                return $this->__call( __FUNCTION__, func_get_args() );
        }
 
-       public function doneWrites() {
-               return $this->__call( __FUNCTION__, func_get_args() );
-       }
-
        public function lastDoneWrites() {
                return $this->__call( __FUNCTION__, func_get_args() );
        }
@@ -218,13 +210,6 @@ class DBConnRef implements IDatabase {
                return $this->__call( __FUNCTION__, func_get_args() );
        }
 
-       /**
-        * @codeCoverageIgnore
-        */
-       public function getWikiID() {
-               return $this->getDomainID();
-       }
-
        public function getType() {
                if ( $this->conn === null ) {
                        // Avoid triggering a database connection
index 60062fb..e82c735 100644 (file)
@@ -61,7 +61,7 @@ abstract class Database implements IDatabase, IMaintainableDatabase, LoggerAware
        protected $cliMode;
        /** @var string Agent name for query profiling */
        protected $agent;
-       /** @var int Bitfield of class DBO_* constants */
+       /** @var int Bit field of class DBO_* constants */
        protected $flags;
        /** @var array LoadBalancer tracking information */
        protected $lbInfo = [];
@@ -217,6 +217,18 @@ abstract class Database implements IDatabase, IMaintainableDatabase, LoggerAware
        /** @var float Assume an insert of this many rows or less should be fast to replicate */
        private static $SMALL_WRITE_ROWS = 100;
 
+       /** @var string[] List of DBO_* flags that can be changed after connection */
+       protected static $MUTABLE_FLAGS = [
+               'DBO_DEBUG',
+               'DBO_NOBUFFER',
+               'DBO_TRX',
+               'DBO_DDLMODE',
+       ];
+       /** @var int Bit field of all DBO_* flags that can be changed after connection */
+       protected static $DBO_MUTABLE = (
+               self::DBO_DEBUG | self::DBO_NOBUFFER | self::DBO_TRX | self::DBO_DDLMODE
+       );
+
        /**
         * @note exceptions for missing libraries/drivers should be thrown in initConnection()
         * @param array $params Parameters passed from Database::factory()
@@ -283,23 +295,18 @@ abstract class Database implements IDatabase, IMaintainableDatabase, LoggerAware
        /**
         * Actually connect to the database over the wire (or to local files)
         *
-        * @throws InvalidArgumentException
         * @throws DBConnectionError
         * @since 1.31
         */
        protected function doInitConnection() {
-               if ( strlen( $this->connectionParams['user'] ) ) {
-                       $this->open(
-                               $this->connectionParams['host'],
-                               $this->connectionParams['user'],
-                               $this->connectionParams['password'],
-                               $this->connectionParams['dbname'],
-                               $this->connectionParams['schema'],
-                               $this->connectionParams['tablePrefix']
-                       );
-               } else {
-                       throw new InvalidArgumentException( "No database user provided" );
-               }
+               $this->open(
+                       $this->connectionParams['host'],
+                       $this->connectionParams['user'],
+                       $this->connectionParams['password'],
+                       $this->connectionParams['dbname'],
+                       $this->connectionParams['schema'],
+                       $this->connectionParams['tablePrefix']
+               );
        }
 
        /**
@@ -335,7 +342,7 @@ abstract class Database implements IDatabase, IMaintainableDatabase, LoggerAware
         *      equivalent to a "database" in MySQL. Note that MySQL and SQLite do not use schemas.
         *   - tablePrefix : Optional table prefix that is implicitly added on to all table names
         *      recognized in queries. This can be used in place of schemas for handle site farms.
-        *   - flags : Optional bitfield of DBO_* constants that define connection, protocol,
+        *   - flags : Optional bit field of DBO_* constants that define connection, protocol,
         *      buffering, and transaction behavior. It is STRONGLY adviced to leave the DBO_DEFAULT
         *      flag in place UNLESS this this database simply acts as a key/value store.
         *   - driver: Optional name of a specific DB client driver. For MySQL, there is only the
@@ -613,10 +620,6 @@ abstract class Database implements IDatabase, IMaintainableDatabase, LoggerAware
                return $this->lazyMasterHandle;
        }
 
-       public function implicitGroupby() {
-               return true;
-       }
-
        public function implicitOrderby() {
                return true;
        }
@@ -625,10 +628,6 @@ abstract class Database implements IDatabase, IMaintainableDatabase, LoggerAware
                return $this->lastQuery;
        }
 
-       public function doneWrites() {
-               return (bool)$this->lastWriteTime;
-       }
-
        public function lastDoneWrites() {
                return $this->lastWriteTime ?: false;
        }
@@ -741,24 +740,32 @@ abstract class Database implements IDatabase, IMaintainableDatabase, LoggerAware
        }
 
        public function setFlag( $flag, $remember = self::REMEMBER_NOTHING ) {
-               if ( ( $flag & self::DBO_IGNORE ) ) {
-                       throw new UnexpectedValueException( "Modifying DBO_IGNORE is not allowed" );
+               if ( $flag & ~static::$DBO_MUTABLE ) {
+                       throw new DBUnexpectedError(
+                               $this,
+                               "Got $flag (allowed: " . implode( ', ', static::$MUTABLE_FLAGS ) . ')'
+                       );
                }
 
                if ( $remember === self::REMEMBER_PRIOR ) {
                        array_push( $this->priorFlags, $this->flags );
                }
+
                $this->flags |= $flag;
        }
 
        public function clearFlag( $flag, $remember = self::REMEMBER_NOTHING ) {
-               if ( ( $flag & self::DBO_IGNORE ) ) {
-                       throw new UnexpectedValueException( "Modifying DBO_IGNORE is not allowed" );
+               if ( $flag & ~static::$DBO_MUTABLE ) {
+                       throw new DBUnexpectedError(
+                               $this,
+                               "Got $flag (allowed: " . implode( ', ', static::$MUTABLE_FLAGS ) . ')'
+                       );
                }
 
                if ( $remember === self::REMEMBER_PRIOR ) {
                        array_push( $this->priorFlags, $this->flags );
                }
+
                $this->flags &= ~$flag;
        }
 
@@ -776,26 +783,13 @@ abstract class Database implements IDatabase, IMaintainableDatabase, LoggerAware
        }
 
        public function getFlag( $flag ) {
-               return (bool)( $this->flags & $flag );
-       }
-
-       /**
-        * @param string $name Class field name
-        * @return mixed
-        * @deprecated Since 1.28
-        */
-       public function getProperty( $name ) {
-               return $this->$name;
+               return ( ( $this->flags & $flag ) === $flag );
        }
 
        public function getDomainID() {
                return $this->currentDomain->getId();
        }
 
-       final public function getWikiID() {
-               return $this->getDomainID();
-       }
-
        /**
         * Get information about an index into an object
         * @param string $table Table name
@@ -929,7 +923,7 @@ abstract class Database implements IDatabase, IMaintainableDatabase, LoggerAware
                        $closed = true; // already closed; nothing to do
                }
 
-               $this->conn = false;
+               $this->conn = null;
 
                // Throw any unexpected errors after having disconnected
                if ( $exception instanceof Exception ) {
@@ -1177,7 +1171,7 @@ abstract class Database implements IDatabase, IMaintainableDatabase, LoggerAware
         *
         * @param string $sql Original SQL query
         * @param string $fname Name of the calling function
-        * @param int $flags Bitfield of class QUERY_* constants
+        * @param int $flags Bit field of class QUERY_* constants
         * @return array An n-tuple of:
         *   - mixed|bool: An object, resource, or true on success; false on failure
         *   - string: The result of calling lastError()
@@ -1265,7 +1259,7 @@ abstract class Database implements IDatabase, IMaintainableDatabase, LoggerAware
         * @param string $commentedSql SQL query with debugging/trace comment
         * @param bool $isPermWrite Whether the query is a (non-temporary table) write
         * @param string $fname Name of the calling function
-        * @param int $flags Bitfield of class QUERY_* constants
+        * @param int $flags Bit field of class QUERY_* constants
         * @return array An n-tuple of:
         *   - mixed|bool: An object, resource, or true on success; false on failure
         *   - string: The result of calling lastError()
@@ -1343,7 +1337,7 @@ abstract class Database implements IDatabase, IMaintainableDatabase, LoggerAware
                // Avoid the overhead of logging calls unless debug mode is enabled
                if ( $this->getFlag( self::DBO_DEBUG ) ) {
                        $this->queryLogger->debug(
-                               "{method} [{runtime}s]: $sql",
+                               "{method} [{runtime}s] {db_host}: $sql",
                                [
                                        'method' => $fname,
                                        'db_host' => $this->getServer(),
@@ -1570,9 +1564,7 @@ abstract class Database implements IDatabase, IMaintainableDatabase, LoggerAware
                if ( $ignore ) {
                        $this->queryLogger->debug( "SQL ERROR (ignored): $error" );
                } else {
-                       $exception = $this->getQueryExceptionAndLog( $error, $errno, $sql, $fname );
-
-                       throw $exception;
+                       throw $this->getQueryExceptionAndLog( $error, $errno, $sql, $fname );
                }
        }
 
@@ -1584,19 +1576,18 @@ abstract class Database implements IDatabase, IMaintainableDatabase, LoggerAware
         * @return DBError
         */
        private function getQueryExceptionAndLog( $error, $errno, $sql, $fname ) {
-               $sql1line = mb_substr( str_replace( "\n", "\\n", $sql ), 0, 5 * 1024 );
                $this->queryLogger->error(
                        "{fname}\t{db_server}\t{errno}\t{error}\t{sql1line}",
                        $this->getLogContext( [
                                'method' => __METHOD__,
                                'errno' => $errno,
                                'error' => $error,
-                               'sql1line' => $sql1line,
+                               'sql1line' => mb_substr( str_replace( "\n", "\\n", $sql ), 0, 5 * 1024 ),
                                'fname' => $fname,
                                'trace' => ( new RuntimeException() )->getTraceAsString()
                        ] )
                );
-               $this->queryLogger->debug( "SQL ERROR: " . $error . "" );
+
                if ( $this->wasQueryTimeout( $error, $errno ) ) {
                        $e = new DBQueryTimeoutError( $this, $error, $errno, $sql, $fname );
                } elseif ( $this->wasConnectionError( $errno ) ) {
@@ -1608,6 +1599,25 @@ abstract class Database implements IDatabase, IMaintainableDatabase, LoggerAware
                return $e;
        }
 
+       /**
+        * @param string $error
+        * @return DBConnectionError
+        */
+       final protected function newExceptionAfterConnectError( $error ) {
+               // Connection was not fully initialized and is not safe for use
+               $this->conn = null;
+
+               $this->connLogger->error(
+                       "Error connecting to {db_server} as user {db_user}: {error}",
+                       $this->getLogContext( [
+                               'error' => $error,
+                               'trace' => ( new RuntimeException() )->getTraceAsString()
+                       ] )
+               );
+
+               return new DBConnectionError( $this, $error );
+       }
+
        public function freeResult( $res ) {
        }
 
@@ -4297,7 +4307,7 @@ abstract class Database implements IDatabase, IMaintainableDatabase, LoggerAware
         */
        protected function replaceLostConnection( $fname ) {
                $this->closeConnection();
-               $this->conn = false;
+               $this->conn = null;
 
                $this->handleSessionLossPreconnect();
 
@@ -4876,7 +4886,7 @@ abstract class Database implements IDatabase, IMaintainableDatabase, LoggerAware
 
                if ( $this->isOpen() ) {
                        // Open a new connection resource without messing with the old one
-                       $this->conn = false;
+                       $this->conn = null;
                        $this->trxEndCallbacks = []; // don't copy
                        $this->trxSectionCancelCallbacks = []; // don't copy
                        $this->handleSessionLossPreconnect(); // no trx or locks anymore
index d06bcb9..db029a3 100644 (file)
@@ -59,10 +59,6 @@ class DatabaseMssql extends Database {
        /** @var string[] */
        protected $ignoreErrors = [];
 
-       public function implicitGroupby() {
-               return false;
-       }
-
        public function implicitOrderby() {
                return false;
        }
@@ -79,53 +75,50 @@ class DatabaseMssql extends Database {
        }
 
        protected function open( $server, $user, $password, $dbName, $schema, $tablePrefix ) {
-               // Test for driver support, to avoid suppressed fatal error
                if ( !function_exists( 'sqlsrv_connect' ) ) {
                        throw new DBConnectionError(
                                $this,
-                               "Microsoft SQL Server Native (sqlsrv) functions missing.
-                               You can download the driver from: http://go.microsoft.com/fwlink/?LinkId=123470\n"
+                               "Microsoft SQL Server Native (sqlsrv) functions missing.\n
+                               You can download the driver from: http://go.microsoft.com/fwlink/?LinkId=123470"
                        );
                }
 
                $this->close();
+
+               if ( $schema !== null ) {
+                       throw $this->newExceptionAfterConnectError( "Got schema '$schema'; not supported." );
+               }
+
                $this->server = $server;
                $this->user = $user;
                $this->password = $password;
 
                $connectionInfo = [];
-
-               if ( $dbName != '' ) {
+               if ( strlen( $dbName ) ) {
                        $connectionInfo['Database'] = $dbName;
                }
-
-               // Decide which auth scenerio to use
-               // if we are using Windows auth, then don't add credentials to $connectionInfo
                if ( !$this->useWindowsAuth ) {
                        $connectionInfo['UID'] = $user;
                        $connectionInfo['PWD'] = $password;
                }
 
                AtEase::suppressWarnings();
-               $this->conn = sqlsrv_connect( $server, $connectionInfo );
+               $this->conn = sqlsrv_connect( $server, $connectionInfo ) ?: null;
                AtEase::restoreWarnings();
 
-               if ( $this->conn === false ) {
-                       $error = $this->lastError();
-                       $this->connLogger->error(
-                               "Error connecting to {db_server}: {error}",
-                               $this->getLogContext( [ 'method' => __METHOD__, 'error' => $error ] )
-                       );
-                       throw new DBConnectionError( $this, $error );
+               if ( !$this->conn ) {
+                       throw $this->newExceptionAfterConnectError( $this->lastError() );
                }
 
-               $this->currentDomain = new DatabaseDomain(
-                       ( $dbName != '' ) ? $dbName : null,
-                       null,
-                       $tablePrefix
-               );
-
-               return (bool)$this->conn;
+               try {
+                       $this->currentDomain = new DatabaseDomain(
+                               strlen( $dbName ) ? $dbName : null,
+                               null,
+                               $tablePrefix
+                       );
+               } catch ( Exception $e ) {
+                       throw $this->newExceptionAfterConnectError( $e->getMessage() );
+               }
        }
 
        /**
index 1e3fa84..b1a88ed 100644 (file)
@@ -125,7 +125,7 @@ abstract class DatabaseMysqlBase extends Database {
                $this->close();
 
                if ( $schema !== null ) {
-                       throw new DBExpectedError( $this, __CLASS__ . ": cannot use schemas ('$schema')" );
+                       throw $this->newExceptionAfterConnectError( "Got schema '$schema'; not supported." );
                }
 
                $this->server = $server;
@@ -135,23 +135,14 @@ abstract class DatabaseMysqlBase extends Database {
                $this->installErrorHandler();
                try {
                        $this->conn = $this->mysqlConnect( $this->server, $dbName );
-               } catch ( Exception $ex ) {
+               } catch ( Exception $e ) {
                        $this->restoreErrorHandler();
-                       throw $ex;
+                       throw $this->newExceptionAfterConnectError( $e->getMessage() );
                }
                $error = $this->restoreErrorHandler();
 
-               # Always log connection errors
                if ( !$this->conn ) {
-                       $error = $error ?: $this->lastError();
-                       $this->connLogger->error(
-                               "Error connecting to {db_server}: {error}",
-                               $this->getLogContext( [ 'method' => __METHOD__, 'error' => $error ] )
-                       );
-                       $this->connLogger->debug( "DB connection error\n" .
-                               "Server: $server, User: $user, Password: " .
-                               substr( $password, 0, 3 ) . "..., error: " . $error . "\n" );
-                       throw new DBConnectionError( $this, $error );
+                       throw $this->newExceptionAfterConnectError( $error ?: $this->lastError() );
                }
 
                try {
@@ -160,7 +151,6 @@ abstract class DatabaseMysqlBase extends Database {
                                null,
                                $tablePrefix
                        );
-
                        // Abstract over any insane MySQL defaults
                        $set = [ 'group_concat_max_len = 262144' ];
                        // Set SQL mode, default is turning them all off, can be overridden or skipped with null
@@ -185,11 +175,8 @@ abstract class DatabaseMysqlBase extends Database {
                                );
                        }
                } catch ( Exception $e ) {
-                       // Connection was not fully initialized and is not safe for use
-                       $this->conn = false;
+                       throw $this->newExceptionAfterConnectError( $e->getMessage() );
                }
-
-               return true;
        }
 
        protected function doSelectDomain( DatabaseDomain $domain ) {
@@ -234,7 +221,7 @@ abstract class DatabaseMysqlBase extends Database {
         *
         * @param string $realServer
         * @param string|null $dbName
-        * @return mixed Raw connection
+        * @return mixed|null Driver connection handle
         * @throws DBConnectionError
         */
        abstract protected function mysqlConnect( $realServer, $dbName );
index 0f444cd..ddb3944 100644 (file)
@@ -54,14 +54,14 @@ class DatabaseMysqli extends DatabaseMysqlBase {
        /**
         * @param string $realServer
         * @param string|null $dbName
-        * @return bool|mysqli
+        * @return mysqli|null
         * @throws DBConnectionError
         */
        protected function mysqlConnect( $realServer, $dbName ) {
-               # Avoid suppressed fatal error, which is very hard to track down
                if ( !function_exists( 'mysqli_init' ) ) {
-                       throw new DBConnectionError( $this, "MySQLi functions missing,"
-                               . " have you compiled PHP with the --with-mysqli option?\n" );
+                       throw $this->newExceptionAfterConnectError(
+                               "MySQLi functions missing, have you compiled PHP with the --with-mysqli option?"
+                       );
                }
 
                // Other than mysql_connect, mysqli_real_connect expects an explicit port
@@ -84,7 +84,7 @@ class DatabaseMysqli extends DatabaseMysqlBase {
                $mysqli = mysqli_init();
 
                $connFlags = 0;
-               if ( $this->flags & self::DBO_SSL ) {
+               if ( $this->getFlag( self::DBO_SSL ) ) {
                        $connFlags |= MYSQLI_CLIENT_SSL;
                        $mysqli->ssl_set(
                                $this->sslKeyPath,
@@ -94,10 +94,10 @@ class DatabaseMysqli extends DatabaseMysqlBase {
                                $this->sslCiphers
                        );
                }
-               if ( $this->flags & self::DBO_COMPRESS ) {
+               if ( $this->getFlag( self::DBO_COMPRESS ) ) {
                        $connFlags |= MYSQLI_CLIENT_COMPRESS;
                }
-               if ( $this->flags & self::DBO_PERSISTENT ) {
+               if ( $this->getFlag( self::DBO_PERSISTENT ) ) {
                        $realServer = 'p:' . $realServer;
                }
 
@@ -122,7 +122,7 @@ class DatabaseMysqli extends DatabaseMysqlBase {
                        return $mysqli;
                }
 
-               return false;
+               return null;
        }
 
        /**
index 840b428..a7ebc86 100644 (file)
@@ -31,22 +31,19 @@ use Exception;
  * @ingroup Database
  */
 class DatabasePostgres extends Database {
-       /** @var int|bool */
-       protected $port;
-
-       /** @var resource */
-       protected $lastResultHandle = null;
-
-       /** @var float|string */
-       private $numericVersion = null;
-       /** @var string Connect string to open a PostgreSQL connection */
-       private $connectString;
+       /** @var int|null */
+       private $port;
        /** @var string */
        private $coreSchema;
        /** @var string */
        private $tempSchema;
        /** @var string[] Map of (reserved table name => alternate table name) */
        private $keywordTableMap = [];
+       /** @var float|string */
+       private $numericVersion;
+
+       /** @var resource|null */
+       private $lastResultHandle;
 
        /**
         * @see Database::__construct()
@@ -54,7 +51,7 @@ class DatabasePostgres extends Database {
         *   - keywordTableMap : Map of reserved table names to alternative table names to use
         */
        public function __construct( array $params ) {
-               $this->port = $params['port'] ?? false;
+               $this->port = intval( $params['port'] ?? null );
                $this->keywordTableMap = $params['keywordTableMap'] ?? [];
 
                parent::__construct( $params );
@@ -64,10 +61,6 @@ class DatabasePostgres extends Database {
                return 'postgres';
        }
 
-       public function implicitGroupby() {
-               return false;
-       }
-
        public function implicitOrderby() {
                return false;
        }
@@ -87,13 +80,11 @@ class DatabasePostgres extends Database {
        }
 
        protected function open( $server, $user, $password, $dbName, $schema, $tablePrefix ) {
-               // Test for Postgres support, to avoid suppressed fatal error
                if ( !function_exists( 'pg_connect' ) ) {
-                       throw new DBConnectionError(
-                               $this,
+                       throw $this->newExceptionAfterConnectError(
                                "Postgres functions missing, have you compiled PHP with the --with-pgsql\n" .
                                "option? (Note: if you recently installed PHP, you may need to restart your\n" .
-                               "webserver and database)\n"
+                               "webserver and database)"
                        );
                }
 
@@ -104,60 +95,47 @@ class DatabasePostgres extends Database {
                $this->password = $password;
 
                $connectVars = [
-                       // pg_connect() user $user as the default database. Since a database is *required*,
-                       // at least pick a "don't care" database that is more likely to exist. This case
-                       // arrises when LoadBalancer::getConnection( $i, [], '' ) is used.
+                       // pg_connect() user $user as the default database. Since a database is required,
+                       // then pick a "don't care" database that is more likely to exist than that one.
                        'dbname' => strlen( $dbName ) ? $dbName : 'postgres',
                        'user' => $user,
                        'password' => $password
                ];
-               if ( $server != false && $server != '' ) {
+               if ( strlen( $server ) ) {
                        $connectVars['host'] = $server;
                }
-               if ( (int)$this->port > 0 ) {
-                       $connectVars['port'] = (int)$this->port;
+               if ( $this->port > 0 ) {
+                       $connectVars['port'] = $this->port;
                }
-               if ( $this->flags & self::DBO_SSL ) {
+               if ( $this->getFlag( self::DBO_SSL ) ) {
                        $connectVars['sslmode'] = 'require';
                }
-
-               $this->connectString = $this->makeConnectionString( $connectVars );
+               $connectString = $this->makeConnectionString( $connectVars );
 
                $this->installErrorHandler();
                try {
-                       // Use new connections to let LoadBalancer/LBFactory handle reuse
-                       $this->conn = pg_connect( $this->connectString, PGSQL_CONNECT_FORCE_NEW );
-               } catch ( Exception $ex ) {
+                       $this->conn = pg_connect( $connectString, PGSQL_CONNECT_FORCE_NEW ) ?: null;
+               } catch ( Exception $e ) {
                        $this->restoreErrorHandler();
-                       throw $ex;
+                       throw $this->newExceptionAfterConnectError( $e->getMessage() );
                }
-               $phpError = $this->restoreErrorHandler();
+               $error = $this->restoreErrorHandler();
 
                if ( !$this->conn ) {
-                       $this->queryLogger->debug(
-                               "DB connection error\n" .
-                               "Server: $server, Database: $dbName, User: $user, Password: " .
-                               substr( $password, 0, 3 ) . "...\n"
-                       );
-                       $this->queryLogger->debug( $this->lastError() . "\n" );
-                       throw new DBConnectionError( $this, str_replace( "\n", ' ', $phpError ) );
+                       throw $this->newExceptionAfterConnectError( $error ?: $this->lastError() );
                }
 
                try {
-                       // If called from the command-line (e.g. importDump), only show errors.
-                       // No transaction should be open at this point, so the problem of the SET
-                       // effects being rolled back should not be an issue.
+                       // Since no transaction is active at this point, any SET commands should apply
+                       // for the entire session (e.g. will not be reverted on transaction rollback).
                        // See https://www.postgresql.org/docs/8.3/sql-set.html
-                       $variables = [];
-                       if ( $this->cliMode ) {
-                               $variables['client_min_messages'] = 'ERROR';
-                       }
-                       $variables += [
+                       $variables = [
                                'client_encoding' => 'UTF8',
                                'datestyle' => 'ISO, YMD',
                                'timezone' => 'GMT',
                                'standard_conforming_strings' => 'on',
-                               'bytea_output' => 'escape'
+                               'bytea_output' => 'escape',
+                               'client_min_messages' => 'ERROR'
                        ];
                        foreach ( $variables as $var => $val ) {
                                $this->query(
@@ -166,12 +144,10 @@ class DatabasePostgres extends Database {
                                        self::QUERY_IGNORE_DBO_TRX | self::QUERY_NO_RETRY
                                );
                        }
-
                        $this->determineCoreSchema( $schema );
                        $this->currentDomain = new DatabaseDomain( $dbName, $schema, $tablePrefix );
                } catch ( Exception $e ) {
-                       // Connection was not fully initialized and is not safe for use
-                       $this->conn = false;
+                       throw $this->newExceptionAfterConnectError( $e->getMessage() );
                }
        }
 
@@ -1026,7 +1002,7 @@ __INDEXATTR__;
         * Values may contain magic keywords like "$user"
         * @since 1.19
         *
-        * @param array $search_path List of schemas to be searched by default
+        * @param string[] $search_path List of schemas to be searched by default
         */
        private function setSearchPath( $search_path ) {
                $this->query(
@@ -1066,11 +1042,7 @@ __INDEXATTR__;
                                $this->queryLogger->debug(
                                        "Schema \"" . $desiredSchema . "\" already in the search path\n" );
                        } else {
-                               /**
-                                * Prepend our schema (e.g. 'mediawiki') in front
-                                * of the search path
-                                * Fixes T17816
-                                */
+                               // Prepend the desired schema to the search path (T17816)
                                $search_path = $this->getSearchPath();
                                array_unshift( $search_path, $this->addIdentifierQuotes( $desiredSchema ) );
                                $this->setSearchPath( $search_path );
index 11dda2f..cb1b842 100644 (file)
@@ -60,6 +60,9 @@ class DatabaseSqlite extends Database {
        /** @var bool Whether full text is enabled */
        private static $fulltextEnabled = null;
 
+       /** @var string[] See https://www.sqlite.org/lang_transaction.html */
+       private static $VALID_TRX_MODES = [ '', 'DEFERRED', 'IMMEDIATE', 'EXCLUSIVE' ];
+
        /**
         * Additional params include:
         *   - dbDirectory : directory containing the DB and the lock file directory
@@ -77,8 +80,7 @@ class DatabaseSqlite extends Database {
                        $this->dbDir = $p['dbDirectory'];
                }
 
-               // Set a dummy user to make initConnection() trigger open()
-               parent::__construct( [ 'user' => '@' ] + $p );
+               parent::__construct( $p );
 
                $this->trxMode = strtoupper( $p['trxMode'] ?? '' );
 
@@ -123,22 +125,13 @@ class DatabaseSqlite extends Database {
                return 'sqlite';
        }
 
-       /**
-        * @todo Check if it should be true like parent class
-        *
-        * @return bool
-        */
-       public function implicitGroupby() {
-               return false;
-       }
-
        protected function open( $server, $user, $pass, $dbName, $schema, $tablePrefix ) {
                $this->close();
 
                // Note that for SQLite, $server, $user, and $pass are ignored
 
                if ( $schema !== null ) {
-                       throw new DBExpectedError( $this, __CLASS__ . ": cannot use schemas ('$schema')" );
+                       throw $this->newExceptionAfterConnectError( "Got schema '$schema'; not supported." );
                }
 
                if ( $this->dbPath !== null ) {
@@ -146,59 +139,48 @@ class DatabaseSqlite extends Database {
                } elseif ( $this->dbDir !== null ) {
                        $path = self::generateFileName( $this->dbDir, $dbName );
                } else {
-                       throw new DBExpectedError( $this, __CLASS__ . ": DB path or directory required" );
+                       throw $this->newExceptionAfterConnectError( "DB path or directory required" );
                }
 
-               if ( !in_array( $this->trxMode, [ '', 'DEFERRED', 'IMMEDIATE', 'EXCLUSIVE' ], true ) ) {
-                       throw new DBExpectedError(
-                               $this,
-                               __CLASS__ . ": invalid transaction mode '{$this->trxMode}'"
-                       );
+               if ( !self::isProcessMemoryPath( $path ) && !is_readable( $path ) ) {
+                       throw $this->newExceptionAfterConnectError( 'SQLite database file is not readable' );
+               } elseif ( !in_array( $this->trxMode, self::$VALID_TRX_MODES, true ) ) {
+                       throw $this->newExceptionAfterConnectError( "Got mode '{$this->trxMode}' for BEGIN" );
                }
 
-               if ( !self::isProcessMemoryPath( $path ) && !is_readable( $path ) ) {
-                       $error = "SQLite database file not readable";
-                       $this->connLogger->error(
-                               "Error connecting to {db_server}: {error}",
-                               $this->getLogContext( [ 'method' => __METHOD__, 'error' => $error ] )
-                       );
-                       throw new DBConnectionError( $this, $error );
+               $attributes = [];
+               if ( $this->getFlag( self::DBO_PERSISTENT ) ) {
+                       // Persistent connections can avoid some schema index reading overhead.
+                       // On the other hand, they can cause horrible contention with DBO_TRX.
+                       if ( $this->getFlag( self::DBO_TRX ) || $this->getFlag( self::DBO_DEFAULT ) ) {
+                               $this->connLogger->warning(
+                                       __METHOD__ . ": ignoring DBO_PERSISTENT due to DBO_TRX or DBO_DEFAULT",
+                                       $this->getLogContext()
+                               );
+                       } else {
+                               $attributes[PDO::ATTR_PERSISTENT] = true;
+                       }
                }
 
                try {
-                       $conn = new PDO(
-                               "sqlite:$path",
-                               '',
-                               '',
-                               [ PDO::ATTR_PERSISTENT => (bool)( $this->flags & self::DBO_PERSISTENT ) ]
-                       );
-                       // Set error codes only, don't raise exceptions
-                       $conn->setAttribute( PDO::ATTR_ERRMODE, PDO::ERRMODE_SILENT );
+                       $this->conn = new PDO( "sqlite:$path", null, null, $attributes );
                } catch ( PDOException $e ) {
-                       $error = $e->getMessage();
-                       $this->connLogger->error(
-                               "Error connecting to {db_server}: {error}",
-                               $this->getLogContext( [ 'method' => __METHOD__, 'error' => $error ] )
-                       );
-                       throw new DBConnectionError( $this, $error );
+                       throw $this->newExceptionAfterConnectError( $e->getMessage() );
                }
 
-               $this->conn = $conn;
                $this->currentDomain = new DatabaseDomain( $dbName, null, $tablePrefix );
 
                try {
                        $flags = self::QUERY_IGNORE_DBO_TRX | self::QUERY_NO_RETRY;
                        // Enforce LIKE to be case sensitive, just like MySQL
                        $this->query( 'PRAGMA case_sensitive_like = 1', __METHOD__, $flags );
-                       // Apply an optimizations or requirements regarding fsync() usage
+                       // Apply optimizations or requirements regarding fsync() usage
                        $sync = $this->connectionVariables['synchronous'] ?? null;
                        if ( in_array( $sync, [ 'EXTRA', 'FULL', 'NORMAL', 'OFF' ], true ) ) {
                                $this->query( "PRAGMA synchronous = $sync", __METHOD__, $flags );
                        }
                } catch ( Exception $e ) {
-                       // Connection was not fully initialized and is not safe for use
-                       $this->conn = false;
-                       throw $e;
+                       throw $this->newExceptionAfterConnectError( $e->getMessage() );
                }
        }
 
index ef7f1e2..7e54221 100644 (file)
@@ -87,7 +87,7 @@ interface IDatabase {
        /** @var int Combine list with OR clauses */
        const LIST_OR = 4;
 
-       /** @var int Enable debug logging */
+       /** @var int Enable debug logging of all SQL queries */
        const DBO_DEBUG = 1;
        /** @var int Disable query buffering (only one result set can be iterated at a time) */
        const DBO_NOBUFFER = 2;
@@ -238,14 +238,6 @@ interface IDatabase {
         */
        public function setLazyMasterHandle( IDatabase $conn );
 
-       /**
-        * Returns true if this database does an implicit sort when doing GROUP BY
-        *
-        * @return bool
-        * @deprecated Since 1.30; only use grouped or aggregated fields in the SELECT
-        */
-       public function implicitGroupby();
-
        /**
         * Returns true if this database does an implicit order by when the column has an index
         * For example: SELECT page_title FROM page LIMIT 1
@@ -260,15 +252,6 @@ interface IDatabase {
         */
        public function lastQuery();
 
-       /**
-        * Returns true if the connection may have been used for write queries.
-        * Should return true if unsure.
-        *
-        * @return bool
-        * @deprecated Since 1.31; use lastDoneWrites()
-        */
-       public function doneWrites();
-
        /**
         * Returns the last time the connection may have been used for write queries.
         * Should return a timestamp if unsure.
@@ -336,13 +319,7 @@ interface IDatabase {
        /**
         * Set a flag for this connection
         *
-        * @param int $flag DBO_* constants from Defines.php:
-        *   - DBO_DEBUG: output some debug info (same as debug())
-        *   - DBO_NOBUFFER: don't buffer results (inverse of bufferResults())
-        *   - DBO_TRX: automatically start transactions
-        *   - DBO_DEFAULT: automatically sets DBO_TRX if not in command line mode
-        *       and removes it in command line mode
-        *   - DBO_PERSISTENT: use persistant database connection
+        * @param int $flag IDatabase::DBO_DEBUG, IDatabase::DBO_NOBUFFER, or IDatabase::DBO_TRX
         * @param string $remember IDatabase::REMEMBER_* constant [default: REMEMBER_NOTHING]
         */
        public function setFlag( $flag, $remember = self::REMEMBER_NOTHING );
@@ -350,13 +327,7 @@ interface IDatabase {
        /**
         * Clear a flag for this connection
         *
-        * @param int $flag DBO_* constants from Defines.php:
-        *   - DBO_DEBUG: output some debug info (same as debug())
-        *   - DBO_NOBUFFER: don't buffer results (inverse of bufferResults())
-        *   - DBO_TRX: automatically start transactions
-        *   - DBO_DEFAULT: automatically sets DBO_TRX if not in command line mode
-        *       and removes it in command line mode
-        *   - DBO_PERSISTENT: use persistant database connection
+        * @param int $flag IDatabase::DBO_DEBUG, IDatabase::DBO_NOBUFFER, or IDatabase::DBO_TRX
         * @param string $remember IDatabase::REMEMBER_* constant [default: REMEMBER_NOTHING]
         */
        public function clearFlag( $flag, $remember = self::REMEMBER_NOTHING );
@@ -372,11 +343,7 @@ interface IDatabase {
        /**
         * Returns a boolean whether the flag $flag is set for this connection
         *
-        * @param int $flag DBO_* constants from Defines.php:
-        *   - DBO_DEBUG: output some debug info (same as debug())
-        *   - DBO_NOBUFFER: don't buffer results (inverse of bufferResults())
-        *   - DBO_TRX: automatically start transactions
-        *   - DBO_PERSISTENT: use persistant database connection
+        * @param int $flag One of the class IDatabase::DBO_* constants
         * @return bool
         */
        public function getFlag( $flag );
@@ -390,14 +357,6 @@ interface IDatabase {
         */
        public function getDomainID();
 
-       /**
-        * Alias for getDomainID()
-        *
-        * @return string
-        * @deprecated 1.30
-        */
-       public function getWikiID();
-
        /**
         * Get the type of the DBMS, as it appears in $wgDBtype.
         *
index 812064a..4b6afe7 100644 (file)
@@ -31,9 +31,12 @@ use InvalidArgumentException;
  * @since 1.28
  */
 interface ILBFactory {
+       /** @var int Don't save DB positions at all */
        const SHUTDOWN_NO_CHRONPROT = 0; // don't save DB positions at all
-       const SHUTDOWN_CHRONPROT_ASYNC = 1; // save DB positions, but don't wait on remote DCs
-       const SHUTDOWN_CHRONPROT_SYNC = 2; // save DB positions, waiting on all DCs
+       /** @var int Save DB positions, but don't wait on remote DCs */
+       const SHUTDOWN_CHRONPROT_ASYNC = 1;
+       /** @var int Save DB positions, waiting on all DCs */
+       const SHUTDOWN_CHRONPROT_SYNC = 2;
 
        /**
         * Construct a manager of ILoadBalancer objects
@@ -140,6 +143,8 @@ interface ILBFactory {
        /**
         * Get cached (tracked) load balancers for all main database clusters
         *
+        * The default cluster name is ILoadBalancer::CLUSTER_MAIN_DEFAULT
+        *
         * @return ILoadBalancer[] Map of (cluster name => ILoadBalancer)
         * @since 1.29
         */
@@ -154,7 +159,8 @@ interface ILBFactory {
        public function getAllExternalLBs();
 
        /**
-        * Execute a function for each tracked load balancer
+        * Execute a function for each currently tracked (instantiated) load balancer
+        *
         * The callback is called with the load balancer as the first parameter,
         * and $params passed as the subsequent parameters.
         *
@@ -164,7 +170,8 @@ interface ILBFactory {
        public function forEachLB( $callback, array $params = [] );
 
        /**
-        * Prepare all tracked load balancers for shutdown
+        * Prepare all currently tracked (instantiated) load balancers for shutdown
+        *
         * @param int $mode One of the class SHUTDOWN_* constants
         * @param callable|null $workCallback Work to mask ChronologyProtector writes
         * @param int|null &$cpIndex Position key write counter for ChronologyProtector
index a85e1e5..4426654 100644 (file)
@@ -651,14 +651,6 @@ abstract class LBFactory implements ILBFactory {
                $this->indexAliases = $aliases;
        }
 
-       /**
-        * @param string $prefix
-        * @deprecated Since 1.33
-        */
-       public function setDomainPrefix( $prefix ) {
-               $this->setLocalDomainPrefix( $prefix );
-       }
-
        public function setLocalDomainPrefix( $prefix ) {
                $this->localDomain = new DatabaseDomain(
                        $this->localDomain->getDatabase(),
index 1ef1d09..98607ce 100644 (file)
@@ -177,11 +177,10 @@ class LoadBalancer implements ILoadBalancer {
                                $server['replica'] = true;
                        }
                        $this->servers[$i] = $server;
-                       $serverGroupLoads = [ self::GROUP_GENERIC => $server['load'] ];
-                       $serverGroupLoads += ( $server['groupLoads'] ?? [] );
-                       foreach ( $serverGroupLoads as $group => $ratio ) {
+                       foreach ( ( $server['groupLoads'] ?? [] ) as $group => $ratio ) {
                                $this->groupLoads[$group][$i] = $ratio;
                        }
+                       $this->groupLoads[self::GROUP_GENERIC][$i] = $server['load'];
                }
 
                $localDomain = isset( $params['localDomain'] )
@@ -656,7 +655,7 @@ class LoadBalancer implements ILoadBalancer {
                                $ok = true; // no applicable loads
                        }
                } finally {
-                       # Restore the old position, as this is not used for lag-protection but for throttling
+                       // Restore the old position; this is used for throttling, not lag-protection
                        $this->waitForPos = $oldPos;
                }
 
@@ -673,7 +672,7 @@ class LoadBalancer implements ILoadBalancer {
 
                        $ok = true;
                        for ( $i = 1; $i < $serverCount; $i++ ) {
-                               if ( $this->groupLoads[self::GROUP_GENERIC][$i] > 0 ) {
+                               if ( $this->serverHasLoadInAnyGroup( $i ) ) {
                                        $start = microtime( true );
                                        $ok = $this->doWait( $i, true, $timeout ) && $ok;
                                        $timeout -= intval( microtime( true ) - $start );
@@ -683,13 +682,27 @@ class LoadBalancer implements ILoadBalancer {
                                }
                        }
                } finally {
-                       # Restore the old position, as this is not used for lag-protection but for throttling
+                       // Restore the old position; this is used for throttling, not lag-protection
                        $this->waitForPos = $oldPos;
                }
 
                return $ok;
        }
 
+       /**
+        * @param int $i Specific server index
+        * @return bool
+        */
+       private function serverHasLoadInAnyGroup( $i ) {
+               foreach ( $this->groupLoads as $loadsByIndex ) {
+                       if ( ( $loadsByIndex[$i] ?? 0 ) > 0 ) {
+                               return true;
+                       }
+               }
+
+               return false;
+       }
+
        /**
         * @param DBMasterPos|bool $pos
         */
@@ -1922,15 +1935,6 @@ class LoadBalancer implements ILoadBalancer {
                return $this->laggedReplicaMode;
        }
 
-       /**
-        * @return bool
-        * @since 1.27
-        * @deprecated Since 1.28; use laggedReplicaUsed()
-        */
-       public function laggedSlaveUsed() {
-               return $this->laggedReplicaUsed();
-       }
-
        public function getReadOnlyReason( $domain = false, IDatabase $conn = null ) {
                if ( $this->readOnlyReason !== false ) {
                        return $this->readOnlyReason;
@@ -2203,14 +2207,6 @@ class LoadBalancer implements ILoadBalancer {
                $this->indexAliases = $aliases;
        }
 
-       /**
-        * @param string $prefix
-        * @deprecated Since 1.33
-        */
-       public function setDomainPrefix( $prefix ) {
-               $this->setLocalDomainPrefix( $prefix );
-       }
-
        public function setLocalDomainPrefix( $prefix ) {
                // Find connections to explicit foreign domains still marked as in-use...
                $domainsInUse = [];
index e07b166..f7fc46f 100644 (file)
@@ -131,7 +131,7 @@ class BitmapHandler extends TransformationalImageHandler {
         * @param File $image File associated with this thumbnail
         * @param array $params Array with scaler params
         *
-        * @return MediaTransformError|bool Error object if error occurred, false (=no error) otherwise
+        * @return MediaTransformError|false Error object if error occurred, false (=no error) otherwise
         */
        protected function transformImageMagick( $image, $params ) {
                # use ImageMagick
@@ -275,7 +275,7 @@ class BitmapHandler extends TransformationalImageHandler {
         * @param File $image File associated with this thumbnail
         * @param array $params Array with scaler params
         *
-        * @return MediaTransformError Error|bool object if error occurred, false (=no error) otherwise
+        * @return MediaTransformError|false Error object if error occurred, false (=no error) otherwise
         */
        protected function transformImageMagickExt( $image, $params ) {
                global $wgSharpenReductionThreshold, $wgSharpenParameter, $wgMaxAnimatedGifArea,
@@ -368,7 +368,7 @@ class BitmapHandler extends TransformationalImageHandler {
         * @param File $image File associated with this thumbnail
         * @param array $params Array with scaler params
         *
-        * @return MediaTransformError Error|bool object if error occurred, false (=no error) otherwise
+        * @return MediaTransformError|false Error object if error occurred, false (=no error) otherwise
         */
        protected function transformCustom( $image, $params ) {
                # Use a custom convert command
index 9a9c0a6..2f5fa90 100644 (file)
@@ -53,7 +53,7 @@ class BmpHandler extends BitmapHandler {
         *
         * @param File|FSFile $image
         * @param string $filename
-        * @return array
+        * @return array|false
         */
        function getImageSize( $image, $filename ) {
                $f = fopen( $filename, 'rb' );
index 13a39ed..92fad52 100644 (file)
@@ -61,7 +61,7 @@ class DjVuImage {
 
        /**
         * Return data in the style of getimagesize()
-        * @return array|bool Array or false on failure
+        * @return array|false Array or false on failure
         */
        public function getImageSize() {
                $data = $this->getInfo();
index 1760eb8..fa9e1dc 100644 (file)
@@ -166,7 +166,7 @@ class ExifBitmapHandler extends BitmapHandler {
         *
         * @param File|FSFile $image
         * @param string $path
-        * @return array
+        * @return array|false
         */
        function getImageSize( $image, $path ) {
                $gis = parent::getImageSize( $image, $path );
index 7dd0491..3abc6b7 100644 (file)
@@ -103,7 +103,7 @@ abstract class MediaHandler {
         * @param File|FSFile $image The image object, or false if there isn't one.
         *   Warning, FSFile::getPropsFromPath might pass an FSFile instead of File (!)
         * @param string $path The filename
-        * @return array|bool Follow the format of PHP getimagesize() internal function.
+        * @return array|false Follow the format of PHP getimagesize() internal function.
         *   See https://www.php.net/getimagesize. MediaWiki will only ever use the
         *   first two array keys (the width and height), and the 'bits' associative
         *   key. All other array keys are ignored. Returning a 'bits' key is optional
index ac332b7..6ae6426 100644 (file)
 
 /**
  * @ingroup Media
+ * @deprecated since 1.34
  */
 class SVGMetadataExtractor {
-       static function getMetadata( $filename ) {
+       /**
+        * @param string $filename
+        * @return array
+        * @deprecated since 1.34, use SVGReader->getMetadata() directly
+        */
+       public static function getMetadata( $filename ) {
+               wfDeprecated( __METHOD__, '1.34' );
+
                $svg = new SVGReader( $filename );
 
                return $svg->getMetadata();
index bdda674..98c22a3 100644 (file)
@@ -381,7 +381,7 @@ class SvgHandler extends ImageHandler {
         * @param File|FSFile $file
         * @param string $path Unused
         * @param bool|array $metadata
-        * @return array
+        * @return array|false
         */
        function getImageSize( $file, $path, $metadata = false ) {
                if ( $metadata === false && $file instanceof File ) {
@@ -438,8 +438,10 @@ class SvgHandler extends ImageHandler {
         */
        public function getMetadata( $file, $filename ) {
                $metadata = [ 'version' => self::SVG_METADATA_VERSION ];
+
                try {
-                       $metadata += SVGMetadataExtractor::getMetadata( $filename );
+                       $svgReader = new SVGReader( $filename );
+                       $metadata += $svgReader->getMetadata();
                } catch ( Exception $e ) { // @todo SVG specific exceptions
                        // File not found, broken, etc.
                        $metadata['error'] = [
index e2cc1b2..5294530 100644 (file)
@@ -58,7 +58,7 @@ class XCFHandler extends BitmapHandler {
         *
         * @param File|FSFile $image
         * @param string $filename
-        * @return array
+        * @return array|false
         */
        function getImageSize( $image, $filename ) {
                $header = self::getXCFMetaData( $filename );
index 9226875..6cc63bf 100644 (file)
@@ -27,6 +27,7 @@ use Wikimedia\Rdbms\IDatabase;
 use Wikimedia\Rdbms\DBError;
 use Wikimedia\Rdbms\DBQueryError;
 use Wikimedia\Rdbms\DBConnectionError;
+use Wikimedia\Rdbms\IMaintainableDatabase;
 use Wikimedia\Rdbms\LoadBalancer;
 use Wikimedia\ScopedCallback;
 use Wikimedia\WaitConditionLoop;
@@ -165,7 +166,7 @@ class SqlBagOStuff extends MediumSpecificBagOStuff {
         * Get a connection to the specified database
         *
         * @param int $serverIndex
-        * @return Database
+        * @return IMaintainableDatabase
         * @throws MWException
         */
        protected function getDB( $serverIndex ) {
@@ -199,11 +200,11 @@ class SqlBagOStuff extends MediumSpecificBagOStuff {
                        $index = $this->replicaOnly ? DB_REPLICA : DB_MASTER;
                        if ( $lb->getServerType( $lb->getWriterIndex() ) !== 'sqlite' ) {
                                // Keep a separate connection to avoid contention and deadlocks
-                               $db = $lb->getConnection( $index, [], false, $lb::CONN_TRX_AUTOCOMMIT );
+                               $db = $lb->getConnectionRef( $index, [], false, $lb::CONN_TRX_AUTOCOMMIT );
                        } else {
                                // However, SQLite has the opposite behavior due to DB-level locking.
                                // Stock sqlite MediaWiki installs use a separate sqlite cache DB instead.
-                               $db = $lb->getConnection( $index );
+                               $db = $lb->getConnectionRef( $index );
                        }
                }
 
index 54036eb..9426dce 100644 (file)
@@ -33,7 +33,7 @@ abstract class AlphabeticPager extends IndexPager {
         *
         * @return string HTML
         */
-       function getNavigationBar() {
+       public function getNavigationBar() {
                if ( !$this->isNavigationBarShown() ) {
                        return '';
                }
index 9cfbfbf..45c310a 100644 (file)
@@ -30,7 +30,7 @@
  * @ingroup Pager
  */
 interface Pager {
-       function getNavigationBar();
+       public function getNavigationBar();
 
-       function getBody();
+       public function getBody();
 }
index 5aa1a69..a7916c5 100644 (file)
@@ -1187,39 +1187,44 @@ class CoreParserFunctions {
         */
        public static function pageid( $parser, $title = null ) {
                $t = Title::newFromText( $title );
-               if ( is_null( $t ) ) {
+               if ( !$t ) {
                        return '';
+               } elseif ( !$t->canExist() || $t->isExternal() ) {
+                       return 0; // e.g. special page or interwiki link
                }
-               // Use title from parser to have correct pageid after edit
+
+               $parserOutput = $parser->getOutput();
+
                if ( $t->equals( $parser->getTitle() ) ) {
-                       $t = $parser->getTitle();
-                       return $t->getArticleID();
-               }
+                       // Revision is for the same title that is currently being parsed.
+                       // Use the title from Parser in case a new page ID was injected into it.
+                       $parserOutput->setFlag( 'vary-page-id' );
+                       $id = $parser->getTitle()->getArticleID();
+                       if ( $id ) {
+                               $parserOutput->setSpeculativePageIdUsed( $id );
+                       }
 
-               // These can't have ids
-               if ( !$t->canExist() || $t->isExternal() ) {
-                       return 0;
+                       return $id;
                }
 
-               // Check the link cache, maybe something already looked it up.
+               // Check the link cache for the title
                $linkCache = MediaWikiServices::getInstance()->getLinkCache();
                $pdbk = $t->getPrefixedDBkey();
                $id = $linkCache->getGoodLinkID( $pdbk );
-               if ( $id != 0 ) {
-                       $parser->mOutput->addLink( $t, $id );
-                       return $id;
-               }
-               if ( $linkCache->isBadLink( $pdbk ) ) {
-                       $parser->mOutput->addLink( $t, 0 );
+               if ( $id != 0 || $linkCache->isBadLink( $pdbk ) ) {
+                       $parserOutput->addLink( $t, $id );
+
                        return $id;
                }
 
                // We need to load it from the DB, so mark expensive
                if ( $parser->incrementExpensiveFunctionCount() ) {
                        $id = $t->getArticleID();
-                       $parser->mOutput->addLink( $t, $id );
+                       $parserOutput->addLink( $t, $id );
+
                        return $id;
                }
+
                return null;
        }
 
index 6600f8f..e5bf94a 100644 (file)
@@ -2783,15 +2783,16 @@ class Parser {
                                $value = wfEscapeWikiText( $subjPage->getPrefixedURL() );
                                break;
                        case 'pageid': // requested in T25427
-                               $pageid = $this->getTitle()->getArticleID();
-                               if ( $pageid == 0 ) {
-                                       # 0 means the page doesn't exist in the database,
-                                       # which means the user is previewing a new page.
-                                       # The vary-revision flag must be set, because the magic word
-                                       # will have a different value once the page is saved.
-                                       $this->setOutputFlag( 'vary-revision', '{{PAGEID}} on new page' );
+                               # Inform the edit saving system that getting the canonical output
+                               # after page insertion requires a parse that used that exact page ID
+                               $this->setOutputFlag( 'vary-page-id', '{{PAGEID}} used' );
+                               $value = $this->mTitle->getArticleID();
+                               if ( !$value ) {
+                                       $value = $this->mOptions->getSpeculativePageId();
+                                       if ( $value ) {
+                                               $this->mOutput->setSpeculativePageIdUsed( $value );
+                                       }
                                }
-                               $value = $pageid ?: null;
                                break;
                        case 'revisionid':
                                if (
@@ -2810,7 +2811,7 @@ class Parser {
                                        }
                                } else {
                                        # Inform the edit saving system that getting the canonical output after
-                                       # revision insertion requires another parse using the actual revision ID
+                                       # revision insertion requires a parse that used that exact revision ID
                                        $this->setOutputFlag( 'vary-revision-id', '{{REVISIONID}} used' );
                                        $value = $this->getRevisionId();
                                        if ( $value === 0 ) {
@@ -5934,19 +5935,21 @@ class Parser {
         * @since 1.23 (public since 1.23)
         */
        public function getRevisionObject() {
-               if ( !is_null( $this->mRevisionObject ) ) {
+               if ( $this->mRevisionObject ) {
                        return $this->mRevisionObject;
                }
 
                // NOTE: try to get the RevisionObject even if mRevisionId is null.
-               // This is useful when parsing revision that has not yet been saved.
+               // This is useful when parsing revision that has not yet been saved.
                // However, if we get back a saved revision even though we are in
                // preview mode, we'll have to ignore it, see below.
                // NOTE: This callback may be used to inject an OLD revision that was
                // already loaded, so "current" is a bit of a misnomer. We can't just
                // skip it if mRevisionId is set.
                $rev = call_user_func(
-                       $this->mOptions->getCurrentRevisionCallback(), $this->getTitle(), $this
+                       $this->mOptions->getCurrentRevisionCallback(),
+                       $this->getTitle(),
+                       $this
                );
 
                if ( $this->mRevisionId === null && $rev && $rev->getId() ) {
index 709f159..5a159fb 100644 (file)
@@ -62,6 +62,7 @@ class ParserOptions {
        private static $lazyOptions = [
                'dateformat' => [ __CLASS__, 'initDateFormat' ],
                'speculativeRevId' => [ __CLASS__, 'initSpeculativeRevId' ],
+               'speculativePageId' => [ __CLASS__, 'initSpeculativePageId' ],
        ];
 
        /**
@@ -117,11 +118,6 @@ class ParserOptions {
         */
        private $mExtraKey = '';
 
-       /**
-        * @name Option accessors
-        * @{
-        */
-
        /**
         * Fetch an option and track that is was accessed
         * @since 1.30
@@ -856,11 +852,25 @@ class ParserOptions {
                return $this->getOption( 'speculativeRevId' );
        }
 
+       /**
+        * A guess for {{PAGEID}}, calculated using the callback provided via
+        * setSpeculativeRevPageCallback(). For consistency, the value will be calculated upon the
+        * first call of this method, and re-used for subsequent calls.
+        *
+        * If no callback was defined via setSpeculativePageIdCallback(), this method will return false.
+        *
+        * @since 1.34
+        * @return int|false
+        */
+       public function getSpeculativePageId() {
+               return $this->getOption( 'speculativePageId' );
+       }
+
        /**
         * Callback registered with ParserOptions::$lazyOptions, triggered by getSpeculativeRevId().
         *
         * @param ParserOptions $popt
-        * @return bool|false
+        * @return int|false
         */
        private static function initSpeculativeRevId( ParserOptions $popt ) {
                $cb = $popt->getOption( 'speculativeRevIdCallback' );
@@ -871,27 +881,40 @@ class ParserOptions {
        }
 
        /**
-        * Callback to generate a guess for {{REVISIONID}}
-        * @since 1.28
-        * @deprecated since 1.32, use getSpeculativeRevId() instead!
-        * @return callable|null
+        * Callback registered with ParserOptions::$lazyOptions, triggered by getSpeculativePageId().
+        *
+        * @param ParserOptions $popt
+        * @return int|false
         */
-       public function getSpeculativeRevIdCallback() {
-               return $this->getOption( 'speculativeRevIdCallback' );
+       private static function initSpeculativePageId( ParserOptions $popt ) {
+               $cb = $popt->getOption( 'speculativePageIdCallback' );
+               $id = $cb ? $cb() : null;
+
+               // returning null would result in this being re-called every access
+               return $id ?? false;
        }
 
        /**
         * Callback to generate a guess for {{REVISIONID}}
-        * @since 1.28
-        * @param callable|null $x New value (null is no change)
+        * @param callable|null $x New value
         * @return callable|null Old value
+        * @since 1.28
         */
        public function setSpeculativeRevIdCallback( $x ) {
                $this->setOption( 'speculativeRevId', null ); // reset
-               return $this->setOptionLegacy( 'speculativeRevIdCallback', $x );
+               return $this->setOption( 'speculativeRevIdCallback', $x );
        }
 
-       /**@}*/
+       /**
+        * Callback to generate a guess for {{PAGEID}}
+        * @param callable|null $x New value
+        * @return callable|null Old value
+        * @since 1.34
+        */
+       public function setSpeculativePageIdCallback( $x ) {
+               $this->setOption( 'speculativePageId', null ); // reset
+               return $this->setOption( 'speculativePageIdCallback', $x );
+       }
 
        /**
         * Timestamp used for {{CURRENTDAY}} etc.
@@ -1102,6 +1125,8 @@ class ParserOptions {
                                'templateCallback' => [ Parser::class, 'statelessFetchTemplate' ],
                                'speculativeRevIdCallback' => null,
                                'speculativeRevId' => null,
+                               'speculativePageIdCallback' => null,
+                               'speculativePageId' => null,
                        ];
 
                        Hooks::run( 'ParserOptionsRegister', [
index 23e5911..1922f7d 100644 (file)
@@ -210,9 +210,17 @@ class ParserOutput extends CacheTime {
         */
        private $mFlags = [];
 
+       /** @var string[] */
+       private static $speculativeFields = [
+               'speculativePageIdUsed',
+               'mSpeculativeRevId',
+               'revisionTimestampUsed'
+       ];
+
        /** @var int|null Assumed rev ID for {{REVISIONID}} if no revision is set */
        private $mSpeculativeRevId;
-
+       /** @var int|null Assumed page ID for {{PAGEID}} if no revision is set */
+       private $speculativePageIdUsed;
        /** @var int|null Assumed rev timestamp for {{REVISIONTIMESTAMP}} if no revision is set */
        private $revisionTimestampUsed;
 
@@ -451,6 +459,22 @@ class ParserOutput extends CacheTime {
                return $this->mSpeculativeRevId;
        }
 
+       /**
+        * @param int $id
+        * @since 1.34
+        */
+       public function setSpeculativePageIdUsed( $id ) {
+               $this->speculativePageIdUsed = $id;
+       }
+
+       /**
+        * @return int|null
+        * @since 1.34
+        */
+       public function getSpeculativePageIdUsed() {
+               return $this->speculativePageIdUsed;
+       }
+
        /**
         * @param string $timestamp TS_MW timestamp
         * @since 1.34
@@ -1302,9 +1326,18 @@ class ParserOutput extends CacheTime {
        }
 
        public function __sleep() {
-               return array_diff(
-                       array_keys( get_object_vars( $this ) ),
-                       [ 'mParseStartTime' ]
+               return array_filter( array_keys( get_object_vars( $this ) ),
+                       function ( $field ) {
+                               if ( $field === 'mParseStartTime' ) {
+                                       return false;
+                               } elseif ( strpos( $field, "\0" ) !== false ) {
+                                       // Unserializing unknown private fields in HHVM causes
+                                       // member variables with nulls in their names (T229366)
+                                       return false;
+                               } else {
+                                       return true;
+                               }
+                       }
                );
        }
 
@@ -1320,18 +1353,13 @@ class ParserOutput extends CacheTime {
                $this->mWarnings = self::mergeMap( $this->mWarnings, $source->mWarnings ); // don't use getter
                $this->mTimestamp = $this->useMaxValue( $this->mTimestamp, $source->getTimestamp() );
 
-               if ( $this->mSpeculativeRevId && $source->mSpeculativeRevId
-                       && $this->mSpeculativeRevId !== $source->mSpeculativeRevId
-               ) {
-                       wfLogWarning(
-                               'Inconsistent speculative revision ID encountered while merging parser output!'
-                       );
+               foreach ( self::$speculativeFields as $field ) {
+                       if ( $this->$field && $source->$field && $this->$field !== $source->$field ) {
+                               wfLogWarning( __METHOD__ . ": inconsistent '$field' properties!" );
+                       }
+                       $this->$field = $this->useMaxValue( $this->$field, $source->$field );
                }
 
-               $this->mSpeculativeRevId = $this->useMaxValue(
-                       $this->mSpeculativeRevId,
-                       $source->getSpeculativeRevIdUsed()
-               );
                $this->mParseStartTime = $this->useEachMinValue(
                        $this->mParseStartTime,
                        $source->mParseStartTime
index beed60b..001c975 100644 (file)
@@ -1295,7 +1295,7 @@ class DefaultPreferencesFactory implements PreferencesFactory {
                // Display the skin if the user has set it as a preference already before it was hidden.
                $currentUserSkin = $user->getOption( 'skin' );
                if ( isset( $allInstalledSkins[$currentUserSkin] )
-                       && $context->msg( "skinname-$useSkin" )->exists()
+                       && $context->msg( "skinname-$currentUserSkin" )->exists()
                ) {
                        $validSkinNames[$currentUserSkin] = $currentUserSkin;
                }
index fca06c9..d0f6729 100644 (file)
@@ -224,7 +224,7 @@ class MessageBlobStore implements LoggerAwareInterface {
                        }
                }
 
-               $json = FormatJson::encode( (object)$messages );
+               $json = FormatJson::encode( (object)$messages, false, FormatJson::UTF8_OK );
                // @codeCoverageIgnoreStart
                if ( $json === false ) {
                        $this->logger->warning( 'Failed to encode message blob for {module} ({lang})', [
index 7e623b5..9892b15 100644 (file)
@@ -36,66 +36,41 @@ use Wikimedia\WrappedString;
  *    https://www.mediawiki.org/wiki/ResourceLoader
  */
 class ResourceLoader implements LoggerAwareInterface {
-       /** @var int */
-       const CACHE_VERSION = 8;
+       /** @var Config $config */
+       protected $config;
+       /** @var MessageBlobStore */
+       protected $blobStore;
 
-       /** @var bool */
-       protected static $debugMode = null;
+       /** @var LoggerInterface */
+       private $logger;
 
-       /**
-        * Module name/ResourceLoaderModule object pairs
-        * @var array
-        */
+       /** @var ResourceLoaderModule[] Map of (module name => ResourceLoaderModule) */
        protected $modules = [];
-
-       /**
-        * Associative array mapping module name to info associative array
-        * @var array
-        */
+       /** @var array[] Map of (module name => associative info array) */
        protected $moduleInfos = [];
-
-       /** @var Config $config */
-       protected $config;
-
        /**
         * Associative array mapping framework ids to a list of names of test suite modules
         * like [ 'qunit' => [ 'mediawiki.tests.qunit.suites', 'ext.foo.tests', ... ], ... ]
         * @var array
         */
        protected $testModuleNames = [];
+       /** @var string[] List of module names that contain QUnit test suites */
+       protected $testSuiteModuleNames = [];
 
-       /**
-        * E.g. [ 'source-id' => 'http://.../load.php' ]
-        * @var array
-        */
+       /** @var array Map of (source => path); E.g. [ 'source-id' => 'http://.../load.php' ] */
        protected $sources = [];
-
-       /**
-        * Errors accumulated during current respond() call.
-        * @var array
-        */
+       /** @var array Errors accumulated during current respond() call */
        protected $errors = [];
-
-       /**
-        * List of extra HTTP response headers provided by loaded modules.
-        *
-        * Populated by makeModuleResponse().
-        *
-        * @var array
-        */
+       /** @var string[] Extra HTTP response headers from modules loaded in makeModuleResponse() */
        protected $extraHeaders = [];
 
-       /**
-        * @var MessageBlobStore
-        */
-       protected $blobStore;
+       /** @var bool */
+       protected static $debugMode = null;
 
-       /**
-        * @var LoggerInterface
-        */
-       private $logger;
+       /** @var int */
+       const CACHE_VERSION = 8;
 
-       /** @var string JavaScript / CSS pragma to disable minification. **/
+       /** @var string JavaScript / CSS pragma to disable minification. * */
        const FILTER_NOMIN = '/*@nomin*/';
 
        /**
@@ -374,6 +349,7 @@ class ResourceLoader implements LoggerAwareInterface {
 
        /**
         * @internal For use by ServiceWiring only
+        * @codeCoverageIgnore
         */
        public function registerTestModules() {
                global $IP;
@@ -384,39 +360,37 @@ class ResourceLoader implements LoggerAwareInterface {
                                . 'Edit your <code>LocalSettings.php</code> to enable it.' );
                }
 
-               $testModules = [
-                       'qunit' => [],
-               ];
+               // This has a 'qunit' key for compat with the below hook.
+               $testModulesMeta = [ 'qunit' => [] ];
 
                // Get test suites from extensions
                // Avoid PHP 7.1 warning from passing $this by reference
                $rl = $this;
-               Hooks::run( 'ResourceLoaderTestModules', [ &$testModules, &$rl ] );
+               Hooks::run( 'ResourceLoaderTestModules', [ &$testModulesMeta, &$rl ] );
                $extRegistry = ExtensionRegistry::getInstance();
                // In case of conflict, the deprecated hook has precedence.
-               $testModules['qunit'] += $extRegistry->getAttribute( 'QUnitTestModules' );
+               $testModules = $testModulesMeta['qunit'] + $extRegistry->getAttribute( 'QUnitTestModules' );
 
-               // Add the QUnit testrunner as implicit dependency to extension test suites.
-               foreach ( $testModules['qunit'] as &$module ) {
-                       // Shuck any single-module dependency as an array
+               $testSuiteModuleNames = [];
+               foreach ( $testModules as $name => &$module ) {
+                       // Turn any single-module dependency into an array
                        if ( isset( $module['dependencies'] ) && is_string( $module['dependencies'] ) ) {
                                $module['dependencies'] = [ $module['dependencies'] ];
                        }
 
+                       // Ensure the testrunner loads before any test suites
                        $module['dependencies'][] = 'test.mediawiki.qunit.testrunner';
-               }
 
-               // Get core test suites
-               $testModules['qunit'] =
-                       ( include "$IP/tests/qunit/QUnitTestResources.php" ) + $testModules['qunit'];
+                       // Keep track of the test suites to load on SpecialJavaScriptTest
+                       $testSuiteModuleNames[] = $name;
+               }
 
-               foreach ( $testModules as $id => $names ) {
-                       // Register test modules
-                       $this->register( $testModules[$id] );
+               // Core test suites (their names have further precedence).
+               $testModules = ( include "$IP/tests/qunit/QUnitTestResources.php" ) + $testModules;
+               $testSuiteModuleNames[] = 'test.mediawiki.qunit.suites';
 
-                       // Keep track of their names so that they can be loaded together
-                       $this->testModuleNames[$id] = array_keys( $testModules[$id] );
-               }
+               $this->register( $testModules );
+               $this->testSuiteModuleNames = $testSuiteModuleNames;
        }
 
        /**
@@ -470,25 +444,14 @@ class ResourceLoader implements LoggerAwareInterface {
        }
 
        /**
-        * Get a list of test module names for one (or all) frameworks.
+        * Get a list of module names with QUnit test suites.
         *
-        * If the given framework id is unknkown, or if the in-object variable is not an array,
-        * then it will return an empty array.
-        *
-        * @param string $framework Get only the test module names for one
-        *   particular framework (optional)
+        * @internal For use by SpecialJavaScriptTest only
         * @return array
+        * @codeCoverageIgnore
         */
-       public function getTestModuleNames( $framework = 'all' ) {
-               if ( $framework == 'all' ) {
-                       return $this->testModuleNames;
-               } elseif ( isset( $this->testModuleNames[$framework] )
-                       && is_array( $this->testModuleNames[$framework] )
-               ) {
-                       return $this->testModuleNames[$framework];
-               } else {
-                       return [];
-               }
+       public function getTestSuiteModuleNames() {
+               return $this->testSuiteModuleNames;
        }
 
        /**
@@ -738,6 +701,8 @@ class ResourceLoader implements LoggerAwareInterface {
                        if ( $this->tryRespondFromFileCache( $fileCache, $context, $etag ) ) {
                                return; // output handled
                        }
+               } else {
+                       $fileCache = null;
                }
 
                // Generate a response
@@ -752,15 +717,17 @@ class ResourceLoader implements LoggerAwareInterface {
                        }
                }
 
-               // Save response to file cache unless there are errors
-               if ( isset( $fileCache ) && !$this->errors && $missing === [] ) {
-                       // Cache single modules and images...and other requests if there are enough hits
-                       if ( ResourceFileCache::useFileCache( $context ) ) {
-                               if ( $fileCache->isCacheWorthy() ) {
-                                       $fileCache->saveText( $response );
-                               } else {
-                                       $fileCache->incrMissesRecent( $context->getRequest() );
-                               }
+               // Consider saving the response to file cache (unless there are errors).
+               if ( $fileCache &&
+                       !$this->errors &&
+                       $missing === [] &&
+                       ResourceFileCache::useFileCache( $context )
+               ) {
+                       if ( $fileCache->isCacheWorthy() ) {
+                               // There were enough hits, save the response to the cache
+                               $fileCache->saveText( $response );
+                       } else {
+                               $fileCache->incrMissesRecent( $context->getRequest() );
                        }
                }
 
index 1f06ede..94e8a3e 100644 (file)
@@ -66,8 +66,8 @@ class ResourceLoaderContext implements MessageLocalizer {
                $this->request = $request;
                $this->logger = $resourceLoader->getLogger();
 
-               // Future developers: Use WebRequest::getRawVal() instead of getVal().
-               // The getVal() method performs slow Language+UTF logic. (f303bb9360)
+               // Optimisation: Use WebRequest::getRawVal() instead of getVal(). We don't
+               // need the slow Language+UTF logic meant for user input here. (f303bb9360)
 
                // List of modules
                $modules = $request->getRawVal( 'modules' );
@@ -95,19 +95,6 @@ class ResourceLoaderContext implements MessageLocalizer {
                }
        }
 
-       /**
-        * Reverse the process done by ResourceLoader::makePackedModulesString().
-        *
-        * @deprecated since 1.33 Use ResourceLoader::expandModuleNames instead.
-        * @param string $modules Packed module name list
-        * @return array Array of module names
-        * @codeCoverageIgnore
-        */
-       public static function expandModuleNames( $modules ) {
-               wfDeprecated( __METHOD__, '1.33' );
-               return ResourceLoader::expandModuleNames( $modules );
-       }
-
        /**
         * Return a dummy ResourceLoaderContext object suitable for passing into
         * things that don't "really" need a context.
@@ -140,8 +127,10 @@ class ResourceLoaderContext implements MessageLocalizer {
         * @deprecated since 1.34 Use ResourceLoaderModule::getConfig instead
         * inside module methods. Use ResourceLoader::getConfig elsewhere.
         * @return Config
+        * @codeCoverageIgnore
         */
        public function getConfig() {
+               wfDeprecated( __METHOD__, '1.34' );
                return $this->getResourceLoader()->getConfig();
        }
 
index 9003951..6497543 100644 (file)
@@ -437,7 +437,8 @@ class ResourceLoaderImage {
 
                        file_put_contents( $tempFilenameSvg, $svg );
 
-                       $metadata = SVGMetadataExtractor::getMetadata( $tempFilenameSvg );
+                       $svgReader = new SVGReader( $tempFilenameSvg );
+                       $metadata = $svgReader->getMetadata();
                        if ( !isset( $metadata['width'] ) || !isset( $metadata['height'] ) ) {
                                unlink( $tempFilenameSvg );
                                return false;
index c376fa7..ed2d09c 100644 (file)
@@ -26,6 +26,7 @@ use MediaWiki\MediaWikiServices;
 use Psr\Log\LoggerAwareInterface;
 use Psr\Log\LoggerInterface;
 use Psr\Log\NullLogger;
+use Wikimedia\AtEase\AtEase;
 use Wikimedia\RelPath;
 use Wikimedia\ScopedCallback;
 
@@ -33,65 +34,63 @@ use Wikimedia\ScopedCallback;
  * Abstraction for ResourceLoader modules, with name registration and maxage functionality.
  */
 abstract class ResourceLoaderModule implements LoggerAwareInterface {
-       # Type of resource
-       const TYPE_SCRIPTS = 'scripts';
-       const TYPE_STYLES = 'styles';
-       const TYPE_COMBINED = 'combined';
-
-       # Desired load type
-       // Module only has styles (loaded via <style> or <link rel=stylesheet>)
-       const LOAD_STYLES = 'styles';
-       // Module may have other resources (loaded via mw.loader from a script)
-       const LOAD_GENERAL = 'general';
-
-       # sitewide core module like a skin file or jQuery component
-       const ORIGIN_CORE_SITEWIDE = 1;
-
-       # per-user module generated by the software
-       const ORIGIN_CORE_INDIVIDUAL = 2;
-
-       # sitewide module generated from user-editable files, like MediaWiki:Common.js, or
-       # modules accessible to multiple users, such as those generated by the Gadgets extension.
-       const ORIGIN_USER_SITEWIDE = 3;
-
-       # per-user module generated from user-editable files, like User:Me/vector.js
-       const ORIGIN_USER_INDIVIDUAL = 4;
-
-       # an access constant; make sure this is kept as the largest number in this group
-       const ORIGIN_ALL = 10;
+       /** @var Config */
+       protected $config;
+       /** @var LoggerInterface */
+       protected $logger;
 
-       # script and style modules form a hierarchy of trustworthiness, with core modules like
-       # skins and jQuery as most trustworthy, and user scripts as least trustworthy.  We can
-       # limit the types of scripts and styles we allow to load on, say, sensitive special
-       # pages like Special:UserLogin and Special:Preferences
+       /**
+        * Script and style modules form a hierarchy of trustworthiness, with core modules
+        * like skins and jQuery as most trustworthy, and user scripts as least trustworthy. We can
+        * limit the types of scripts and styles we allow to load on, say, sensitive special
+        * pages like Special:UserLogin and Special:Preferences
+        * @var int
+        */
        protected $origin = self::ORIGIN_CORE_SITEWIDE;
 
+       /** @var string|null Module name */
        protected $name = null;
+       /** @var string[] What client platforms the module targets (e.g. desktop, mobile) */
        protected $targets = [ 'desktop' ];
 
-       // In-object cache for file dependencies
+       /** @var array Map of (variant => indirect file dependencies) */
        protected $fileDeps = [];
-       // In-object cache for message blob (keyed by language)
+       /** @var array Map of (language => in-object cache for message blob) */
        protected $msgBlobs = [];
-       // In-object cache for version hash
+       /** @var array Map of (context hash => cached module version hash) */
        protected $versionHash = [];
-       // In-object cache for module content
+       /** @var array Map of (context hash => cached module content) */
        protected $contents = [];
 
-       /**
-        * @var Config
-        */
-       protected $config;
-
-       /**
-        * @var array|bool
-        */
+       /** @var string|bool Deprecation string or true if deprecated; false otherwise */
        protected $deprecated = false;
 
+       /** @var string Scripts only */
+       const TYPE_SCRIPTS = 'scripts';
+       /** @var string Styles only */
+       const TYPE_STYLES = 'styles';
+       /** @var string Scripts and styles */
+       const TYPE_COMBINED = 'combined';
+
+       /** @var string Module only has styles (loaded via <style> or <link rel=stylesheet>) */
+       const LOAD_STYLES = 'styles';
+       /** @var string Module may have other resources (loaded via mw.loader from a script) */
+       const LOAD_GENERAL = 'general';
+
+       /** @var int Sitewide core module like a skin file or jQuery component */
+       const ORIGIN_CORE_SITEWIDE = 1;
+       /** @var int Per-user module generated by the software */
+       const ORIGIN_CORE_INDIVIDUAL = 2;
        /**
-        * @var LoggerInterface
+        * Sitewide module generated from user-editable files, like MediaWiki:Common.js,
+        * or modules accessible to multiple users, such as those generated by the Gadgets extension.
+        * @var int
         */
-       protected $logger;
+       const ORIGIN_USER_SITEWIDE = 3;
+       /** @var int Per-user module generated from user-editable files, like User:Me/vector.js */
+       const ORIGIN_USER_INDIVIDUAL = 4;
+       /** @var int An access constant; make sure this is kept as the largest number in this group */
+       const ORIGIN_ALL = 10;
 
        /**
         * Get this module's name. This is set when the module is registered
@@ -210,7 +209,6 @@ abstract class ResourceLoaderModule implements LoggerAwareInterface {
        /**
         * @since 1.27
         * @param LoggerInterface $logger
-        * @return null
         */
        public function setLogger( LoggerInterface $logger ) {
                $this->logger = $logger;
@@ -950,12 +948,12 @@ abstract class ResourceLoaderModule implements LoggerAwareInterface {
                                $parser = self::javaScriptParser();
                                $err = null;
                                try {
-                                       Wikimedia\suppressWarnings();
+                                       AtEase::suppressWarnings();
                                        $parser->parse( $contents, $fileName, 1 );
                                } catch ( Exception $e ) {
                                        $err = $e;
                                } finally {
-                                       Wikimedia\restoreWarnings();
+                                       AtEase::restoreWarnings();
                                }
                                if ( $err ) {
                                        // Send the error to the browser console client-side.
@@ -989,9 +987,9 @@ abstract class ResourceLoaderModule implements LoggerAwareInterface {
         * @return int UNIX timestamp
         */
        protected static function safeFilemtime( $filePath ) {
-               Wikimedia\suppressWarnings();
+               AtEase::suppressWarnings();
                $mtime = filemtime( $filePath ) ?: 1;
-               Wikimedia\restoreWarnings();
+               AtEase::restoreWarnings();
                return $mtime;
        }
 
index 7880f6f..a982de2 100644 (file)
@@ -105,7 +105,6 @@ class ResourceLoaderStartUpModule extends ResourceLoaderModule {
                        'wgCaseSensitiveNamespaces' => $caseSensitiveNamespaces,
                        'wgLegalTitleChars' => Title::convertByteClassToUnicodeClass( Title::legalChars() ),
                        'wgIllegalFileChars' => Title::convertByteClassToUnicodeClass( $illegalFileChars ),
-                       'wgResourceLoaderStorageEnabled' => $conf->get( 'ResourceLoaderStorageEnabled' ),
                        'wgForeignUploadTargets' => $conf->get( 'ForeignUploadTargets' ),
                        'wgEnableUploads' => $conf->get( 'EnableUploads' ),
                        'wgCommentByteLimit' => null,
@@ -284,8 +283,9 @@ class ResourceLoaderStartUpModule extends ResourceLoaderModule {
                        }
 
                        if ( $versionHash !== '' && strlen( $versionHash ) !== 7 ) {
-                               $this->getLogger()->warning(
-                                       "Module '{module}' produced an invalid version hash: '{version}'.",
+                               $e = new RuntimeException( "Badly formatted module version hash" );
+                               $resourceLoader->outputErrorAndLog( $e,
+                                               "Module '{module}' produced an invalid version hash: '{version}'.",
                                        [
                                                'module' => $name,
                                                'version' => $versionHash,
@@ -340,16 +340,6 @@ class ResourceLoaderStartUpModule extends ResourceLoaderModule {
                return $out;
        }
 
-       /**
-        * @private For internal use by SpecialJavaScriptTest
-        * @since 1.32
-        * @return array
-        * @codeCoverageIgnore
-        */
-       public function getBaseModulesInternal() {
-               return $this->getBaseModules();
-       }
-
        /**
         * Base modules implicitly available to all modules.
         *
@@ -422,6 +412,14 @@ class ResourceLoaderStartUpModule extends ResourceLoaderModule {
                        '$VARS.maxQueryLength' => ResourceLoader::encodeJsonForScript(
                                $conf->get( 'ResourceLoaderMaxQueryLength' )
                        ),
+                       // The client-side module cache can be disabled by site configuration.
+                       // It is also always disabled in debug mode.
+                       '$VARS.storeEnabled' => ResourceLoader::encodeJsonForScript(
+                               $conf->get( 'ResourceLoaderStorageEnabled' ) && !$context->getDebug()
+                       ),
+                       '$VARS.wgLegacyJavaScriptGlobals' => ResourceLoader::encodeJsonForScript(
+                               $conf->get( 'LegacyJavaScriptGlobals' )
+                       ),
                        '$VARS.storeKey' => ResourceLoader::encodeJsonForScript( $this->getStoreKey() ),
                        '$VARS.storeVary' => ResourceLoader::encodeJsonForScript( $this->getStoreVary( $context ) ),
                ];
@@ -442,9 +440,6 @@ class ResourceLoaderStartUpModule extends ResourceLoaderModule {
 
                // Perform string replacements for startup.js
                $pairs = [
-                       '$VARS.wgLegacyJavaScriptGlobals' => ResourceLoader::encodeJsonForScript(
-                               $conf->get( 'LegacyJavaScriptGlobals' )
-                       ),
                        '$VARS.configuration' => ResourceLoader::encodeJsonForScript(
                                $this->getConfigSettings( $context )
                        ),
index b104073..a2501c4 100644 (file)
@@ -412,6 +412,7 @@ class ResourceLoaderWikiModule extends ResourceLoaderModule {
                return $titleInfo;
        }
 
+       /** @return array */
        protected static function fetchTitleInfo( IDatabase $db, array $pages, $fname = __METHOD__ ) {
                $titleInfo = [];
                $batch = new LinkBatch;
diff --git a/includes/search/BaseSearchResultSet.php b/includes/search/BaseSearchResultSet.php
new file mode 100644 (file)
index 0000000..d8aed0e
--- /dev/null
@@ -0,0 +1,76 @@
+<?php
+
+/**
+ * BaseSearchResultSet is the base class that must be extended by SearchEngine
+ * search result set implementations.
+ *
+ * This base class is meant to hold B/C behaviors and to be useful it must never:
+ * - be used as type hints (ISearchResultSet must be used for this)
+ * - implement a constructor
+ * - declare utility methods
+ *
+ * @ingroup Search
+ */
+abstract class BaseSearchResultSet implements ISearchResultSet {
+
+       /**
+        * @var ArrayIterator|null Iterator supporting BC iteration methods
+        */
+       private $bcIterator;
+
+       /**
+        * Fetches next search result, or false.
+        * @deprecated since 1.32; Use self::extractResults() or foreach
+        * @return SearchResult|false
+        */
+       public function next() {
+               wfDeprecated( __METHOD__, '1.32' );
+               $it = $this->bcIterator();
+               $searchResult = $it->current();
+               $it->next();
+               return $searchResult ?? false;
+       }
+
+       /**
+        * Rewind result set back to beginning
+        * @deprecated since 1.32; Use self::extractResults() or foreach
+        */
+       public function rewind() {
+               wfDeprecated( __METHOD__, '1.32' );
+               $this->bcIterator()->rewind();
+       }
+
+       private function bcIterator() {
+               if ( $this->bcIterator === null ) {
+                       $this->bcIterator = 'RECURSION';
+                       $this->bcIterator = $this->getIterator();
+               } elseif ( $this->bcIterator === 'RECURSION' ) {
+                       // Either next/rewind or extractResults must be implemented.  This
+                       // class was potentially instantiated directly. It should be
+                       // abstract with abstract methods to enforce this but that's a
+                       // breaking change...
+                       wfDeprecated( static::class . ' without implementing extractResults', '1.32' );
+                       $this->bcIterator = new ArrayIterator( [] );
+               }
+               return $this->bcIterator;
+       }
+
+       /**
+        * Fetch an array of regular expression fragments for matching
+        * the search terms as parsed by this engine in a text extract.
+        * STUB
+        *
+        * @return string[]
+        * @deprecated since 1.34 (use SqlSearchResult)
+        */
+       public function termMatches() {
+               return [];
+       }
+
+       /**
+        * Frees the result set, if applicable.
+        * @deprecated noop since 1.34
+        */
+       public function free() {
+       }
+}
index 1b30f5a..5faa445 100644 (file)
@@ -1,6 +1,11 @@
 <?php
 
 /**
+ * A set of SearchEngine results.
+ * Must not be directly implemented by extension, please extend BaseSearchResultSet instead.
+ * This interface must only be used for type hinting.
+ *
+ * @see BaseSearchResultSet
  * @ingroup Search
  */
 interface ISearchResultSet extends \Countable, \IteratorAggregate {
@@ -19,7 +24,7 @@ interface ISearchResultSet extends \Countable, \IteratorAggregate {
        /**
         * @return int
         */
-       function numRows();
+       public function numRows();
 
        /**
         * Some search modes return a total hit count for the query
@@ -31,7 +36,7 @@ interface ISearchResultSet extends \Countable, \IteratorAggregate {
         *
         * @return int|null
         */
-       function getTotalHits();
+       public function getTotalHits();
 
        /**
         * Some search modes will run an alternative query that it thinks gives
@@ -40,19 +45,19 @@ interface ISearchResultSet extends \Countable, \IteratorAggregate {
         *
         * @return bool
         */
-       function hasRewrittenQuery();
+       public function hasRewrittenQuery();
 
        /**
         * @return string|null The search the query was internally rewritten to,
         *  or null when the result of the original query was returned.
         */
-       function getQueryAfterRewrite();
+       public function getQueryAfterRewrite();
 
        /**
         * @return string|null Same as self::getQueryAfterRewrite(), but in HTML
         *  and with changes highlighted. Null when the query was not rewritten.
         */
-       function getQueryAfterRewriteSnippet();
+       public function getQueryAfterRewriteSnippet();
 
        /**
         * Some search modes return a suggested alternate term if there are
@@ -60,17 +65,17 @@ interface ISearchResultSet extends \Countable, \IteratorAggregate {
         *
         * @return bool
         */
-       function hasSuggestion();
+       public function hasSuggestion();
 
        /**
         * @return string|null Suggested query, null if none
         */
-       function getSuggestionQuery();
+       public function getSuggestionQuery();
 
        /**
         * @return string HTML highlighted suggested query, '' if none
         */
-       function getSuggestionSnippet();
+       public function getSuggestionSnippet();
 
        /**
         * Return a result set of hits on other (multiple) wikis associated with this one
@@ -78,7 +83,7 @@ interface ISearchResultSet extends \Countable, \IteratorAggregate {
         * @param int $type
         * @return ISearchResultSet[]
         */
-       function getInterwikiResults( $type = self::SECONDARY_RESULTS );
+       public function getInterwikiResults( $type = self::SECONDARY_RESULTS );
 
        /**
         * Check if there are results on other wikis
@@ -86,7 +91,7 @@ interface ISearchResultSet extends \Countable, \IteratorAggregate {
         * @param int $type
         * @return bool
         */
-       function hasInterwikiResults( $type = self::SECONDARY_RESULTS );
+       public function hasInterwikiResults( $type = self::SECONDARY_RESULTS );
 
        /**
         * Did the search contain search syntax?  If so, Special:Search won't offer
index 0c5d4da..a057aa8 100644 (file)
@@ -61,7 +61,7 @@ abstract class SearchDatabase extends SearchEngine {
         * Perform a full text search query and return a result set.
         *
         * @param string $term Raw search term
-        * @return SqlSearchResultSet
+        * @return SqlSearchResultSet|null
         */
        abstract protected function doSearchTextInDB( $term );
 
@@ -77,7 +77,7 @@ abstract class SearchDatabase extends SearchEngine {
         * Perform a title-only search query and return a result set.
         *
         * @param string $term Raw search term
-        * @return SqlSearchResultSet
+        * @return SqlSearchResultSet|null
         */
        abstract protected function doSearchTitleInDB( $term );
 
index 32b0f06..87a7861 100644 (file)
@@ -433,10 +433,13 @@ abstract class SearchEngine {
        /**
         * Find snippet highlight settings for all users
         * @return array Contextlines, contextchars
+        * @deprecated in 1.34 use the SearchHighlighter constants directly
+        * @see SearchHighlighter::DEFAULT_CONTEXT_CHARS
+        * @see SearchHighlighter::DEFAULT_CONTEXT_LINES
         */
        public static function userHighlightPrefs() {
-               $contextlines = 2; // Hardcode this. Old defaults sucked. :)
-               $contextchars = 75; // same as above.... :P
+               $contextlines = SearchHighlighter::DEFAULT_CONTEXT_LINES;
+               $contextchars = SearchHighlighter::DEFAULT_CONTEXT_CHARS;
                return [ $contextlines, $contextchars ];
        }
 
@@ -486,6 +489,7 @@ abstract class SearchEngine {
         * @param Title $t Title we're indexing
         * @param Content|null $c Content of the page to index
         * @return string
+        * @deprecated since 1.34 use Content::getTextForSearchIndex directly
         */
        public function getTextFromContent( Title $t, Content $c = null ) {
                return $c ? $c->getTextForSearchIndex() : '';
@@ -497,6 +501,7 @@ abstract class SearchEngine {
         * rather silly handling, it should return true here instead.
         *
         * @return bool
+        * @deprecated since 1.34 no longer needed since getTextFromContent is being deprecated
         */
        public function textAlreadyUpdatedForIndex() {
                return false;
@@ -803,6 +808,10 @@ abstract class SearchEngine {
                        $setAugmentors[$name] = new PerRowAugmentor( $row );
                }
 
+               /**
+                * @var string $name
+                * @var ResultSetAugmentor $augmentor
+                */
                foreach ( $setAugmentors as $name => $augmentor ) {
                        $data = $augmentor->augmentAll( $resultSet );
                        if ( $data ) {
index 6c01f79..2579942 100644 (file)
@@ -29,6 +29,9 @@ use MediaWiki\MediaWikiServices;
  * @ingroup Search
  */
 class SearchHighlighter {
+       const DEFAULT_CONTEXT_LINES = 2;
+       const DEFAULT_CONTEXT_CHARS = 75;
+
        protected $mCleanWikitext = true;
 
        /**
@@ -50,7 +53,12 @@ class SearchHighlighter {
         * @param int $contextchars
         * @return string
         */
-       public function highlightText( $text, $terms, $contextlines, $contextchars ) {
+       public function highlightText(
+               $text,
+               $terms,
+               $contextlines = self::DEFAULT_CONTEXT_LINES,
+               $contextchars = self::DEFAULT_CONTEXT_CHARS
+       ) {
                global $wgSearchHighlightBoundaries;
 
                if ( $text == '' ) {
@@ -507,7 +515,12 @@ class SearchHighlighter {
         * @param int $contextchars
         * @return string
         */
-       public function highlightSimple( $text, $terms, $contextlines, $contextchars ) {
+       public function highlightSimple(
+               $text,
+               $terms,
+               $contextlines = self::DEFAULT_CONTEXT_LINES,
+               $contextchars = self::DEFAULT_CONTEXT_CHARS
+       ) {
                $lines = explode( "\n", $text );
 
                $terms = implode( '|', $terms );
@@ -557,7 +570,11 @@ class SearchHighlighter {
         * @param int $contextchars Average number of characters per line
         * @return string
         */
-       public function highlightNone( $text, $contextlines, $contextchars ) {
+       public function highlightNone(
+               $text,
+               $contextlines = self::DEFAULT_CONTEXT_LINES,
+               $contextchars = self::DEFAULT_CONTEXT_CHARS
+       ) {
                $match = [];
                $text = ltrim( $text ) . "\n"; // make sure the preg_match may find the last line
                $text = str_replace( "\n\n", "\n", $text ); // remove empty lines
index 6a23bb3..3c0675f 100644 (file)
@@ -33,7 +33,7 @@ class SearchMssql extends SearchDatabase {
         * Perform a full text search query and return a result set.
         *
         * @param string $term Raw search term
-        * @return SqlSearchResultSet
+        * @return SqlSearchResultSet|null
         */
        protected function doSearchTextInDB( $term ) {
                $dbr = $this->lb->getConnectionRef( DB_REPLICA );
@@ -46,7 +46,7 @@ class SearchMssql extends SearchDatabase {
         * Perform a title-only search query and return a result set.
         *
         * @param string $term Raw search term
-        * @return SqlSearchResultSet
+        * @return SqlSearchResultSet|null
         */
        protected function doSearchTitleInDB( $term ) {
                $dbr = $this->lb->getConnectionRef( DB_REPLICA );
index 4a6b93b..ff21367 100644 (file)
@@ -163,7 +163,7 @@ class SearchMySQL extends SearchDatabase {
         * Perform a full text search query and return a result set.
         *
         * @param string $term Raw search term
-        * @return SqlSearchResultSet
+        * @return SqlSearchResultSet|null
         */
        protected function doSearchTextInDB( $term ) {
                return $this->searchInternal( $term, true );
@@ -173,7 +173,7 @@ class SearchMySQL extends SearchDatabase {
         * Perform a title-only search query and return a result set.
         *
         * @param string $term Raw search term
-        * @return SqlSearchResultSet
+        * @return SqlSearchResultSet|null
         */
        protected function doSearchTitleInDB( $term ) {
                return $this->searchInternal( $term, false );
index 7240e81..d0869bc 100644 (file)
@@ -64,11 +64,11 @@ class SearchOracle extends SearchDatabase {
         * Perform a full text search query and return a result set.
         *
         * @param string $term Raw search term
-        * @return SqlSearchResultSet
+        * @return SqlSearchResultSet|null
         */
        protected function doSearchTextInDB( $term ) {
                if ( $term == '' ) {
-                       return new SqlSearchResultSet( false, '' );
+                       return null;
                }
 
                $dbr = $this->lb->getConnectionRef( DB_REPLICA );
@@ -80,11 +80,11 @@ class SearchOracle extends SearchDatabase {
         * Perform a title-only search query and return a result set.
         *
         * @param string $term Raw search term
-        * @return SqlSearchResultSet
+        * @return SqlSearchResultSet|null
         */
        protected function doSearchTitleInDB( $term ) {
                if ( $term == '' ) {
-                       return new SqlSearchResultSet( false, '' );
+                       return null;
                }
 
                $dbr = $this->lb->getConnectionRef( DB_REPLICA );
index a862e17..b924b29 100644 (file)
@@ -104,7 +104,7 @@ class SearchResult {
         *
         * @return bool
         */
-       function isBrokenTitle() {
+       public function isBrokenTitle() {
                return is_null( $this->mTitle );
        }
 
@@ -113,14 +113,14 @@ class SearchResult {
         *
         * @return bool
         */
-       function isMissingRevision() {
+       public function isMissingRevision() {
                return !$this->mRevision && !$this->mImage;
        }
 
        /**
         * @return Title
         */
-       function getTitle() {
+       public function getTitle() {
                return $this->mTitle;
        }
 
@@ -128,7 +128,7 @@ class SearchResult {
         * Get the file for this page, if one exists
         * @return File|null
         */
-       function getFile() {
+       public function getFile() {
                return $this->mImage;
        }
 
@@ -138,8 +138,8 @@ class SearchResult {
        protected function initText() {
                if ( !isset( $this->mText ) ) {
                        if ( $this->mRevision != null ) {
-                               $this->mText = $this->searchEngine->getTextFromContent(
-                                               $this->mTitle, $this->mRevision->getContent() );
+                               $content = $this->mRevision->getContent();
+                               $this->mText = $content !== null ? $content->getTextForSearchIndex() : '';
                        } else { // TODO: can we fetch raw wikitext for commons images?
                                $this->mText = '';
                        }
@@ -150,35 +150,35 @@ class SearchResult {
         * @param string[] $terms Terms to highlight (this parameter is deprecated and ignored)
         * @return string Highlighted text snippet, null (and not '') if not supported
         */
-       function getTextSnippet( $terms = [] ) {
+       public function getTextSnippet( $terms = [] ) {
                return '';
        }
 
        /**
         * @return string Highlighted title, '' if not supported
         */
-       function getTitleSnippet() {
+       public function getTitleSnippet() {
                return '';
        }
 
        /**
         * @return string Highlighted redirect name (redirect to this page), '' if none or not supported
         */
-       function getRedirectSnippet() {
+       public function getRedirectSnippet() {
                return '';
        }
 
        /**
         * @return Title|null Title object for the redirect to this page, null if none or not supported
         */
-       function getRedirectTitle() {
+       public function getRedirectTitle() {
                return null;
        }
 
        /**
         * @return string Highlighted relevant section name, null if none or not supported
         */
-       function getSectionSnippet() {
+       public function getSectionSnippet() {
                return '';
        }
 
@@ -186,7 +186,7 @@ class SearchResult {
         * @return Title|null Title object (pagename+fragment) for the section,
         *  null if none or not supported
         */
-       function getSectionTitle() {
+       public function getSectionTitle() {
                return null;
        }
 
@@ -200,7 +200,7 @@ class SearchResult {
        /**
         * @return string Timestamp
         */
-       function getTimestamp() {
+       public function getTimestamp() {
                if ( $this->mRevision ) {
                        return $this->mRevision->getTimestamp();
                } elseif ( $this->mImage ) {
@@ -212,7 +212,7 @@ class SearchResult {
        /**
         * @return int Number of words
         */
-       function getWordCount() {
+       public function getWordCount() {
                $this->initText();
                return str_word_count( $this->mText );
        }
@@ -220,7 +220,7 @@ class SearchResult {
        /**
         * @return int Size in bytes
         */
-       function getByteSize() {
+       public function getByteSize() {
                $this->initText();
                return strlen( $this->mText );
        }
@@ -228,14 +228,14 @@ class SearchResult {
        /**
         * @return string Interwiki prefix of the title (return iw even if title is broken)
         */
-       function getInterwikiPrefix() {
+       public function getInterwikiPrefix() {
                return '';
        }
 
        /**
         * @return string Interwiki namespace of the title (since we likely can't resolve it locally)
         */
-       function getInterwikiNamespaceText() {
+       public function getInterwikiNamespaceText() {
                return '';
        }
 
@@ -243,7 +243,7 @@ class SearchResult {
         * Did this match file contents (eg: PDF/DJVU)?
         * @return bool
         */
-       function isFileMatch() {
+       public function isFileMatch() {
                return false;
        }
 
index 84f8bcf..73e33a2 100644 (file)
@@ -24,7 +24,8 @@
 /**
  * @ingroup Search
  */
-class SearchResultSet implements ISearchResultSet {
+class SearchResultSet extends BaseSearchResultSet {
+       use SearchResultSetTrait;
 
        protected $containedSyntax = false;
 
@@ -42,25 +43,11 @@ class SearchResultSet implements ISearchResultSet {
         */
        protected $results;
 
-       /**
-        * Set of result's extra data, indexed per result id
-        * and then per data item name.
-        * The structure is:
-        * PAGE_ID => [ augmentor name => data, ... ]
-        * @var array[]
-        */
-       protected $extraData = [];
-
        /**
         * @var boolean True when there are more pages of search results available.
         */
        private $hasMoreResults;
 
-       /**
-        * @var ArrayIterator|null Iterator supporting BC iteration methods
-        */
-       private $bcIterator;
-
        /**
         * @param bool $containedSyntax True when query is not requesting a simple
         *  term match
@@ -78,19 +65,7 @@ class SearchResultSet implements ISearchResultSet {
                $this->hasMoreResults = $hasMoreResults;
        }
 
-       /**
-        * Fetch an array of regular expression fragments for matching
-        * the search terms as parsed by this engine in a text extract.
-        * STUB
-        *
-        * @return string[]
-        * @deprecated since 1.34 (use SqlSearchResult)
-        */
-       function termMatches() {
-               return [];
-       }
-
-       function numRows() {
+       public function numRows() {
                return $this->count();
        }
 
@@ -108,7 +83,7 @@ class SearchResultSet implements ISearchResultSet {
         *
         * @return int
         */
-       function getTotalHits() {
+       public function getTotalHits() {
                return null;
        }
 
@@ -119,7 +94,7 @@ class SearchResultSet implements ISearchResultSet {
         *
         * @return bool
         */
-       function hasRewrittenQuery() {
+       public function hasRewrittenQuery() {
                return false;
        }
 
@@ -127,7 +102,7 @@ class SearchResultSet implements ISearchResultSet {
         * @return string|null The search the query was internally rewritten to,
         *  or null when the result of the original query was returned.
         */
-       function getQueryAfterRewrite() {
+       public function getQueryAfterRewrite() {
                return null;
        }
 
@@ -135,7 +110,7 @@ class SearchResultSet implements ISearchResultSet {
         * @return string|null Same as self::getQueryAfterRewrite(), but in HTML
         *  and with changes highlighted. Null when the query was not rewritten.
         */
-       function getQueryAfterRewriteSnippet() {
+       public function getQueryAfterRewriteSnippet() {
                return null;
        }
 
@@ -145,21 +120,21 @@ class SearchResultSet implements ISearchResultSet {
         *
         * @return bool
         */
-       function hasSuggestion() {
+       public function hasSuggestion() {
                return false;
        }
 
        /**
         * @return string|null Suggested query, null if none
         */
-       function getSuggestionQuery() {
+       public function getSuggestionQuery() {
                return null;
        }
 
        /**
         * @return string HTML highlighted suggested query, '' if none
         */
-       function getSuggestionSnippet() {
+       public function getSuggestionSnippet() {
                return '';
        }
 
@@ -169,7 +144,7 @@ class SearchResultSet implements ISearchResultSet {
         * @param int $type
         * @return ISearchResultSet[]
         */
-       function getInterwikiResults( $type = self::SECONDARY_RESULTS ) {
+       public function getInterwikiResults( $type = self::SECONDARY_RESULTS ) {
                return null;
        }
 
@@ -179,54 +154,10 @@ class SearchResultSet implements ISearchResultSet {
         * @param int $type
         * @return bool
         */
-       function hasInterwikiResults( $type = self::SECONDARY_RESULTS ) {
+       public function hasInterwikiResults( $type = self::SECONDARY_RESULTS ) {
                return false;
        }
 
-       /**
-        * Fetches next search result, or false.
-        * @deprecated since 1.32; Use self::extractResults() or foreach
-        * @return SearchResult|false
-        */
-       public function next() {
-               wfDeprecated( __METHOD__, '1.32' );
-               $it = $this->bcIterator();
-               $searchResult = $it->current();
-               $it->next();
-               return $searchResult ?? false;
-       }
-
-       /**
-        * Rewind result set back to beginning
-        * @deprecated since 1.32; Use self::extractResults() or foreach
-        */
-       public function rewind() {
-               wfDeprecated( __METHOD__, '1.32' );
-               $this->bcIterator()->rewind();
-       }
-
-       private function bcIterator() {
-               if ( $this->bcIterator === null ) {
-                       $this->bcIterator = 'RECURSION';
-                       $this->bcIterator = $this->getIterator();
-               } elseif ( $this->bcIterator === 'RECURSION' ) {
-                       // Either next/rewind or extractResults must be implemented.  This
-                       // class was potentially instantiated directly. It should be
-                       // abstract with abstract methods to enforce this but that's a
-                       // breaking change...
-                       wfDeprecated( static::class . ' without implementing extractResults', '1.32' );
-                       $this->bcIterator = new ArrayIterator( [] );
-               }
-               return $this->bcIterator;
-       }
-
-       /**
-        * Frees the result set, if applicable.
-        * @deprecated noop since 1.34
-        */
-       function free() {
-       }
-
        /**
         * Did the search contain search syntax?  If so, Special:Search won't offer
         * the user a link to a create a page named by the search string because the
@@ -304,43 +235,4 @@ class SearchResultSet implements ISearchResultSet {
                }
                return $this->titles;
        }
-
-       /**
-        * Sets augmented data for result set.
-        * @param string $name Extra data item name
-        * @param array[] $data Extra data as PAGEID => data
-        */
-       public function setAugmentedData( $name, $data ) {
-               foreach ( $data as $id => $resultData ) {
-                       $this->extraData[$id][$name] = $resultData;
-               }
-       }
-
-       /**
-        * Returns extra data for specific result and store it in SearchResult object.
-        * @param SearchResult $result
-        */
-       public function augmentResult( SearchResult $result ) {
-               $id = $result->getTitle()->getArticleID();
-               if ( $id === -1 ) {
-                       return;
-               }
-               $result->setExtensionData( function () use ( $id ) {
-                       return $this->extraData[$id] ?? [];
-               } );
-       }
-
-       /**
-        * @return int|null The offset the current page starts at. Typically
-        *  this should be null to allow the UI to decide on its own, but in
-        *  special cases like interleaved AB tests specifying explicitly is
-        *  necessary.
-        */
-       public function getOffset() {
-               return null;
-       }
-
-       final public function getIterator() {
-               return new ArrayIterator( $this->extractResults() );
-       }
 }
diff --git a/includes/search/SearchResultSetTrait.php b/includes/search/SearchResultSetTrait.php
new file mode 100644 (file)
index 0000000..f36a7b5
--- /dev/null
@@ -0,0 +1,59 @@
+<?php
+
+/**
+ * Trait useful for SearchResultSet implementations.
+ * It holds the functions that are rarely needed to be overridden.
+ *
+ * This trait can be used directly by extensions providing a SearchEngine.
+ *
+ * @ingroup Search
+ */
+trait SearchResultSetTrait {
+       /**
+        * Set of result's extra data, indexed per result id
+        * and then per data item name.
+        * The structure is:
+        * PAGE_ID => [ augmentor name => data, ... ]
+        * @var array[]
+        */
+       private $extraData = [];
+
+       /**
+        * Sets augmented data for result set.
+        * @param string $name Extra data item name
+        * @param array[] $data Extra data as PAGEID => data
+        */
+       public function setAugmentedData( $name, $data ) {
+               foreach ( $data as $id => $resultData ) {
+                       $this->extraData[$id][$name] = $resultData;
+               }
+       }
+
+       /**
+        * Returns extra data for specific result and store it in SearchResult object.
+        * @param SearchResult $result
+        */
+       public function augmentResult( SearchResult $result ) {
+               $id = $result->getTitle()->getArticleID();
+               if ( $id === -1 ) {
+                       return;
+               }
+               $result->setExtensionData( function () use ( $id ) {
+                       return $this->extraData[$id] ?? [];
+               } );
+       }
+
+       /**
+        * @return int|null The offset the current page starts at. Typically
+        *  this should be null to allow the UI to decide on its own, but in
+        *  special cases like interleaved AB tests specifying explicitly is
+        *  necessary.
+        */
+       public function getOffset() {
+               return null;
+       }
+
+       final public function getIterator() {
+               return new ArrayIterator( $this->extractResults() );
+       }
+}
index dedcdff..9375ef2 100644 (file)
@@ -165,7 +165,7 @@ class SearchSqlite extends SearchDatabase {
         * Perform a full text search query and return a result set.
         *
         * @param string $term Raw search term
-        * @return SqlSearchResultSet
+        * @return SqlSearchResultSet|null
         */
        protected function doSearchTextInDB( $term ) {
                return $this->searchInternal( $term, true );
@@ -175,7 +175,7 @@ class SearchSqlite extends SearchDatabase {
         * Perform a title-only search query and return a result set.
         *
         * @param string $term Raw search term
-        * @return SqlSearchResultSet
+        * @return SqlSearchResultSet|null
         */
        protected function doSearchTitleInDB( $term ) {
                return $this->searchInternal( $term, false );
index 25e87e7..9804e44 100644 (file)
@@ -51,18 +51,15 @@ class SqlSearchResult extends SearchResult {
                global $wgAdvancedSearchHighlighting;
                $this->initText();
 
-               // TODO: make highliter take a content object. Make ContentHandler a factory for SearchHighliter.
-               list( $contextlines, $contextchars ) = $this->searchEngine->userHighlightPrefs();
-
                $h = new SearchHighlighter();
                if ( count( $this->terms ) > 0 ) {
                        if ( $wgAdvancedSearchHighlighting ) {
-                               return $h->highlightText( $this->mText, $this->terms, $contextlines, $contextchars );
+                               return $h->highlightText( $this->mText, $this->terms );
                        } else {
-                               return $h->highlightSimple( $this->mText, $this->terms, $contextlines, $contextchars );
+                               return $h->highlightSimple( $this->mText, $this->terms );
                        }
                } else {
-                       return $h->highlightNone( $this->mText, $contextlines, $contextchars );
+                       return $h->highlightNone( $this->mText );
                }
        }
 
index 918c761..bbad648 100644 (file)
@@ -275,7 +275,7 @@ abstract class Skin extends ContextSource {
 
                // Check, if the page can hold some kind of content, otherwise do nothing
                $title = $this->getRelevantTitle();
-               if ( $title->canExist() ) {
+               if ( $title->canExist() && $title->canHaveTalkPage() ) {
                        if ( $title->isTalkPage() ) {
                                $titles[] = $title->getSubjectPage();
                        } else {
index eb179bf..6cc6e4e 100644 (file)
@@ -21,6 +21,7 @@
  * @ingroup SpecialPage
  */
 
+use MediaWiki\Linker\LinkTarget;
 use MediaWiki\MediaWikiServices;
 use Wikimedia\Rdbms\IResultWrapper;
 use Wikimedia\Rdbms\IDatabase;
@@ -312,7 +313,7 @@ abstract class QueryPage extends SpecialPage {
                                $num = $res->numRows();
                                # Fetch results
                                $vals = [];
-                               foreach ( $res as $row ) {
+                               foreach ( $res as $i => $row ) {
                                        if ( isset( $row->value ) ) {
                                                if ( $this->usesTimestamps() ) {
                                                        $value = wfTimestamp( TS_UNIX,
@@ -321,7 +322,7 @@ abstract class QueryPage extends SpecialPage {
                                                        $value = intval( $row->value ); // T16414
                                                }
                                        } else {
-                                               $value = 0;
+                                               $value = $i;
                                        }
 
                                        $vals[] = [
@@ -375,6 +376,23 @@ abstract class QueryPage extends SpecialPage {
                return wfGetDB( DB_REPLICA, [ $this->getName(), 'QueryPage::recache', 'vslow' ] );
        }
 
+       /**
+        * Remove a cached result.
+        * Useful for interactive backlogs where the user can fix problems in-place.
+        * @param LinkTarget $title The page to remove.
+        * @since 1.34
+        */
+       public function delete( LinkTarget $title ) {
+               if ( $this->isCached() ) {
+                       $dbw = wfGetDB( DB_MASTER );
+                       $dbw->delete( 'querycache', [
+                               'qc_type' => $this->getName(),
+                               'qc_namespace' => $title->getNamespace(),
+                               'qc_title' => $title->getDBkey(),
+                       ], __METHOD__ );
+               }
+       }
+
        /**
         * Run the query and return the result
         * @param int|bool $limit Numerical limit or false for no limit
index ea4f18d..b2161db 100644 (file)
@@ -754,8 +754,6 @@ class SpecialBlock extends FormSpecialPage {
         * @return bool|array
         */
        public static function processForm( array $data, IContextSource $context ) {
-               global $wgBlockAllowsUTEdit, $wgHideUserContribLimit;
-
                $performer = $context->getUser();
                $enablePartialBlocks = $context->getConfig()->get( 'EnablePartialBlocks' );
                $isPartialBlock = $enablePartialBlocks &&
@@ -843,23 +841,33 @@ class SpecialBlock extends FormSpecialPage {
                        }
 
                        # Recheck params here...
+                       $hideUserContribLimit = $context->getConfig()->get( 'HideUserContribLimit' );
                        if ( $type != DatabaseBlock::TYPE_USER ) {
                                $data['HideUser'] = false; # IP users should not be hidden
                        } elseif ( !wfIsInfinity( $data['Expiry'] ) ) {
                                # Bad expiry.
                                return [ 'ipb_expiry_temp' ];
-                       } elseif ( $wgHideUserContribLimit !== false
-                               && $user->getEditCount() > $wgHideUserContribLimit
+                       } elseif ( $hideUserContribLimit !== false
+                               && $user->getEditCount() > $hideUserContribLimit
                        ) {
                                # Typically, the user should have a handful of edits.
                                # Disallow hiding users with many edits for performance.
                                return [ [ 'ipb_hide_invalid',
-                                       Message::numParam( $wgHideUserContribLimit ) ] ];
+                                       Message::numParam( $hideUserContribLimit ) ] ];
                        } elseif ( !$data['Confirm'] ) {
                                return [ 'ipb-confirmhideuser', 'ipb-confirmaction' ];
                        }
                }
 
+               $blockAllowsUTEdit = $context->getConfig()->get( 'BlockAllowsUTEdit' );
+               $userTalkEditAllowed = !$blockAllowsUTEdit || !$data['DisableUTEdit'];
+               if ( !$userTalkEditAllowed &&
+                       $isPartialBlock &&
+                       !in_array( NS_USER_TALK, explode( "\n", $data['NamespaceRestrictions'] ) )
+               ) {
+                       return [ 'ipb-prevent-user-talk-edit' ];
+               }
+
                # Create block object.
                $block = new DatabaseBlock();
                $block->setTarget( $target );
@@ -867,7 +875,7 @@ class SpecialBlock extends FormSpecialPage {
                $block->setReason( $data['Reason'][0] );
                $block->setExpiry( $expiryTime );
                $block->isCreateAccountBlocked( $data['CreateAccount'] );
-               $block->isUsertalkEditAllowed( !$wgBlockAllowsUTEdit || !$data['DisableUTEdit'] );
+               $block->isUsertalkEditAllowed( $userTalkEditAllowed );
                $block->isEmailBlocked( $data['DisableEmail'] );
                $block->isHardblock( $data['HardBlock'] );
                $block->isAutoblocking( $data['AutoBlock'] );
@@ -1130,8 +1138,6 @@ class SpecialBlock extends FormSpecialPage {
                if ( $performer->getBlock() ) {
                        if ( $target instanceof User && $target->getId() == $performer->getId() ) {
                                # User is trying to unblock themselves
-                               // @TODO Ensure that the block does not apply to the `unblockself`
-                               //       right.
                                if ( $performer->isAllowed( 'unblockself' ) ) {
                                        return true;
                                        # User blocked themselves and is now trying to reverse it
index 87c899f..184cdbb 100644 (file)
@@ -106,6 +106,13 @@ class SpecialChangeContentModel extends FormSpecialPage {
                        }
                        $fields['pagetitle']['readonly'] = true;
                        $fields += [
+                               'currentmodel' => [
+                                       'type' => 'text',
+                                       'name' => 'currentcontentmodel',
+                                       'default' => $this->title->getContentModel(),
+                                       'label-message' => 'changecontentmodel-current-label',
+                                       'readonly' => true
+                               ],
                                'model' => [
                                        'type' => 'select',
                                        'name' => 'model',
index 3e9676c..8c137aa 100644 (file)
@@ -103,7 +103,7 @@ class SpecialJavaScriptTest extends SpecialPage {
                $query['only'] = 'scripts';
                $startupContext = new ResourceLoaderContext( $rl, new FauxRequest( $query ) );
 
-               $modules = $rl->getTestModuleNames( 'qunit' );
+               $modules = $rl->getTestSuiteModuleNames();
 
                // Disable autostart because we load modules asynchronously. By default, QUnit would start
                // at domready when there are no tests loaded and also fire 'QUnit.done' which then instructs
@@ -170,7 +170,7 @@ JAVASCRIPT
                );
 
                // Use 'raw' because QUnit loads before ResourceLoader initialises (omit mw.loader.state call)
-               // Use 'test' to ensure OutputPage doesn't use the "async" attribute because QUnit must
+               // Use 'sync' to ensure OutputPage doesn't use the "async" attribute because QUnit must
                // load before qunit/export.
                $scripts = $out->makeResourceLoaderLink( 'jquery.qunit',
                        ResourceLoaderModule::TYPE_SCRIPTS,
index ecbbc25..161b41a 100644 (file)
@@ -150,10 +150,16 @@ class MovePageForm extends UnlistedSpecialPage {
                $out->addModules( 'mediawiki.misc-authed-ooui' );
                $this->addHelpLink( 'Help:Moving a page' );
 
-               $out->addWikiMsg( $this->getConfig()->get( 'FixDoubleRedirects' ) ?
-                       'movepagetext' :
-                       'movepagetext-noredirectfixer'
-               );
+               $handlerSupportsRedirects = ContentHandler::getForTitle( $this->oldTitle )
+                       ->supportsRedirects();
+
+               if ( $this->getConfig()->get( 'FixDoubleRedirects' ) ) {
+                       $out->addWikiMsg( 'movepagetext' );
+               } else {
+                       $out->addWikiMsg( $handlerSupportsRedirects ?
+                               'movepagetext-noredirectfixer' :
+                               'movepagetext-noredirectsupport' );
+               }
 
                if ( $this->oldTitle->getNamespace() == NS_USER && !$this->oldTitle->isSubpage() ) {
                        $out->wrapWikiMsg(
@@ -306,8 +312,6 @@ class MovePageForm extends UnlistedSpecialPage {
                        }
                }
 
-               $handler = ContentHandler::getForTitle( $this->oldTitle );
-
                $out->enableOOUI();
                $fields = [];
 
@@ -371,7 +375,7 @@ class MovePageForm extends UnlistedSpecialPage {
                }
 
                if ( $user->isAllowed( 'suppressredirect' ) ) {
-                       if ( $handler->supportsRedirects() ) {
+                       if ( $handlerSupportsRedirects ) {
                                $isChecked = $this->leaveRedirect;
                                $isDisabled = false;
                        } else {
index 9564d53..72afb03 100644 (file)
@@ -38,6 +38,21 @@ class SpecialUnlinkAccounts extends AuthManagerSpecialPage {
        public function execute( $subPage ) {
                $this->setHeaders();
                $this->loadAuth( $subPage );
+
+               if ( !$this->isActionAllowed( $this->authAction ) ) {
+                       if ( $this->authAction === AuthManager::ACTION_UNLINK ) {
+                               // Looks like there are no linked accounts to unlink
+                               $titleMessage = $this->msg( 'cannotunlink-no-provider-title' );
+                               $errorMessage = $this->msg( 'cannotunlink-no-provider' );
+                               throw new ErrorPageError( $titleMessage, $errorMessage );
+                       } else {
+                               // user probably back-button-navigated into an auth session that no longer exists
+                               // FIXME would be nice to show a message
+                               $this->getOutput()->redirect( $this->getPageTitle()->getFullURL( '', false, PROTO_HTTPS ) );
+                               return;
+                       }
+               }
+
                $this->outputHeader();
 
                $status = $this->trySubmit();
index f20afdb..9e94233 100644 (file)
@@ -18,8 +18,8 @@
                <td class="mw-changeslist-line-inner">{{!
                        }}{{# rev-deleted-event }}<span class="history-deleted">{{{ . }}}</span>{{/ rev-deleted-event }}{{!
                        }}{{{ articleLink }}}{{{ languageDirMark }}}{{{ logText }}}{{!
-                       }}<span class="mw-changeslist-separator"></span>{{!
-                       }}{{# charDifference }}{{{ . }}} <span class="mw-changeslist-separator"></span>{{/ charDifference }}{{!
+                       }} <span class="mw-changeslist-separator"></span> {{!
+                       }}{{# charDifference }}{{{ . }}} <span class="mw-changeslist-separator"></span> {{/ charDifference }}{{!
                        }}<span class="changedby">{{{ users }}}</span>{{!
                        }}{{ numberofWatchingusers }}{{!
                }}</td>
index 5b15e82..fb9dcf5 100644 (file)
@@ -704,6 +704,33 @@ abstract class UploadBase {
                return $warnings;
        }
 
+       /**
+        * Convert the warnings array returned by checkWarnings() to something that
+        * can be serialized. File objects will be converted to an associative array
+        * with the following keys:
+        *
+        *   - fileName: The name of the file
+        *   - timestamp: The upload timestamp
+        *
+        * @param mixed[] $warnings
+        * @return mixed[]
+        */
+       public static function makeWarningsSerializable( $warnings ) {
+               array_walk_recursive( $warnings, function ( &$param, $key ) {
+                       if ( $param instanceof File ) {
+                               $param = [
+                                       'fileName' => $param->getName(),
+                                       'timestamp' => $param->getTimestamp()
+                               ];
+                       } elseif ( is_object( $param ) ) {
+                               throw new InvalidArgumentException(
+                                       'UploadBase::makeWarningsSerializable: ' .
+                                       'Unexpected object of class ' . get_class( $param ) );
+                       }
+               } );
+               return $warnings;
+       }
+
        /**
         * Check whether the resulting filename is different from the desired one,
         * but ignore things like ucfirst() and spaces/underscore things
index fdac4a2..2261fcb 100644 (file)
@@ -148,72 +148,66 @@ class UserGroupMembership {
         * the function fails if there is a conflicting membership entry (same user and
         * group) already in the table.
         *
-        * @throws MWException
+        * @throws UnexpectedValueException
         * @param bool $allowUpdate Whether to perform "upsert" instead of INSERT
         * @param IDatabase|null $dbw If you have one available
         * @return bool Whether or not anything was inserted
         */
        public function insert( $allowUpdate = false, IDatabase $dbw = null ) {
-               if ( $dbw === null ) {
-                       $dbw = wfGetDB( DB_MASTER );
-               }
-
-               // Purge old, expired memberships from the DB
-               $hasExpiredRow = $dbw->selectField(
-                       'user_groups',
-                       '1',
-                       [ 'ug_expiry < ' . $dbw->addQuotes( $dbw->timestamp() ) ],
-                       __METHOD__
-               );
-               if ( $hasExpiredRow ) {
-                       JobQueueGroup::singleton()->lazyPush( new UserGroupExpiryJob() );
-               }
-
-               // Check that the values make sense
                if ( $this->group === null ) {
                        throw new UnexpectedValueException(
-                               'Don\'t try inserting an uninitialized UserGroupMembership object' );
+                               'Cannot insert an uninitialized UserGroupMembership instance'
+                       );
                } elseif ( $this->userId <= 0 ) {
                        throw new UnexpectedValueException(
                                'UserGroupMembership::insert() needs a positive user ID. ' .
-                               'Did you forget to add your User object to the database before calling addGroup()?' );
+                               'Perhaps addGroup() was called before the user was added to the database.'
+                       );
                }
 
+               $dbw = $dbw ?: wfGetDB( DB_MASTER );
                $row = $this->getDatabaseArray( $dbw );
+
+               $dbw->startAtomic( __METHOD__ );
                $dbw->insert( 'user_groups', $row, __METHOD__, [ 'IGNORE' ] );
                $affected = $dbw->affectedRows();
-
-               // Don't collide with expired user group memberships
-               // Do this after trying to insert, in order to avoid locking
                if ( !$affected ) {
-                       $conds = [
-                               'ug_user' => $row['ug_user'],
-                               'ug_group' => $row['ug_group'],
-                       ];
-                       // if we're unconditionally updating, check that the expiry is not already the
-                       // same as what we are trying to update it to; otherwise, only update if
-                       // the expiry date is in the past
+                       // Conflicting row already exists; it should be overriden if it is either expired
+                       // or if $allowUpdate is true and the current row is different than the loaded row.
+                       $conds = [ 'ug_user' => $row['ug_user'], 'ug_group' => $row['ug_group'] ];
                        if ( $allowUpdate ) {
-                               if ( $this->expiry ) {
-                                       $conds[] = 'ug_expiry IS NULL OR ug_expiry != ' .
-                                               $dbw->addQuotes( $dbw->timestamp( $this->expiry ) );
-                               } else {
-                                       $conds[] = 'ug_expiry IS NOT NULL';
-                               }
+                               // Update the current row if its expiry does not match that of the loaded row
+                               $conds[] = $this->expiry
+                                       ? 'ug_expiry IS NULL OR ug_expiry != ' .
+                                               $dbw->addQuotes( $dbw->timestamp( $this->expiry ) )
+                                       : 'ug_expiry IS NOT NULL';
                        } else {
+                               // Update the current row if it is expired
                                $conds[] = 'ug_expiry < ' . $dbw->addQuotes( $dbw->timestamp() );
                        }
+                       $dbw->update(
+                               'user_groups',
+                               [ 'ug_expiry' => $this->expiry ? $dbw->timestamp( $this->expiry ) : null ],
+                               $conds,
+                               __METHOD__
+                       );
+                       $affected = $dbw->affectedRows();
+               }
+               $dbw->endAtomic( __METHOD__ );
 
-                       $row = $dbw->selectRow( 'user_groups', $this::selectFields(), $conds, __METHOD__ );
-                       if ( $row ) {
-                               $dbw->update(
-                                       'user_groups',
-                                       [ 'ug_expiry' => $this->expiry ? $dbw->timestamp( $this->expiry ) : null ],
-                                       [ 'ug_user' => $row->ug_user, 'ug_group' => $row->ug_group ],
-                                       __METHOD__ );
-                               $affected = $dbw->affectedRows();
+               // Purge old, expired memberships from the DB
+               $fname = __METHOD__;
+               DeferredUpdates::addCallableUpdate( function () use ( $dbw, $fname ) {
+                       $hasExpiredRow = $dbw->selectField(
+                               'user_groups',
+                               '1',
+                               [ 'ug_expiry < ' . $dbw->addQuotes( $dbw->timestamp() ) ],
+                               $fname
+                       );
+                       if ( $hasExpiredRow ) {
+                               JobQueueGroup::singleton()->push( new UserGroupExpiryJob() );
                        }
-               }
+               } );
 
                return $affected > 0;
        }
index de0e4a6..a946653 100644 (file)
@@ -24,6 +24,7 @@ class SelectWithInputWidget extends \OOUI\Widget {
         *   - array $config['dropdowninput'] Configuration for the DropdownInputWidget
         *   - bool $config['or'] Configuration for whether the widget is dropdown AND input
         *       or dropdown OR input
+        *   - bool $config['required'] Configuration for whether the widget is a required input.
         */
        public function __construct( array $config = [] ) {
                // Configuration initialization
@@ -31,7 +32,8 @@ class SelectWithInputWidget extends \OOUI\Widget {
                        [
                                'textinput' => [],
                                'dropdowninput' => [],
-                               'or' => false
+                               'or' => false,
+                               'required' => false,
                        ],
                        $config
                );
@@ -41,6 +43,9 @@ class SelectWithInputWidget extends \OOUI\Widget {
                        $config['dropdowninput']['disabled'] = true;
                }
 
+               $config['textinput']['required'] = $config['or'] ? false : $config['required'];
+               $config['dropdowninput']['required'] = $config['required'];
+
                parent::__construct( $config );
 
                // Properties
@@ -63,6 +68,7 @@ class SelectWithInputWidget extends \OOUI\Widget {
                $config['dropdowninput'] = $this->config['dropdowninput'];
                $config['dropdowninput']['dropdown']['$overlay'] = true;
                $config['or'] = $this->config['or'];
+               $config['required'] = $this->config['required'];
                return parent::getConfig( $config );
        }
 }
index ee5df2e..3089341 100644 (file)
@@ -78,7 +78,8 @@
                        "Omar Ghrida",
                        "AHmed Khaled",
                        "البراء صالح",
-                       "Dyolf77 (WMF)"
+                       "Dyolf77 (WMF)",
+                       "HitomiAkane"
                ]
        },
        "tog-underline": "سطر تحت الوصلات:",
        "passwordreset-invalidemail": "عنوان بريد إلكتروني غير صالح",
        "passwordreset-nodata": "لا اسم مستخدم ولا عنوان بريد الإلكتروي تم توفيره",
        "changeemail": "تغيير أو إزالة عنوان البريد الإلكتروني",
-       "changeemail-header": "Ø¥Ù\83Ù\85اÙ\84 Ù\87ذا Ø§Ù\84Ù\86Ù\85Ù\88ذج Ù\84تغÙ\8aÙ\8aر Ø¹Ù\86Ù\88اÙ\86 Ø§Ù\84برÙ\8aد Ø§Ù\84Ø¥Ù\84Ù\83ترÙ\88Ù\86Ù\8a Ø§Ù\84خاص Ø¨Ù\83. Ø¥Ø°Ø§ Ù\83Ù\86ت ØªØ±ØºØ¨ Ù\81Ù\8a Ø¥Ø²Ø§Ù\84Ø© Ø¬Ù\85عÙ\8aØ© Ø£Ù\8a Ø¹Ù\86Ù\88اÙ\86 Ø§Ù\84برÙ\8aد Ø§Ù\84Ø¥Ù\84Ù\83ترÙ\88Ù\86Ù\8a Ù\85Ù\86 Ø­Ø³Ø§Ø¨Ù\83Ø\8c Ù\88ترÙ\83 Ø§Ù\84Ù\81راغ Ø¹Ù\86Ù\88اÙ\86 Ø§Ù\84برÙ\8aد Ø§Ù\84Ø¥Ù\84Ù\83ترÙ\88Ù\86Ù\8a Ø§Ù\84جدÙ\8aد Ø¹Ù\86د ØªÙ\82دÙ\8aÙ\85 Ø§Ù\84Ù\86Ù\85Ù\88ذج",
+       "changeemail-header": "Ø£Ù\83Ù\85Ù\84 Ù\87Ø°Ù\87 Ø§Ù\84استÙ\85ارة Ù\84تغÙ\8aÙ\8aر Ø¹Ù\86Ù\88اÙ\86 Ø§Ù\84برÙ\8aد Ø§Ù\84Ø¥Ù\84Ù\83ترÙ\88Ù\86Ù\8a Ø§Ù\84خاص Ø¨Ù\83. Ø¥Ø°Ø§ Ù\83Ù\86ت ØªØ±ØºØ¨ Ù\81Ù\8a Ø¥Ø²Ø§Ù\84Ø© Ø£Ù\8a Ø¹Ù\86Ù\88اÙ\86 Ø¨Ø±Ù\8aد Ø¥Ù\84Ù\83ترÙ\88Ù\86Ù\8a Ù\85Ù\82ترÙ\86 Ù\85ع Ø­Ø³Ø§Ø¨Ù\83Ø\8c Ø§ØªØ±Ù\83 Ø­Ù\82Ù\84 Ø¹Ù\86Ù\88اÙ\86 Ø§Ù\84برÙ\8aد Ø§Ù\84Ø¥Ù\84Ù\83ترÙ\88Ù\86Ù\8a Ø§Ù\84جدÙ\8aد Ù\81ارغا Ø¹Ù\86د ØªÙ\82دÙ\8aÙ\85 Ø§Ù\84استÙ\85ارة.",
        "changeemail-no-info": "يجب تسجيل الدخول للوصول إلى هذه الصفحة مباشرة.",
        "changeemail-oldemail": "عنوان البريد الإلكتروني الحالي:",
        "changeemail-newemail": "عنوان البريد الإلكتروني الجديد:",
        "rcfilters-clear-all-filters": "مسح كل المرشحات",
        "rcfilters-show-new-changes": "عرض التغييرات الجديدة منذ $1",
        "rcfilters-search-placeholder": "رشح التغييرات (استخدم القائمة أو ابحث عن اسم المرشح)",
+       "rcfilters-search-placeholder-mobile": "مرشحات",
        "rcfilters-invalid-filter": "مرشح غير صحيح",
        "rcfilters-empty-filter": "لا مرشحات فعالة. كل المساهمات معروضة.",
        "rcfilters-filterlist-title": "مرشحات",
        "zip-bad": "ملف ZIP هذا معطوب أو لا يمكن قراءته لسبب آخر.\nلا يمكن التحقق من سلامته.",
        "zip-unsupported": "هذا ملف ZIP يستخدم بعض مزايا ZIP التي لا يدعمها ميدياويكي.\nلا يمكن التحقق من سلامته.",
        "uploadstash": "تحميل مخبأ",
-       "uploadstash-summary": " توفر هذه الصفحة الوصول إلى الملفات التي يتم تحميلها (أو في أثناء عملية التحميل) ولكنها لم تنشر بعد. هذه الملفات هي غير مرئية لأحد إلا للمستخدم الذين تم الرفع لهم.",
+       "uploadstash-summary": "توفر هذه الصفحة الوصول إلى الملفات التي يتم رفعها ولكنها لم تُنشَر بعد في الويكي، هذه الملفات غير مرئية لأحد إلا للمستخدم الذي رفعها.",
        "uploadstash-clear": "مسح الملفات المخبأة",
        "uploadstash-nofiles": "ليس لديك أي ملفات مخبأة.",
        "uploadstash-badtoken": "فشل أداء ذلك العمل، ربما لأن وثائق تفويض التحرير الخاصة بك منتهية الصلاحية. من فضلك حاول مرة أخرى.",
        "changecontentmodel": "تغيير نموذج المحتوى لصفحة",
        "changecontentmodel-legend": "غير نموذج المحتوى",
        "changecontentmodel-title-label": "عنوان الصفحة",
+       "changecontentmodel-current-label": "نموذج المحتوى الحالي",
        "changecontentmodel-model-label": "نموذج محتوى جديد",
        "changecontentmodel-reason-label": "السبب:",
        "changecontentmodel-submit": "تغيير",
        "block-log-flags-angry-autoblock": "المنع التلقائي المتقدم مفعل",
        "block-log-flags-hiddenname": "اسم المستخدم مخفي",
        "range_block_disabled": "إمكانية مدير النظام لمنع نطاق معطلة.",
+       "ipb-prevent-user-talk-edit": "يجب السماح بتحرير صفحة نقاشه لمنع جزئي، ما لم يتضمن تقييدا على نطاق نقاش المستخدم.",
        "ipb_expiry_invalid": "تاريخ الانتهاء غير صحيح.",
        "ipb_expiry_old": "توقيت انتهاء المنع واقع في الماضي.",
        "ipb_expiry_temp": "عمليات منع أسماء المستخدمين المخفية يجب أن تكون دائمة.",
        "move-page-legend": "نقل الصفحة",
        "movepagetext": "باستخدام  الاستمارة بالأسفل بإمكانك أن تغير اسم الصفحة، وأن تنقل تاريخها إلى الاسم الجديد.\nالعنوان القديم سيصبح تحويلة للعنوان الجديد.\nيمكنك أن تترك التحويلات التي تشير إلى العنوان الأصلي كما هي لتقوم البوتات بتحديثها تلقائياً.\nإذا اخترت أن تقوم بالتحديث يدوياً، فتأكد من عدم وجود تحويلات [[Special:DoubleRedirects|مزدوجة]] أو [[Special:BrokenRedirects|مكسورة]] وقم بتصحيحها.\nأنت المسؤول عن التأكد من أن الوصلات تصل إلى الصفحات التي يفترض أن تصل إليها.\n\nلاحظ أنه <strong>لن يتم</strong> نقل الصفحة إذا وجدت صفحة في العنوان الجديد، إلا إذا كانت صفحة تحويل، ولا تاريخ لها.\nهذا يعني أنك تستطيع استرجاع الصفحة إلى مكانها لو قمت بخطأ، ولا يمكنك نسخ هذه الصفحة فوق صفحة موجودة.\n\n<strong>ملاحظة:</strong>\n\nهذا قد يكون تغييراً كارثياً وغير متوقع لصفحة مشهورة؛\nمن فضلك تأكد أنك تفهم عواقب هذا الفعل قبل أن تستمر.",
        "movepagetext-noredirectfixer": "باستخدام  الاستمارة بالأسفل بإمكانك أن تغير اسم الصفحة، وأن تنقل تاريخها إلى الاسم الجديد.\nالعنوان القديم سيصبح تحويلة للعنوان الجديد.\nيمكنك تحديث التحويلات التي تشير إلى العنوان الأصلي تلقائياً.\nلو اخترت ألا تفعل، تأكد من عدم وجود تحويلات [[Special:DoubleRedirects|مزدوجة]] أو [[Special:BrokenRedirects|مكسورة]].\nأنت المسؤول عن التأكد من أن الوصلات تصل إلى الصفحات التي يفترض أن تصل إليها.\n\nلاحظ أنه <strong>لن يتم</strong>  نقل الصفحة إذا كان هناك صفحة بنفس العنوان الجديد، إلا إذا كانت فارغة، أو تحويلة لا تاريخ لها.\nهذا يعني أنك تستطيع استرجاع الصفحة إلى مكانها لو قمت بخطأ، وأنك لا يمكنك الكتابة على صفحة موجودة.\n\n<strong>ملاحظة</strong> \n\nهذا قد يكون تغييراً كارثياً وغير متوقع لصفحة مشهورة؛\nمن فضلك تأكد أنك تفهم عواقب هذا الفعل قبل أن تستمر.",
+       "movepagetext-noredirectsupport": "سيؤدي استخدام النموذج أدناه إلى إعادة تسمية صفحة، مع نقل كل تاريخها إلى الاسم الجديد،\nأنت مسئول عن التأكد من أن الروابط لا تزال تشير إلى المكان الذي من المفترض أن تذهب إليه.\n\nلاحظ أنه <strong>لن</strong> يتم نقل الصفحة إذا كانت هناك بالفعل صفحة في العنوان الجديد:\nهذا يعني أنه يمكنك إعادة تسمية صفحة إلى المكان الذي تمت إعادة تسميتها منه إذا ارتكبت خطأً، ولا يمكنك استبدال صفحة موجودة.\n\n<strong>ملاحظة:</strong>\nيمكن أن يكون هذا تغييرا جذريا وغير متوقع لصفحة شائعة؛\nيُرجَى التأكد من أنك تفهم عواقب هذا قبل المتابعة.",
        "movepagetalktext": "لو علمت على هذا الصندوق، فصفحة النقاش المرفقة يتم نقلها أوتوماتيكيا للعنوان الجديد، إلا لو كانت صفحة نقاش غير فارغة هناك بالفعل.\n\nفي هذه الحالة، فسيتعين عليك نقل أو دمج الصفحة يدويا لو رغبت في ذلك.",
        "moveuserpage-warning": "<strong>تحذير:</strong> أنت على وشك نقل صفحة مستخدم. من فضلك لاحظ أن الصفحة وحدها سوف تنقل وأن المستخدم <em>لن</em> تعاد تسميته.",
        "movecategorypage-warning": "<strong>تحذير:</strong> أنت على وشك نقل صفحة التصنيف إلى عنوان جديد؛ <em>لن</em> تنقل الصفحات المندرجة تحت التصنيف إلى العنوان الجديد.",
        "linkaccounts": "ربط الحسابات",
        "linkaccounts-success-text": "الحساب تم وصله.",
        "linkaccounts-submit": "اربط الحسابات",
+       "cannotunlink-no-provider-title": "لا توجد حسابات مرتبطة لإلغاء الربط",
+       "cannotunlink-no-provider": "لا توجد حسابات مرتبطة يمكن إلغاء ربطها.",
        "unlinkaccounts": "إزالة ربط الحسابات",
        "unlinkaccounts-success": "الحساب تم فك وصله.",
        "authenticationdatachange-ignored": "تغيير بيانات التحقق لم يتم التعامل معه. ربما لم يتم ضبط موفر؟",
index 0fd9873..f863d06 100644 (file)
@@ -25,7 +25,8 @@
                        "Sagsag",
                        "Bodhisattwa",
                        "Vlad5250",
-                       "ৰাজীৱ গোস্বামী"
+                       "ৰাজীৱ গোস্বামী",
+                       "আফতাবুজ্জামান"
                ]
        },
        "tog-underline": "সংযোগসমূহ অধোৰেখিত কৰক:",
        "enotif_body_intro_changed": "{{SITENAME}}ৰ পৃষ্ঠা $1, $PAGEEDITDATE তাৰিখে {{gender:$2|$2}}ৰ দ্বাৰা সলনি কৰা হৈছিল, বৰ্তমানৰ সংস্কৰণৰ বাবে $3 চাওক।",
        "enotif_lastvisited": "আপোনাৰ শেষ পৰিদৰ্শনৰ পিছত হোৱা সকলো সালসলনিৰ বাবে $1 চাওক ।",
        "enotif_lastdiff": "এই পৰিৱৰ্তনটো চাবৰ বাবে $1 চাওক ।",
-       "enotif_anon_editor": "বà§\87নামà§\80 à¦¸à¦¦à¦¸à§\8dয $1",
+       "enotif_anon_editor": "নামহà§\80ন à¦¬à§\8dযবহারà¦\95ারà§\80 $1",
        "enotif_body": "প্ৰিয় $WATCHINGUSERNAME,\n\n$PAGEINTRO $NEWPAGE\n\nসম্পাদকৰ সাৰাংশ: $PAGESUMMARY $PAGEMINOREDIT\n\nসম্পাদকজনক যোগাযোগ কৰক:\nমেইল: $PAGEEDITOR_EMAIL\nৱিকি: $PAGEEDITOR_WIKI\n\nআপুনি এই পৃষ্ঠাটো প্ৰৱেশ কৰি নোচোৱালৈকে আন সালসলনিৰ কোনো জাননী দিয়া নহ’ব। আপুনি আপোনাৰ লক্ষ্য-তালিকাৰ পৃষ্ঠাবোৰৰ জাননী ফ্লেগ পূৰ্বৰ অৱস্থালৈও ঘূৰাই নিব পাৰে ।\n\nআপোনাৰ {{SITENAME}} জাননী ব্যৱস্থা\n\n--\nআপোনাৰ ই-মেইল জাননী ছেটিং সলনি কৰিবলৈ চাওক\n{{canonicalurl:{{#special:Preferences}}}}\n\nআপোনাৰ লক্ষ্য-তালিকাৰ ছেটিং সলনি কৰিবলৈ চাওক\n{{canonicalurl:{{#special:EditWatchlist}}}}\n\nআপোনাৰ লক্ষ্য-তালিকাৰ পৰা পৃষ্ঠা বিলোপ কৰিবলৈ চাওক\n$UNWATCHURL\n\nপ্ৰতিক্ৰিয়া আৰু অধিক সহযোগিতাৰ বাবে:\n$HELPPAGE",
        "created": "সৃষ্টি কৰা হ’ল",
        "changed": "সলোৱা হৈছে",
index ea05ebc..189dddb 100644 (file)
        "search-interwiki-more": "(más)",
        "search-interwiki-more-results": "más resultaos",
        "search-relatedarticle": "Rellacionáu",
+       "search-invalid-sort-order": "Nun se reconoció la ordenación por $1, aplicaráse la ordenación predeterminada. Los tipos d'ordenación válidos son: $2",
        "searchrelated": "rellacionáu",
        "searchall": "toos",
        "showingresults": "Abaxo s'{{PLURAL:$1|amuesa hasta <strong>un</strong> resultáu|amuesen <strong>$1</strong> resultaos}}, principando por #<strong>$2</strong>.",
        "rcfilters-clear-all-filters": "Borrar tolos filtros",
        "rcfilters-show-new-changes": "Ver los cambeos nuevos dende $1",
        "rcfilters-search-placeholder": "Filtriar cambeos (usa'l menú o busca'l nome del filtru)",
+       "rcfilters-search-placeholder-mobile": "Filtros",
        "rcfilters-invalid-filter": "Filtru inválidu",
        "rcfilters-empty-filter": "Nun hai filtros activos. Amuésense toles contribuciones.",
        "rcfilters-filterlist-title": "Filtros",
        "newtitle": "Títulu nuevu:",
        "move-watch": "Vixilar les páxines d'orixe y destín",
        "movepagebtn": "Treslladar la páxina",
-       "pagemovedsub": "Treslláu correctu",
+       "pagemovedsub": "Treslláu correutu",
        "cannotmove": "La páxina nun pudo treslladase {{PLURAL:$1|pol siguiente motivu|polos siguientes motivos}}:",
        "movepage-moved": "<strong>«$1» treslladóse a «$2»</strong>",
        "movepage-moved-redirect": "Creóse una redireición.",
        "passwordpolicies-policy-passwordnotinlargeblacklist": "La contraseña nun pué tar na llista de les 100.000 contraseñes más usaes.",
        "passwordpolicies-policyflag-forcechange": "tien de camudase al aniciar sesión",
        "passwordpolicies-policyflag-suggestchangeonlogin": "suxerir cambiu al aniciar sesión",
+       "mycustomjsredirectprotected": "Nun tienes permisu pa editar esta páxina en JavaScript porque ye una redireición que nun apunta dientro del to espaciu d'usuariu.",
        "easydeflate-invaliddeflate": "El conteníu dau nun ta comprimíu correutamente",
        "unprotected-js": "Por razones de seguridá, JavaScript nun puede cargase dende páxines ensin protexer. Crea javascript sólo nel espaciu de nomes MediaWiki: o como subpáxina d'usuariu",
        "userlogout-continue": "¿Desees zarrar la sesión?"
index e6581ed..51c9ffc 100644 (file)
        "action-userrights-interwiki": "Digər vikilərdəki istifadəçilərin istifadəçi hüquqlarını dəyişdir",
        "action-siteadmin": "Məlumatlar bazasının bloklanması və blokun götürülməsi",
        "action-sendemail": "e-məktub göndər",
-       "action-editmywatchlist": "izləmə siyahınızda redaktə",
+       "action-editmywatchlist": "izləmə siyahınızı redaktə edin",
        "action-viewmywatchlist": "İzləmə siyahınıza baxın",
        "action-viewmyprivateinfo": "Şəxsi məlumatlarınıza baxın",
        "action-editmyprivateinfo": "Şəxsi məlumatlarınızı redaktə edin",
        "rcfilters-savedqueries-add-new-title": "Hazırkı filtr nizamlamalarını yaddaşa ver",
        "rcfilters-restore-default-filters": "Standart filtrləri bərpa et",
        "rcfilters-clear-all-filters": "Bütün filtrləri sil",
-       "rcfilters-show-new-changes": "Ən son dəyişiklikləri göstər: $1",
+       "rcfilters-show-new-changes": "$1 tarixindən bəri ən son dəyişiklikləri göstər",
        "rcfilters-search-placeholder": "Son dəyişiklikləri filtrlə (siyahıdan seçin və ya daxil edin)",
        "rcfilters-empty-filter": "Aktiv filtr yoxdur. Bütün redaktələr göstərilir.",
        "rcfilters-filterlist-title": "Filtrlər",
index e1e663c..91544a0 100644 (file)
        "category-file-count-limited": "جهلیگین {{PLURAL:$1|فایل|$1 فایلان}} ته هنوکین دسته اینت",
        "listingcontinuesabbrev": "ادامه.",
        "index-category": "سرتاک بوتگێن پێجان",
-       "noindex-category": "سرتاک نبوتگین پیجان",
+       "noindex-category": "سرتاک نبݔتگݔں تاکاں",
        "broken-file-category": "پیج گون پرشتگین لینک فایل",
        "about": "بارہ‌ئا",
        "article": "محتوا صفحه",
        "search-interwiki-more": "(گݔشتِر)",
        "search-interwiki-more-results": "گݔشترݔں نتیجہ",
        "search-relatedarticle": "مربوطین",
+       "search-invalid-sort-order": "$1ء ترتیپ شرّ نئں پݔسری ترتیپ پدا سپتءَ بئ اَنت.\nھنیگݔں ترتیپ: $2",
+       "search-unknown-profile": "$1ء شۏھاز نابزانت اِنت،پݔسری شۏھازی سپتءَ بئ اَنت۔",
        "searchrelated": "مربوط",
        "searchall": "کل",
        "showingresults": "جهل پیش دارگنت تا  {{PLURAL:$1|'''1'''نتیجه|'''$1''' نتایج}} شروع بنت گون #'''$2'''.",
        "datedefault": "هچ ترجیح",
        "prefs-labs": "اپشن پر چکاس",
        "prefs-user-pages": "کاربریگین تاکان",
-       "prefs-personal": "نمایه کاربر",
+       "prefs-personal": "کارزورۏکء نما",
        "prefs-rc": "نۏکݔں ٹگلاں",
        "prefs-watchlist": "چارگء لیست",
        "prefs-editwatchlist": "چارگ لیستءِ ٹگلݔنگ",
index f29ed77..4c81857 100644 (file)
                        "Macofe",
                        "Matma Rex",
                        "ShimunUfesoj",
-                       "Vlad5250"
+                       "Vlad5250",
+                       "Brazal.dang"
                ]
        },
        "tog-underline": "Linyahan an kilyawan:",
        "tog-hideminor": "Tagoon an saradít na mga pagliwat sa dae pa sana nahaloy na mga pagbabàgo",
        "tog-hidepatrolled": "Tagóa an patrolyadong mga paghirá sa nakakaági pa sanáng pagbabàgo",
        "tog-newpageshidepatrolled": "Tagoon an patrolyadong mga pahina gikan sa baguhong listahan nin pahina",
+       "tog-hidecategorization": "Itago an pagkagrupo kan mga pahina",
        "tog-extendwatchlist": "Palakbanga an bantay-listahan (watchlist) na maipahiling an gabos na pinagbago, bako sana an pinakahurihang binago",
        "tog-usenewrc": "Pangrupong mga kaliwatan sa kada pahina kan mga dae pa sana nahaloy na mga kaliwatan asin bantay-listahan",
        "tog-numberheadings": "Tolos-bilang na mga pamayohán",
        "tog-watchlisthideliu": "Tagoon an mga pagbabagong nahimo kan mga nakalaog na paragamit gikan sa bantayang listahan",
        "tog-watchlisthideanons": "Tagoon an mga pagbabagong nahimo kan mga bakong bistadong paragamit gikan sa bantayang listahan",
        "tog-watchlisthidepatrolled": "Tagoon an mga patrolyadong pagbabago gikan sa bantayang listahan",
+       "tog-watchlisthidecategorization": "Itago an pagkagrupo kan mga pahina",
        "tog-ccmeonemails": "Ipadara sako an mga kopya kan e-koreo na pinadara ko sa ibang mga paragamit",
        "tog-diffonly": "Dai tabi ihayag an laog kan pahina sa ibaba nin mga diffs",
        "tog-showhiddencats": "Ihayag an nakatagong mga kategorya",
        "tog-norollbackdiff": "Omidohon an diff matapos himoon an pagbalikot",
        "tog-useeditwarning": "Patanidan ako kunsoarin na ako nagbaya sa pahinang pigliliwat na dae naitatagama an mga kaliwatan",
        "tog-prefershttps": "Pirmeng gumamit nin sarong seguradong koneksyon kunsoarin na ika nakalaog na",
-       "underline-always": "Parati",
+       "underline-always": "Pirmi",
        "underline-never": "Dae pa lamang",
        "underline-default": "Kublit o kilyaw na panugmad",
        "editfont-style": "Baguhon an estilo nin kalwig sa sinasakupan",
        "october-date": "Oktobre $1",
        "november-date": "Nobyembre $1",
        "december-date": "Disyembre $1",
+       "period-am": "Aga",
+       "period-pm": "Hapon",
        "pagecategories": "{{PLURAL:$1|Kategorya|Mga kategorya}}",
        "category_header": "Mga pahina sa kategoryang \"$1\"",
        "subcategories": "Mga sub-kategorya",
        "morenotlisted": "Ining listahan bakong kumpleto.",
        "mypage": "Pahina",
        "mytalk": "Mag-ulay",
-       "anontalk": "Urulay para kaining IP estada",
+       "anontalk": "Mag-ulay",
        "navigation": "Paglibotlibot",
        "and": "&#32;asin",
        "faq": "PHK (Pirmehang Hinahapot na mga Kahaputan)",
        "returnto": "Magbalik sa $1.",
        "tagline": "Gikan sa {{SITENAME}}",
        "help": "Katabangan",
+       "help-mediawiki": "Tabang tungkol sa MediaWiki",
        "search": "Maghanap",
        "search-ignored-headings": " #<!-- walaton ining linya eksaktong siring sana kaini --> <pre> \n# Mga Kapamayuhanan na pinagpapabayaan sa paghahanap. \n# Mga Kaliwatan kaini magkaka-epekto matapos na an pahina na igwang kapamayuhanan maipaghukdo. \n# Ika makakapagpuwersa sa pahina na maihuhukdo otro sa paagi nin paghimo nin sarong blangko na pagliwat. # An Sintaks iyo ining minasunod: \n# * An gabos magpoon sa sarong karakter na \"#\" sagkod sa tapos kan linya iyo an sarong komento \n# * An lambang linya na bakong blangko iyo an eksaktong titulo na pababayaan, kaso asin gabos na bagay \nMga Panultulan\nPanluwas na mga sugpon\nHilingon man \n#</pre> <!-- walaton ining linya eksaktong siring sana kaini -->",
        "searchbutton": "Maghanap",
-       "go": "Dumani",
-       "searcharticle": "Lakaw",
+       "go": "Dumanán",
+       "searcharticle": "Dumanán",
        "history": "Historiya nin pahina",
        "history_short": "Historiya",
+       "history_small": "historiya",
        "updatedmarker": "dinagdagan poon kan sakong huring pagbisita",
        "printableversion": "Nalilimbag na bersyon",
-       "permalink": "Permanenteng kilyawan",
+       "permalink": "Permanenteng sugpon",
        "print": "Ilimbag",
        "view": "Tànawon",
        "view-foreign": "Hilingon sa $1",
        "delete": "Puraon",
        "undelete_short": "Dae puraon an {{PLURAL:$1|sarong pagliwat|$1 mga pagliwat}}",
        "viewdeleted_short": "Hilingon {{PLURAL:$1|sarong pinura na pagliwat|$1 mga pinura na pagliwat}}",
-       "protect": "Protektari",
+       "protect": "Protektaran",
        "protect_change": "Ribayan",
        "unprotect": "Ribayan an proteksyon",
        "newpage": "Bàguhong pahina",
        "talk": "Urulayan",
        "views": "Mga Tanawon",
        "toolbox": "Mga gamiton:",
+       "tool-link-userrights": "Ribayan {{GENDER:$1|paragamit}} an grupo",
+       "tool-link-userrights-readonly": "Hilingon {{GENDER:$1|paragamit}} an grupo",
+       "tool-link-emailuser": "E-surat kaining {{GENDER:$1|paragamit}}",
        "imagepage": "Tànawon an pahina nin sagunson (file)",
        "mediawikipage": "Tànawon an pahina kan mensahe",
        "templatepage": "Tànawon an pahina kan panguyog",
        "viewhelppage": "Tànawon an pahina nin pagtabang",
-       "categorypage": "Tànawon an pahina nin kategorya",
+       "categorypage": "Tànawon an pahina nin kategoriya",
        "viewtalkpage": "Tànawon an urulay",
        "otherlanguages": "Sa ibang mga lengguwahe",
        "redirectedfrom": "(Pinagbalikwat gikan sa $1)",
        "pool-timeout": "Timeout naghahalat para makapanugpon",
        "pool-queuefull": "An grupong panproseso panoon",
        "pool-errorunknown": "Bakong bistadong sala",
+       "poolcounter-usage-error": "Sala sa paggamit: $1",
        "aboutsite": "Dapít sa {{SITENAME}}",
        "aboutpage": "Project:Mapanonongod",
        "copyright": "An kalamnan manunumpungan sa laog kan $1 o baya notado na ining laen.",
        "disclaimers": "Mga Pangindahan",
        "disclaimerpage": "Project:Pangkagabsán na pangindahan",
        "edithelp": "Pantabang sa pagliliwat",
+       "helppage-top-gethelp": "Katabangan",
        "mainpage": "Panginot na Pahina",
        "mainpage-description": "Panginot na Pahina",
        "policy-url": "Project:Kalakawan",
        "hidetoc": "tagoon",
        "collapsible-collapse": "Pahalipoton",
        "collapsible-expand": "Pahiwason",
+       "confirmable-confirm": "Sigurado {{GENDER:$1|ka}} na?",
        "confirmable-yes": "Iyo",
        "confirmable-no": "Dae",
        "thisisdeleted": "Hilingon o isulit an $1?",
        "filerenameerror": "Dai natàwan nin bàgong ngaran an file na \"$1\" sa \"$2\".",
        "filedeleteerror": "Dai naparà an file na \"$1\".",
        "directorycreateerror": "Dai nagibo an direktorya na \"$1\".",
+       "directoryreadonlyerror": "Ang directory \"$1\" pambasa lang.",
+       "directorynotreadableerror": "Ang directory \"$1\" dae nababasa.",
        "filenotfound": "Dai nahanap an file na \"$1\".",
        "unexpected": "Dai pighuhunà na balór: \"$1\"=\"$2\".",
        "formerror": "Salâ: Dae maisusumiter an porma.",
        "badarticleerror": "Ining aksyon dae magigibo sa pahinang ini.",
        "cannotdelete": "An pahina o an sagunson (file) na \"$1\" dae tabi napupura.\nIni puwede nang napura kan iba.",
        "cannotdelete-title": "Dae mapura an pahina na \"$1\"",
+       "delete-scheduled": "Ang pahina \"$1\" nakatakda nang puraon.\nMagin mapagpasensiya.",
        "delete-hook-aborted": "An pagpura pinundo kan pangawit.\nIni dae nagtao nin kapaliwanagan.",
        "no-null-revision": "Dae makakamukna nin baguhong bunyaw na rebisyon para sa pahina \"$1\"",
        "badtitle": "Raot na titulo",
        "cascadeprotected": "Ining pahina pinagprotehiran gikan sa pagliliwat nin huli ta kabaling pinagbalyo sa minasunod na {{PLURAL:$1|pahina, na iyo ngani an|mga pahina, na iyo ngani an mga}} protektado na igwa nin \"pasurunod\" na opsyong pinagbuksan:\n$2",
        "namespaceprotected": "Ika mayong permiso sa pagliwat nin mga pahina sa <strong>$1</strong> na ngarang-espasyo.",
        "customcssprotected": "Ika mayong permiso sa pagliwat kaining pahinang CSS, nin huli ta ini naglalaman kan personal na panuytoy (settings) kan ibang paragamit.",
+       "customjsonprotected": "Ika mayong permiso sa pagliwat kaining pahinang JSON, nin huli ta ini naglalaman kan personal na panuytoy (settings) kan ibang paragamit.",
        "customjsprotected": "Ika mayong permiso sa pagliwat kaining pahinang JavaScript, nin huli ta ini naglalaman kan personal na panuytoy (settings) kan ibang paragamit.",
+       "sitecssprotected": "Ika mayong permiso sa pagliwat kaining pahinang CSS, nin huli ta ini pwedeng makaapekto sa ibang bisita.",
+       "sitejsonprotected": "Ika mayong permiso sa pagliwat kaining pahinang JSON, nin huli ta ini pwedeng makaapekto sa ibang bisita.",
+       "sitejsprotected": "Ika mayong permiso sa pagliwat kaining pahinang JavaScript, nin huli ta ini pwedeng makaapekto sa ibang bisita.",
        "mycustomcssprotected": "Ika mayo nin permiso sa pagliwat kaining CSS na pahina.",
+       "mycustomjsonprotected": "Ika mayo nin permiso sa pagliwat kaining JSON na pahina.",
        "mycustomjsprotected": "Ika mayo nin permiso sa pagliwat kaining JavaScript na pahina.",
        "myprivateinfoprotected": "Ika daeng permiso na magliliwat kan pribado mong impormasyon.",
        "mypreferencesprotected": "Ika daeng permiso na magliliwat kan saimong mga kamuyahan.",
        "ns-specialprotected": "Mga espesyal na pahina dae makakapagliwat.",
        "titleprotected": "Ining titulo pinagprotektaran poon pagkamukna ni [[User:$1|$1]].\nAn rason na pinagtao iyo na <em>$2</em>.",
        "filereadonlyerror": "Dae kinayang baguhon an sagunson (file) \"$1$ nin huli ta an repositoryo kan sagunson \"$2\" yaon sa kamugtakan na basahon sana.\n\nAn administrador na iyo an nagkandado kaini nagpahayag kaining kapaliwanagan: \"$3\".",
+       "invalidtitle": "Salang titulo.",
        "invalidtitle-knownnamespace": "Imbalidong titulo na igwang espasyadong ngaran na \"$2\" asin teksto na \"$3\"",
        "invalidtitle-unknownnamespace": "Imbalidong titulo na igwang nin bakong bistado na bilang kan espasyadong ngaran na $1 asin teksto na \"$2\"",
        "exception-nologin": "Dai ka nakapaglaog",
        "virus-scanfailed": "An paghingipid (scan) nagpalya (Koda $1)",
        "virus-unknownscanner": "bakong bistadong antivirus:",
        "logouttext": "'''Ika ngunyan nakaluwas na.'''\n\nTandai tabi na an nagkapirang mga pahina puwedeng maipagpapadagos na ipagpapahiling siring sa ika baga yaon sa laog pa, sagkod na maiklarado an saimong pankilyaw na kaaganan.",
+       "logout-failed": "Dae pa makaluwas ngunyan: $1",
+       "cannotlogoutnow-title": "Dae pa makaluwas ngunyan",
        "welcomeuser": "Marhayong pag-abot, $1!",
        "welcomecreation-msg": "An saimong panindog pinagmukna na.\nDae malingaw na liwaton an saimong [[Special:Preferences|{{SITENAME}} mga kamuyahan]].",
        "yourname": "Pangaran kan paragamit:",
        "createacct-yourpasswordagain-ph": "Pakikaag otro an sekretong panlaog",
        "userlogin-remembermypassword": "Dagos mo akong giromdomon na nakalaog",
        "userlogin-signwithsecure": "Gamiton an seguradong koneksyon",
+       "cannotlogin-title": "Dai makalaog",
+       "cannotlogin-text": "Dai posible an paglaog.",
+       "cannotloginnow-title": "Dae pa makalaog ngunyan",
+       "cannotloginnow-text": "Dai posible an paglaog kun magamit nin $1.",
+       "cannotcreateaccount-title": "Dai makagibo nin account",
        "yourdomainname": "An saimong kasakupan:",
        "password-change-forbidden": "Ika dae makapagliwat nin sekretong panlaog sa wiking ini.",
        "externaldberror": "Igwa gayod sala sa arinman kan patunay sa datos-sarayan o ika dae pinagtugutan na bâgohon an saimong panluwas na panindog.",
        "login": "Maglaog",
+       "login-security": "Patunayan an saimong pagkakabisto",
        "nav-login-createaccount": "Maglaog / magmukna nin panindog",
        "logout": "Magluwas",
        "userlogout": "Magluwas",
        "userlogin-resetpassword-link": "Nalingawan mo an saimong pasa-taramon?",
        "userlogin-helplink2": "Katabangan sa paglalaog",
        "userlogin-loggedin": "Ika nakalaog na tabi bilang si {{GENDER:$1|$1}}.\nGamita an porma sa ibaba sa paglaog bilang ibang paragamit.",
+       "userlogin-reauth": "Kaipuhan maglaog ulit para mapatunayan na ika {{GENDER:$1|$1}}.",
        "userlogin-createanother": "Magmukna nin ibang panindog",
        "createacct-emailrequired": "Estada kan e-surat",
        "createacct-emailoptional": "E-surat na estada (opsyonal)",
        "createacct-reason": "Rason",
        "createacct-reason-ph": "Tadaw ta ika magmumukna nin ibang panindog",
        "createacct-submit": "Muknaon an saimong panindog",
-       "createacct-another-submit": "Magmukna nin ibang panindog",
+       "createacct-another-submit": "Magmukna nin panindog",
+       "createacct-continue-submit": "Magpadagos sa paggibo nin panlaog",
+       "createacct-another-continue-submit": "Magpadagos sa paggibo nin panlaog",
        "createacct-benefit-heading": "{{SITENAME}} pinaghimo kan mga tawong siring mo.",
        "createacct-benefit-body1": "{{PLURAL:$1|niliwat|mga niliwat}}",
        "createacct-benefit-body2": "{{PLURAL:$1|pahina|mga pahina}}",
        "createacct-benefit-body3": "pinakahuring {{PLURAL:$1|paraambag|mga paraambag}}",
        "badretype": "An mga sekretong panlaog mong pinagtatak bakong pareho.",
+       "usernameinprogress": "An pagmukna kaning palaog kan paragamit nagpuon na. Maghalat tabi.",
        "userexists": "Paragamit na ngarang piglaog may naggagamit na.\nPakipili nin ibang ngaran tabi.",
        "loginerror": "An paglaog napasalâ",
        "createacct-error": "Kasalaan sa pagmumukna nin panindog",
        "nocookieslogin": "{{SITENAME}} naggagamit nin mga cookies para sa maglaog na mga paragamit.\nIka igwang mga cookies na dae pinagana.\nTabi paganaha sinda asin otroha giraray.",
        "nocookiesfornew": "An panindog kan paragamit dae pinagmukna, nin huli ta dae nyamo kumpirmado an pinaggikanan kaini.\nPakipaseguro na saimong pinagana an cookies, ikarga giraray ining pahina asin probaran mo otro.",
        "noname": "Ika dae tabi nakapagkaag nin sarong balidong pangaran nin paragamit.",
-       "loginsuccesstitle": "Matrayumpo an saimong paglaog",
+       "loginsuccesstitle": "Nakapaglaog na",
        "loginsuccess": "'''Ika ngunyan nakalaog na sa {{SITENAME}} bilang si \"$1\".'''",
        "nosuchuser": "Dae pang paragamit na ginagamit an pangaran na \"$1\".\nAn mga ngaran nin paragamit sensitibo gayo sa tipahan.\nPakireparo kan saimong espeling, o [[Special:CreateAccount|Magmukna nin bagong panindog]].",
        "nosuchusershort": "Mayo po tabing paragamit na an pangaran \"$1\".\nPaki-tsek an saimong espeling.",
        "eauthentsent": "Sarong pankumpirmasyon na e-surat an ipinadara sa isinambit na estada nin e-surat.\nBago an ibang e-surat ipinapadara sa panindog, ika igwang susunudon na mga instruksyon na yaon sa e-surat, tanganing kumpirmaron na an panindog tunay talagang saimo.",
        "throttled-mailpassword": "Sarong e-surat sa pagliliwat kan sekretong panlaog an ipinadara na, sa laog nin {{PLURAL:$1|hour|$1 hours}}.\nTangarig malikayan an abuso, saro sanang e-surat sa pagliliwat kan sekretong panlaog an ipinapadara sa lambang {{PLURAL:$1|hour|$1 hours}}.",
        "mailerror": "Salâ an pagpadará kan koreo: $1",
-       "acct_creation_throttle_hit": "Mga bisita kaining wiki na ginagamit an saimong IP address nagmukna nin {{PLURAL:$1|1 panindog|$1 mga panindog}} sa nakaaging aldaw, na iyo ngani an maximum na pinagtutugot sa laog kan peryodong panahon.\nBilang resulta, an mga bisita na naggagamit kaining IP address dae nguna makakamukna nin mga panindog.",
+       "acct_creation_throttle_hit": "Mga bisita kaining wiki na ginagamit an saimong IP address nagmukna nin {{PLURAL:$1|1 panindog|$1 mga panindog}} sa nakaaging aldaw $2, na iyo ngani an maximum na pinagtutugot sa laog kan peryodong panahon.\nBilang resulta, an mga bisita na naggagamit kaining IP address dae nguna makakamukna nin mga panindog.",
        "emailauthenticated": "An saimong e-surat na estada pinagkumpirma kan $2 mga alas $3.",
        "emailnotauthenticated": "An saimong e-surat na estada dae pa tabi pinagkumpirma.\nMayo tabing e-surat na ipagpapadara para sa arinman kan mga minasunod na mga estima.",
        "noemailprefs": "Magkaag nin sarong e-koreong address sa saimong mga kabotan para gumana ining mga estima.",
        "createacct-another-realname-tip": "An totoong pangaran opsyonal.\nKun gustuhon mong itao ini, ini paggagamiton sa pagtatao nin pagkakabistohan kan paragamit para sa saindang mga kaggibohan.",
        "pt-login": "Maglaog",
        "pt-login-button": "Maglaog",
+       "pt-login-continue-button": "Magpadagos sa paglaog",
        "pt-createaccount": "Magmukna nin panindog",
        "pt-userlogout": "Magluwas",
        "php-mail-error-unknown": "Bakong bantog na kasalaan sa PHP mail() function.",
        "retypenew": "Itaták giraray an bàgong panlaog:",
        "resetpass_submit": "Ipwesto an sekretong panlaog dangan maglaog",
        "changepassword-success": "An saimong pasa-taramon matrayumpong pinagliwat na!",
+       "changepassword-throttled": "Nakapaghimo ka na nin grabe kadakol na pagprubar na maglaog sa dae pa sana nahahaloy. Tabi man pakihalat nin $1 bago ka magprubar giraray.",
+       "botpasswords": "Mga sekretong panlaog kan bot",
+       "botpasswords-disabled": "An mga Bot paswords pinugulan.",
+       "botpasswords-existing": "Mga sekretong panlaog kan bot",
+       "botpasswords-createnew": "Magibo nin bagong sekretong panlaog kan bot",
+       "botpasswords-editexisting": "Baguhon an dati nang sekretong panlaog kan bot",
+       "botpasswords-label-needsreset": "(an sekretong panlaog kaipuhan baguhon)",
+       "botpasswords-label-appid": "Ngaran kan bot:",
+       "botpasswords-label-create": "Muknaon",
+       "botpasswords-label-update": "Panumpay",
+       "botpasswords-label-cancel": "Kanselaron",
+       "botpasswords-label-delete": "Puraon",
+       "botpasswords-label-resetpassword": "Pakibago kan sekretong panlaog",
+       "botpasswords-label-grants-column": "Tinugutan",
+       "botpasswords-bad-appid": "An ngaran kan bot \"$1\" dae tugma.",
+       "botpasswords-created-title": "Gibo na an sekretong panlaog kan bot",
+       "botpasswords-deleted-title": "An sekretong panlaog kan bot pinura na",
        "resetpass_forbidden": "An mga sekretong panlaog dae puwedeng maribayan",
+       "resetpass_forbidden-reason": "An mga sekretong panlaog dae puwedeng maribayan: $1",
        "resetpass-no-info": "Ika dapat nakalaog na tanganing direktang makagamit kaining pahina.",
        "resetpass-submit-loggedin": "Ribayan an sekretong panlaog",
        "resetpass-submit-cancel": "I-kansela",
        "resetpass-abort-generic": "Pagliwat kan sikretong panlaog ipinagpauntok kan sarong ekstensyon.",
        "resetpass-expired": "An saimong pasa-taramon nagpalso na. Tabi man pakikaag nin sarong baguhong pasa-taramon tanganing makalaog ka.",
        "resetpass-expired-soft": "An saimong pasa-taramon nagpalso na, asin kinakaipuhan na baguhon. Tabi man pakipili nin sarong baguhong pasa-taramon ngunyan, o i-klik an \"{{int:authprovider-resetpass-skip-label}}\" kun baguhon sa aro-atyan.",
+       "resetpass-validity": "An saimong pasa-taramon nagpalso na. $1\n\nTabi man pakikaag nin sarong baguhong pasa-taramon tanganing makalaog ka.",
+       "resetpass-validity-soft": "An saimong pasa-taramon nagpalso na, asin kinakaipuhan na baguhon. $1\nTabi man pakipili nin sarong baguhong pasa-taramon ngunyan, o i-klik an \"{{int:authprovider-resetpass-skip-label}}\" kun baguhon sa aro-atyan.",
        "passwordreset": "Pakibago kan sekretong panlaog",
        "passwordreset-text-one": "Kumpletuhon ining porma sa pagliwat otro kan saimong pasa-taramon.",
        "passwordreset-text-many": "{{PLURAL:$1|Kaagi an saro sa mga kaaganan tanganing makaresibe nin sarong temporaryong pasa-taramon sa paagi kan e-surat.}}",
        "passwordreset-emailtext-user": "Paragamit $1 sa {{SITENAME}} naghahagad nin sarong pagiromdom kan detalye nin saimong panindog para sa {{SITENAME}}\n($4). An minasunod na paragamit {{PLURAL:$3|panindog iyo an|mga panindog iyo an}} na asosyado kaining e-koreong address:\n\n$2\n\n\n{{PLURAL:$3|Ining temporaryong sekretong panlaog|Ining mga temporaryong panlaog}} mapapaso sa {{PLURAL:$5|sarong aldaw|$5 mga aldaw}}.\nIka dapat na maglaog asin magpili nin sarong bagong sekretong panlaog ngunyan. Kun ibang tawo an naghimo kaining kahagadan, o kun saimo nang nagiromdoman an saimong orihinal na sekretong panlaog, asin habo mo nang ribayan ini, ipasapara mo na sana an mensaheng ini asin ipadagos mo nang gamiton an saimong lumang sekretong panlaog.",
        "passwordreset-emailelement": "Paragamit-ngaran: \n$1\n\nTemporaryong sekretong panlaog: \n$2",
        "passwordreset-emailsentemail": "Sarong e-surat sa pagliliwat kan sekretong panlaog an ipinadara na.",
+       "passwordreset-emailsentusername": "Sarong e-surat sa pagliliwat kan sekretong panlaog an ipinadara na.",
+       "passwordreset-invalidemail": "Dae pwede an e-surat",
        "changeemail": "Ribayan an e-koreong address",
        "changeemail-header": "Ribayan an panindog na e-koreong address",
        "changeemail-no-info": "Ika dapat nakalaog na tanganing direktang makagamit kaining pahina.",
        "sig_tip": "An saimong pirma na igwang tatak-oras",
        "hr_tip": "Pabalagbag na linya (gamiton paminsan-minsan)",
        "summary": "Sumaryo:",
-       "subject": "Subheto/kapamayuhan:",
+       "subject": "Tema",
        "minoredit": "Ini sarong dikiton na pagliwat",
        "watchthis": "Bantayan ining pahina",
        "savearticle": "Itagáma an pahina",
+       "savechanges": "Itagama an mga kaliwatan",
+       "savearticle-start": "Itagama an pahina",
+       "savechanges-start": "Itagama an mga kaliwatan",
        "preview": "Tànawón",
        "showpreview": "Ipahiling an patanaw",
        "showdiff": "Ipahiling an mga kaliwatan",
        "anoneditwarning": "<strong>Patanid:</strong> Ika dae nakalaog. An saimong estada kan IP mahihiling kan publiko kun ika makahimo nin arinman na mga pagliliwat. Kun ika <strong>[$1 naglaog]</strong> o <strong>[$2 magmukna nin panindog]</strong>, an saimong mga pagliliwat ipagpapanungod sa saimong ngaran-paragamit, kaiba an iba pang mga benepisyo.",
        "anonpreviewwarning": "Dae ka tabi nakalaog. An pagtatagama matala kan saimong IP address sa historya nin pagliwat sa pahinang ini.",
        "missingsummary": "<strong>Pagiromdom:</strong>Ika dae pa nakapagtao nin sumaryo sa pagliwat. Kun i-klik mo an \"$1\" giraray, an saimong pagliwat ipagtatagama na mayo kaiyan.",
-       "missingcommenttext": "Pakikaag nin sarong komento sa ibaba.",
+       "missingcommenttext": "Magkaag nin komento sa ibaba.",
        "missingcommentheader": "'''Pagiromdom:''' Ika dae tabi nagtao nin sarong panultol (subject)/Pamayong linya (headline) para kaining sinambit mo.\nKun saimong pinduton an \"$1\" giraray, an saimong pigliwat matatagama na mayo kaiyan.",
-       "summary-preview": "Paenot na patanaw nin sumaryo:",
-       "subject-preview": "Paenot na patanaw sa Subheto/kapamayuhan:",
+       "summary-preview": "Paenot na patanawkang sumaryo kan pagliwat:",
+       "subject-preview": "Paenot na patanaw sa tema:",
        "blockedtitle": "An paragamit pinagbagat",
        "blockedtext": "'''An saimong paragamit na ngaran o IP address pinagkubkob.'''\n\nAn pagkubkob hinimo ni $1.\nAn rason na ipinagtao iyo an  ''$2''.\n\n* Pagpoon kan pagkubkob: $8\n* Pagpasó kan pagkubkob: $6\n* Katuyuhan kan parakubkob: $7\n\nIka puwedeng magkontak sa $1 or ibang [[{{MediaWiki:Grouppage-sysop}}|administrador]] tanganing pag-orolayan an pagkubkob.\nIka dae makakagamit kan 'e-koreo kaining paragamit' na panuytuyan laen lang na may sarong balidong e-koreo address na ipinahayag sa saimong [[Special:Preferences|panindog na mga kabotan]] asin ika dae pinagkubkob para sa paggamit kaini.\nAn saimong presenteng IP address iyo $3, asin an kubkob ID iyo #$5.\nPakibale na lang tabi an gabos na mga detalye sa itaas sa anuman na mga kahaputan na saimong himoon.",
        "autoblockedtext": "An saimong IP address awtomatikong pinagkubkob nin huli ta ini pinaggamit kan ibang paragamit, na pinagkubkob ni $1.\nAn rason na ipinagtao iyo na:\n\n:''$2''\n\n* Pagpoon kan pagkubkob: $8\n* Pagpasó kan pagkubkob: $6\n* Katuyuhan kan parakubkob: $7\n\nPuwede mong kontakon si $1 o saro sa [[{{MediaWiki:Grouppage-sysop}}|mga administrador]] tanganing pag-orolayan an kubkob.\n\nPatanid tabi dae mo puwedeng gamiton an \"e-koreo kaining paragamit\" estima laen lang kun ika igwa nin sarong balidong e-koreo address na rehistrado sa saimong [[Special:Preferences|paragamit na mga kabotan]] asin ika dae pinagkubkob para sa paggamit kaini.\n\nAn saimong presenteng IP address iyo an $3, asin and Kubkob ID iyo an #$5.\nPakibale tabi an gabos na mga detalye sa itaas sa arinman na mga kahaputan na saimong himoon.",
        "userjspreview": "'''Giromdomon tabi na pigtetest/pighihiling mo sana an patanaw kan saimong JavaScript nin paragamit, dai pa ini naitagama!'''",
        "sitecsspreview": "'''Giromdoma baya na ika nagtatanaw pa sana kaining CSS.'''\n'''Ini dae pa tabi naitatagama!'''",
        "sitejspreview": "'''Giromdoma baya na ika nagtatatanaw pa sana kaining koda sa JavaScript.'''\n'''Ini dae pa tabi naitatagama!'''",
-       "userinvalidconfigtitle": "'''Patanid:''' Mayong ''skin'' na \"$1\". Giromdomon tabî na an .css asin .js na mga páhina naggagamit nin titulong nakasurat sa sadit na letras, halimbawa {{ns:user}}:Foo/vector.css bakong {{ns:user}}:Foo/Vector.css.",
+       "userinvalidconfigtitle": "'''Patanid:''' Mayong ''skin'' na \"$1\". Giromdomon tabî na an .css, .json, asin .js na mga páhina naggagamit nin titulong nakasurat sa sadit na letras, halimbawa {{ns:user}}:Foo/vector.css bakong {{ns:user}}:Foo/Vector.css.",
        "updated": "(Pinagsugpunan na)",
        "note": "'''Paisi:'''",
        "previewnote": "'''Giromdoma na ini sarong patanaw pa sana.'''\nAn saimong mga pinagriliwat dae pa tabi naitatagama!",
        "continue-editing": "Magduman sa lugar nin pagliliwat",
        "previewconflict": "Mahihilíng sa patànaw na ini an tekstong nasa itaas na lugar nin paghirá arog sa maipapahiling kun ini an itatagama mo.",
-       "session_fail_preview": "'''Despensa! Dai mi naipadagos an paghirá mo huli sa pagkawara nin datos kan sesyon.\nProbaran tabì giraray. Kun dai man giraray magibo, probaran na magluwas dangan maglaog giraray.'''",
-       "session_fail_preview_html": "'''Sori po! Dae tabi nyamo maiproseso an saimong pagliwat nin huli sa kawaraan kan datos sa sesyon.'''\n\n''Nin huli ta {{SITENAME}} igwa nin bakong pang naprosesong HTML pinagpagana, an patanaw ipinagtago bilang pag-ingat kontra sa atake kan JavaScript.''\n\n'''Kun ini sarong lehitimong pagprubar nin pagliwat, paki-otro tabi giraray.'''\nKun ini dae man giraray guminana, magprubar na [[Special:UserLogout|magluwas]] asin maglaog giraray.",
+       "session_fail_preview": "'''Despensa! Dai mi naipadagos an paghirá mo huli sa pagkawara nin datos kan sesyon.\nProbaran tabì giraray. Kun dai man giraray magibo, probaran [[Special:UserLogout|na magluwas]] dangan maglaog giraray.'''",
+       "session_fail_preview_html": "'''Sori po! Dae tabi nyamo maiproseso an saimong pagliwat nin huli sa kawaraan kan datos sa sesyon.'''\n\n<em>Nin huli ta {{SITENAME}} igwa nin bakong pang naprosesong HTML pinagpagana, an patanaw ipinagtago bilang pag-ingat kontra sa atake kan JavaScript.<em>\n\n'''Kun ini sarong lehitimong pagprubar nin pagliwat, paki-otro tabi giraray.'''\nKun ini dae man giraray guminana, magprubar na [[Special:UserLogout|magluwas]] asin maglaog giraray, asin siguraduhon na ang browser nag-aako nin cookies sa site na ini.",
        "token_suffix_mismatch": "'''Dai pigtogotan an paghirá mo ta sinabrit kan client mo an punctuation characters.\nDai pigtogotan ining paghirá tangarig maibitaran na maraot an teksto kan pahina.\nNanyayari nanggad ini kun naggagamit ka nin bakong maraháy asin dai bistong web-based proxy service.'''",
        "edit_form_incomplete": "'''An ibang mga parte kan porma nin pagliwat dae nakaabot sa serbidor; paki-dobleng mansay na an saimong mga pinagliwat bilog na yaon pa asin paki-otro giraray.'''",
        "editing": "Pigliliwat an $1",
        "readonlywarning": "'''Patanid tabi: An datos-sarayan nakakandado para sa maintenance, kaya ika dae makakapagtagama kan saimong mga pinagriliwat sa ngunyan.'''\nIka mapuwedeng makakopya asin idukot an saimong teksto pasiring sa sarong sagunson kan teksto asin itagama ini sa bandang huri.\n\nAn administrador na iyo an nagkandado kaini naghayag kaining kapaliwanagan: $1",
        "protectedpagewarning": "'''Patanid tabi: Ining pahina pinagprotektaran tanganing an mga paragamit sana na igwang pribilihiyo bilang administrador an makakapagliwat kaini.'''\nAn pinakahuring entrada sa talaan pinaghaya sa ibaba bilang reperensiya:",
        "semiprotectedpagewarning": "'''Note:''' Ining pahina pinagprotektaran na tanganing an mga rehistradong mga paragamit sana an mapuwedeng makapagliwat kaini.\nAn pinakahuring entrada sa talaan pinaghaya sa ibaba bilang reperensiya:",
-       "cascadeprotectedwarning": "<strong>Patanid:</strong> Ining pahina pinagprotehiran na tanganing an mga paragamit na igwa nin pan-administrador na mga pribilihiyo an makakaliwat kaini nin huli ta ini kabaling pinagbalyo sa minasunod na protektadong pasurunod na {{PLURAL:$1|pahina|mga pahina}}:",
+       "cascadeprotectedwarning": "<strong>Patanid:</strong> Ining pahina pinagprotehiran na tanganing an mga paragamit na igwa nin pan-administrador na mga [[Special:ListGroupRights|pribilihiyo]] an makakaliwat kaini nin huli ta ini kabaling pinagbalyo sa minasunod na protektadong pasurunod na {{PLURAL:$1|pahina|mga pahina}}:",
        "titleprotectedwarning": "'''Patanid tabi: Ining pahina pinagprotektaran na tanganing [[Special:ListGroupRights|espesipikong karapatan]] minakaipo tanganing magmukna kaini.'''\nAn pinakahuring entrada sa talaan pinaghaya sa ibaba bilang reperensiya:",
        "templatesused": "{{PLURAL:$1|Template|Mga Panguyog}} na pinaggamit kaining pahina:",
        "templatesusedpreview": "{{PLURAL:$1|Template|Mga Panguyog}} na pinaggamit kaining patanaw:",
        "defaultmessagetext": "Tugmadong mensahe sa teksto",
        "content-failed-to-parse": "Nagpalya sa paglunhay an $2 na laman para sa $1 na modelo: $3",
        "invalid-content-data": "Imbalidong datos nin laman",
-       "content-not-allowed-here": "\"$1\" na laman dae pinagtutugutan sa pahina [[:$2]]",
+       "content-not-allowed-here": "\"$1\" na laman dae pinagtutugutan sa pahina [[:$2]] sa \"$3\"",
        "editwarning-warning": "Sa pagbaya kaining pahina magkakausa saimo na mawara an anuman na mga kaliwatan na saimong pinaghimo. Kun ika nakapaglaog na, ika puwedeng makapagpauntok kaining patanid sa \"{{int:prefs-editing}}\" na seksyon kan saimong mga kamuyahan.",
+       "editpage-invalidcontentmodel-title": "Kalamnan nin pormat bakong suportado",
        "editpage-notsupportedcontentformat-title": "Kalamnan nin pormat bakong suportado",
        "editpage-notsupportedcontentformat-text": "An pormat nin kalamnan na $1 bakong suportado kan modelong kalamnan na $2.",
+       "slot-name-main": "Kapamayuhanan",
        "content-model-wikitext": "wiki-teksto",
        "content-model-text": "yanong-teksto",
        "content-model-javascript": "JavaScript",
        "content-model-css": "CSS",
+       "content-json-empty-object": "Halion an bagay",
+       "content-json-empty-array": "Halion an array",
        "duplicate-args-warning": "<strong>Patanid:</strong> [[:$1]] nag-aapod [[:$2]] na igwa nin sobra sa sarong halaga para sa \"$3\" na parametro. An pinakahuring halaga sanang ipinagtao an magagamit.",
        "expensive-parserfunction-warning": "'''Patanid tabi:''' Ining pahina naglalaman nin grabe kadakulon na ekspensibong programang pambaranga sa punksyon nin mga pag-aapod.\n\nIni dapat magkaigwa nin menos sanang $2 {{PLURAL:$2|apod|mga apod}}, igwa na {{PLURAL:$1|ngunyan nin $1 apod|ngunyan nin $1 mga apod}}.",
        "expensive-parserfunction-category": "Mga pahina na igwa nin grabe kadakulon na mga ekspensibong programang pambaranga sa punksyon nin mga pag-aapod",
        "revdelete-no-file": "An sagunson na pinaghayag dae tabi eksistido.",
        "revdelete-show-file-confirm": "Segurado ka tabi na gusto mo matanaw sarong pinagpurang pagbabago kan sagunson \"<nowiki>$1</nowiki>\" poon $2 sa $3?",
        "revdelete-show-file-submit": "Iyo tabi",
+       "revdelete-selected-text": "{{PLURAL:$1|Selected revision|Mga napiling rebisyon}} kan [[:$2]]:",
        "logdelete-selected": "{{PLURAL:$1|Selected log event|Mga piniling talaan kan mga pangyayari}}:",
        "revdelete-confirm": "Pakikumpirma tabi na ika tuyong gumibo kaini, na saimong naintindihan an mga konsekuwensiya, asin ta ika pinaghihimo ini na uyon sa [[{{MediaWiki:Policy-url}}|an palisiya]].",
        "revdelete-suppress-text": "An paglulubog dapat '''sana''' magagamit para sa minasunod na mga kaso:\n*Potensiyal na libeloso an impormasyon\n*Bakong angay an personal na impormasyon\n*:''mga estada nin ini-erokan asin mga numero kan telepono, nasyunal na numero nin kabistohan, asin iba pa.''",
        "mergehistory-empty": "Mayong mga pagbabago na puwedeng mapagtiripon.",
        "mergehistory-done": "$3 {{PLURAL:$3|pagbabago|mga pagbabago}} sa $1 matrayumpong napagtiripon na magin [[:$2]].",
        "mergehistory-fail": "Dae tabi makayanan na makapaghimo nin historiyang pagtiripon, tabi pakihiling giraray an pahina asin parametro kan oras.",
+       "mergehistory-fail-bad-timestamp": "Imbalido an timestamp.",
+       "mergehistory-fail-invalid-source": "Imbalido an ginikanang pahina.",
+       "mergehistory-fail-invalid-dest": "Imbalido an papadumanan na pahina.",
        "mergehistory-no-source": "Gikanang pahina $1 bakong eksistido.",
        "mergehistory-no-destination": "Destinasyong pahina $1 bakong eksistido.",
        "mergehistory-invalid-source": "Gikanang pahina kaipuhan magin saro na balidong titulo.",
        "diff-multi-manyusers": "({{PLURAL:$1|Sarong intermediate na pagbabago|$1 mga intermediate na mga pagbabago}} na sobra sa $2 {{PLURAL:$2|paragamit|mga paragamit}} dae pinaghahayag)",
        "difference-missing-revision": "{{PLURAL:$2|sarong rebisyon|$2 mga rebisyon}} kaining diperensiya ($1) {{PLURAL:$2|na iyo an|kaidto na iyo an}} dae nanagboan.\n\nIni pirmihan na pinagkakausa sa paagi nin pagsusunod nin luwas sa petsang diff na kasugponan pasiring sa sarong pahina na pinagpura na.\nAn mga detalye mapuwedeng matatagboan sa [{{fullurl:{{#Special:Log}}/delete|page={{FULLPAGENAMEE}}}} talaan kan pinagpuraan].",
        "searchresults": "Resulta kan paghahánap",
+       "search-filter-title-prefix-reset": "Maghanap sa gabos na pahina",
        "searchresults-title": "Resulta kan paghahanap para sa \"$1\"",
        "titlematches": "Angay an título kan artíkulo",
        "textmatches": "Angay an teksto nin páhina",
        "notextmatches": "Mayong ángay na teksto nin páhina",
        "prevn": "an nakaagi{{PLURAL:$1|$1}}",
        "nextn": "an masunód{{PLURAL:$1|$1}}",
+       "prev-page": "← nakaaging pahina",
+       "next-page": "sunod na pahina →",
        "prevn-title": "Dati $1 {{PLURAL:$1|resulta|mga resulta}}",
        "nextn-title": "Sunod $1  {{PLURAL:$1|resulta|mga resulta}}",
        "shown-title": "Ipahiling $1  {{PLURAL:$1|resulta|mga resulta}} sa kada pahina",
        "search-result-category-size": "{{PLURAL:$1|1 miyembro|$1 mga miyembro}} ({{PLURAL:$2|1 subkategorya|$2 mga subkategorya}}, {{PLURAL:$3|1 sagunson|$3 mga sagunson}})",
        "search-redirect": "(panukdong hali sa $1)",
        "search-section": "(Seksyon $1)",
+       "search-category": "(kategorya $1)",
        "search-file-match": "(minatugma sa nilalaog kan saguson)",
        "search-suggest": "Boot mong ipakahulugan: $1",
+       "search-rewritten": "Nagpahiling nin resulta para sa $1. Naghanap nin bako sa $2.",
        "search-interwiki-caption": "Tugang na mga proyekto",
        "search-interwiki-default": "$1 na mga resulta:",
        "search-interwiki-more": "(dakol pa)",
+       "search-interwiki-more-results": "Dakul pang resulta",
        "search-relatedarticle": "Kauyon",
        "searchrelated": "kauyon",
        "searchall": "gabós",
        "powersearch-togglelabel": "Pamili:",
        "powersearch-toggleall": "Gabos",
        "powersearch-togglenone": "Wara",
+       "powersearch-remember": "Girumdumon an pinili para sa mga susunod na paghanap",
        "search-external": "Panluwas na paghahanap",
        "searchdisabled": "Pigpopogolan mûna an paghanap sa {{SITENAME}}. Mientras tanto, pwede ka man maghanap sa Google. Giromdomon tabî na an mga indise kan laog ninda sa {{SITENAME}} pwede ser na lumâ na.",
        "search-error": "May salang nangyari habang naghahanap:$1",
+       "search-warning": "May salang nangyari habang naghahanap: $1",
        "preferences": "Mga kabòtan",
        "mypreferences": "Mga Kamuyahan ko",
        "prefs-edits": "Bilang kan mga hirá:",
-       "prefsnologintext2": "Tabi man $1 tanganing maikaag an mga kamuyahan nin paragamit.",
+       "prefsnologintext2": "Tabi man maglaog tanganing maikaag an mga kamuyahan nin paragamit.",
        "prefs-skin": "''Skin''",
        "skin-preview": "Tânawon",
        "datedefault": "Mayong kabôtan",
        "prefs-personal": "Pambisto nin parágamit",
        "prefs-rc": "Mga kaaagi pa sanang pagribay",
        "prefs-watchlist": "Pigbabantayan",
+       "prefs-editwatchlist": "Hirahón an pigbabantayan",
+       "prefs-editwatchlist-label": "Baguhon an mga entry sa saimong bantay-listahan:",
+       "prefs-editwatchlist-edit": "Hilingon asin magtanggal nin mga titulo sa saimong bantay-listahan",
+       "prefs-editwatchlist-raw": "Liwaton an hilaw na bantay-listahan",
+       "prefs-editwatchlist-clear": "Linigon an bantay-listahan",
        "prefs-watchlist-days": "Mga aldaw na ipahiling sa batay-listahan:",
        "prefs-watchlist-days-max": "Maksimum $1 {{PLURAL:$1|aldaw|mga aldaw}}",
        "prefs-watchlist-edits": "Máximong número nin pagbabâgo na ipapahiling sa pinadakulang lista nin pigbabantayan:",
        "prefs-watchlist-token": "Token sa Bantay-listahan:",
        "prefs-misc": "Lain",
        "prefs-resetpass": "Liwaton an sekretong panlaog",
-       "prefs-changeemail": "Liwaton an e-surat na adres",
+       "prefs-changeemail": "Ribayan an e-koreong address",
        "prefs-setemail": "Tuytuyon an e-surat na adres",
        "prefs-email": "E-surat na mga pagpipilian",
        "prefs-rendering": "Hitsurahon",
        "restoreprefs": "Balikon an gabos na panugmad na mga panuytoy (sa gabos na mga seksyon)",
        "prefs-editing": "Pighihira",
        "searchresultshead": "Hanápon",
-       "stub-threshold": "Kasagkoran kan <a href=\"#\" class=\"stub\">takod kan tambô</a> pigpopormato:",
+       "stub-threshold": "Kasagkoran kan <a href=\"#\" class=\"stub\">takod kan tambô</a> pigpopormato ($1):",
+       "stub-threshold-sample-link": "ehemplo",
        "stub-threshold-disabled": "Pinagpundo",
        "recentchangesdays": "Mga aldáw na ipapahilíng sa mga nakakaági pa sanáng pagbabàgó:",
        "recentchangesdays-max": "Maksimum $1 {{PLURAL:$1|aldaw|mga aldaw}}",
-       "recentchangescount": "Numero kan mga pagliliwat na ipapahiling na pirmihan:",
-       "prefs-help-recentchangescount": "Kabali kaini an dae pa nahaloy na mga kaliwatan, mga historiyang pahina, asin mga talaan.",
+       "recentchangescount": "Numero kan mga pagliliwat na ipapahiling sa pinakabago, historia kan pahina, paglaog:",
+       "prefs-help-recentchangescount": "Maksimum na numero: 1000",
        "prefs-help-watchlist-token2": "Ini an sikretong susi sa bahugan kan web sa saimong bantay-listahan.\nAn siisay man na makaaram kaini makakapagbasa kan saimong bantay-listahan, kaya dae mo ipagheras ini.\n[[Special:ResetTokens|I-klik digde kun kaipo mong baguhon it]].",
        "savedprefs": "Itinagama na an mga kabôtan mo.",
        "timezonelegend": "Pan-oras na sona:",
        "default": "pwestong normal",
        "prefs-files": "Mga dokumento",
        "prefs-custom-css": "Kustombreng CSS",
+       "prefs-custom-json": "Custom JSON",
        "prefs-custom-js": "Kustombreng JavaScript",
-       "prefs-common-config": "Pinagheras na CSS/JavaScript para sa gabos na mga kalapatan:",
+       "prefs-common-config": "Pinagheras na CSS/JSON/JavaScript para sa gabos na mga kalapatan:",
        "prefs-reset-intro": "Ika makakagamit kaining pahina tanganing ilapat giraray an saimong mga kabotan sa panugmad kan sayt.\nIni dae tabi matitingkog.",
        "prefs-emailconfirm-label": "Kumpirmasyon sa E-koreo",
        "youremail": "E-surat:",
        "username": "{{GENDER:$1|Pangaran nin paragamit}}:",
        "prefs-memberingroups": "{{GENDER:$2|Miyembro}} kan {{PLURAL:$1|grupo|mga grupo}}:",
+       "group-membership-link-with-expiry": "$1 (hanggan $2)",
        "prefs-registration": "Rehistrasyong oras:",
        "yourrealname": "Totoong pangaran:",
        "yourlanguage": "Tataramon:",
        "gender-female": "Siya nagliliwat nin mga pahina sa wiki",
        "prefs-help-gender": "An panuytoy kaining kamuyahan opsyonal.\nAn panuklob minagamit kan saiyang kahalagahan sa pagpanungod saimo asin sa pagsambit saimo sa iba pa na naggagamit nin maninigong gramatikal na kabolosan.\nIning impormasyon isasapubliko.",
        "email": "E-koreo",
-       "prefs-help-realname": "Opsyonal an totoong pangaran asin kun itatao mo ini, gagamiton ini yangarig an mga sinurat mo maatribuir saimo.",
+       "prefs-help-realname": "Opsyonal an totoong pangaran asin kun itatao mo ini, gagamiton ini tangarig an mga sinurat mo maatribuir saimo.",
        "prefs-help-email": "An e-surat na adres sarong opsyonal, alagad ini kinakaipohan para sa pagtuytoy otro kan sekretong panlaog, kun ika malingaw kan saimong sekretong panlaog.",
        "prefs-help-email-others": "Ika kan man pumili na magtugot sa iba na makontak ka sa e-surat sa paagi nin sarong kasugponan na yaon sa saimong pahina nin paragamit o olay.\nAn saimong e-surat na adres dae ipagbuyagyag kunsoarin na an ibang paragamit makontak saimo.",
        "prefs-help-email-required": "Kaipuhan an e-koreo.",
        "prefs-dateformat": "Pampetsang pormat",
        "prefs-timeoffset": "Pan-oras na tapal",
        "prefs-advancedediting": "Pankagabsan na mga Pagpipilian",
+       "prefs-developertools": "Mga kagamitan nin Paragibo",
        "prefs-editor": "Paraliwat",
        "prefs-preview": "Patânaw",
        "prefs-advancedrc": "Pangenot na mga pagpipilian",
        "prefs-advancedwatchlist": "Abantidong mga pagpipilian",
        "prefs-displayrc": "Ihayag an mga pagpipilian",
        "prefs-displaywatchlist": "Ipahiling ang mga pagpipilian",
+       "prefs-changesrc": "Ipinahiling an mga pagbabago",
+       "prefs-changeswatchlist": "Ipinahiling an mga pagbabago",
+       "prefs-pageswatchlist": "Mga binabantayan na mga pahina",
        "prefs-tokenwatchlist": "Paduos",
        "prefs-diffs": "Diffs",
        "prefs-help-prefershttps": "Ining kamuyahan magkaka-epekto sa masunod mong paglaog.",
        "prefs-tabs-navigation-hint": "Pantama: Ika makakagamit nin wala asin too na pansusing pana tanganing magnabigar sa tahaw kan mga tanda na yaon sa listahan nin mga panandaan.",
-       "userrights": "Pagmaneho kan mga derecho nin paragamit",
-       "userrights-lookup-user": "Magmaného kan mga grupo nin parágamit",
+       "userrights": "Karapatan kan paragamit",
+       "userrights-lookup-user": "Magpili nin parágamit",
        "userrights-user-editname": "Ilaog an pangaran kan parágamit:",
-       "editusergroup": "Hirahón an mga Grupo kan Parágamit",
+       "editusergroup": "Ipahiling an mga Grupo kan Parágamit",
        "editinguser": "Sinasanglian an mga karapatan kan paragamit na si {{GENDER:$1|paragamit}} <strong>[[User:$1|$1]]</strong> $2",
-       "userrights-editusergroup": "Hirahón an mga grupo kan parágamit",
-       "saveusergroups": "Itagama an mga Grupo nin Páragamit",
+       "userrights-editusergroup": "Hirahón an mga {{GENDER:$1|grupo kan parágamit}}",
+       "userrights-viewusergroup": "Hilingon {{GENDER:$1|paragamit}} an grupo",
+       "saveusergroups": "Itagama an {{GENDER:$1|mga Grupo nin Páragamit}}",
        "userrights-groupsmember": "Myembro kan:",
        "userrights-groupsmember-auto": "Implisitong miyembro kan:",
        "userrights-groups-help": "Ika puwedeng magbago kan mga grupo na kinabalihan kaining paragamit:\n*An natsekan na kahon minapasabot na an paragamit kabali sa grupong yan.\n*An mayong tsek na kahon minapasabot na an paragamit bakong kabali sa grupong yan.\n* A * minapahiwatig na ika dae puwedeng makapaghale kan grupo kun naidagdag mo na ini, or vice versa.",
        "userrights-nodatabase": "An datos-sarayan $1 bakong eksistido o bakong lokal.",
        "userrights-changeable-col": "Mga grupo na mapuwede mong baguhon",
        "userrights-unchangeable-col": "Mga grupo na dae mo mapuwedeng baguhon",
+       "userrights-expiry-current": "Mapalso sa $1",
+       "userrights-expiry-none": "Dai napapalso",
+       "userrights-expiry": "Mápasó:",
+       "userrights-expiry-existing": "Eksistidong oras nin pagpalso: $3, $2",
+       "userrights-expiry-othertime": "Ibang oras:",
+       "userrights-expiry-options": "1 aldaw:1 aldaw, 1 semana:1 semana, 1 bulan:1 bulan, 3 bulan:3 bulan, 6 bulan:6 bulan, 1 taon:1 taon",
        "userrights-conflict": "Kumplikto sa mga kaliwatan nin mga katanosan kan paragamit! Tabi man pakirikisa asin kumpirmaron an saimong mga kaliwatan.",
        "group": "Grupo:",
        "group-user": "Mga Paragamit",
        "group-autoconfirmed-member": "{{GENDER:$1|auto-kumpirmadong paragamit}}",
        "group-bot-member": "{{GENDER:$1|bot}}",
        "group-sysop-member": "{{GENDER:$1|administrador}}",
+       "group-interface-admin-member": "{{GENDER:$1|administrador kan interface}}",
        "group-bureaucrat-member": "{{GENDER:$1|burokrata}}",
        "group-suppress-member": "{{GENDER:$1|tagapagmato}}",
        "grouppage-user": "{{ns:project}}:Mga Paragamit",
        "grouppage-autoconfirmed": "{{ns:project}}:Mga enseguidang nakonpirmar na parágamit",
        "grouppage-bot": "{{ns:project}}:Mga bot",
        "grouppage-sysop": "{{ns:project}}:Mga tagamató",
+       "grouppage-interface-admin": "{{ns:project}}:Mga administrador kan interface",
        "grouppage-bureaucrat": "{{ns:project}}:Mga bureaucrat",
        "grouppage-suppress": "{{ns:project}}:Tagapagmato",
        "right-read": "Magbasa kan mga pahina",
        "right-move": "Ibalyo an mga pahina",
        "right-move-subpages": "Ibalyo an mga pahina kaiba an saindang mga sub-pahina",
        "right-move-rootuserpages": "Ibalyo an ugat nin mga pahina kan paragamit",
+       "right-move-categorypages": "Ilipat an mga pahina kan kategorya",
        "right-movefile": "Ibalyo an mga sagunson",
        "right-suppressredirect": "Dae tabi magmukna nin paotrong direksyon gikan sa ginikanang mga pahina kunsoarin magbabalyo nin mga pahina",
        "right-upload": "Ipagkarga an mga sagunson (file)",
        "right-browsearchive": "Hanapon an pinagpurang mga pahina",
        "right-undelete": "Dae puraon an pahina",
        "right-suppressrevision": "Hilngon otro asin balikon an mga pagbabagong itinago gikan sa mga administrador",
+       "right-viewsuppressed": "Hilingon an mga rebisyon na nakatago sa iba pang mga paragamit",
        "right-suppressionlog": "Tanawon an pribadong mga talaan",
        "right-block": "Kubkubon an ibang mga paragamit sa pagliliwat",
        "right-blockemail": "Kubkubon an paragamit na makapagpadara nin e-koreo",
        "right-editusercss": "Liwaton an CSS na mga sagunson kan ibang mga paragamit",
        "right-edituserjson": "Liwaton an JSON na mga sagunson kan ibang mga paragamit",
        "right-edituserjs": "Liwaton an JavaScript na mga sagunson kan ibang mga paragamit",
+       "right-editsitecss": "Liwaton an buong CSS",
+       "right-editsitejson": "Liwaton an buong JSON",
+       "right-editsitejs": "Liwaton an buong JavaScript",
        "right-editmyusercss": "Liwaton an saimong sadireng paragamit na sagunson sa CSS",
        "right-editmyuserjson": "Liwaton an saimong sadireng paragamit na sagunson sa JSON",
        "right-editmyuserjs": "Liwaton an saimong sadireng paragamit na sagunson sa JavaScript",
        "right-siteadmin": "Kandaduhan asin dae pagkandaduhan an datos-sarayan",
        "right-override-export-depth": "Eksportaron an mga pahina kabali na an pinagkilyawan na mga pahina sagkod sa rarom na 5",
        "right-sendemail": "Magpadara nin e-koreo sa ibang mga paragamit",
-       "grant-editmycssjs": "Liwaton an saimong paragamit CSS/JavaScript",
-       "grant-editmyoptions": "Liwaton an saimong paragamit na mga kamuyahan",
+       "right-managechangetags": "Maggibo asin maghali [[Special:Tags|tags]]",
+       "grant-group-page-interaction": "Interaksyon sa mga pahina",
+       "grant-group-file-interaction": "Interaksyon sa mga medya",
+       "grant-group-watchlist-interaction": "Interaksyon sa saimong bantay-listahan",
+       "grant-group-email": "Ipadara an e-surat",
+       "grant-blockusers": "Bagaton asin maghali nin pagkabagat kan mga paragamit",
+       "grant-createaccount": "Magmukna nin panindog",
+       "grant-createeditmovepage": "Maggibo, magliwat asin maglipat nin pahina",
+       "grant-editmycssjs": "Liwaton an saimong paragamit CSS/JSON/JavaScript",
+       "grant-editmyoptions": "Liwaton an saimong paragamit na mga kamuyahan asin configurayson kan JSON",
+       "grant-editmywatchlist": "Liwaton an saimong bantay-listahan",
+       "grant-editpage": "Liwaton an mga yaun nang pahina",
+       "grant-editprotected": "Liwaton an mga napoprotektaran na pahina",
+       "grant-protect": "Magprotekta asin magtanggal nin proteksyon sa mga pahina",
+       "grant-sendemail": "Magpadara nin e-koreo sa ibang mga paragamit",
+       "grant-uploadeditmovefile": "Magkarga, magribay asin maglipat nin mga sagunson",
+       "grant-uploadfile": "Magkarga nin bagong mga sagunson",
        "grant-viewdeleted": "Tanawon an pinagpurang mga sagunson asin pahina",
+       "grant-viewmywatchlist": "Tanawon an saimong bantay-listahan",
        "newuserlogpage": "Paragamit na talaan nin pagmukna",
        "newuserlogpagetext": "Ini an talaan kan mga pagmukna nin paragamit.",
        "rightslog": "Usip nin derechos nin paragamit",
        "rightslogtext": "Ini an historial kan mga pagbabâgo sa mga derecho nin parágamit.",
        "action-read": "basaha ining pahina",
        "action-edit": "liwatón ining pahina",
-       "action-createpage": "magmukna nin mga pahina",
+       "action-createpage": "Muknaon ining pahina",
        "action-createtalk": "Magmukna nin mga pahina sa orolayan",
        "action-createaccount": "Muknaon ining panindog kan paragamit",
+       "action-history": "Hilingon an historiya kaining pahina",
        "action-minoredit": "marakahan ining pagliwat bilang menor",
        "action-move": "ibalyo ining pahina",
        "action-move-subpages": "ibalyo ining pahina, asin kaiba an mga sub-pahina",
        "action-move-rootuserpages": "ibalyo an ugat kan mga pahina nin mga paragamit",
+       "action-move-categorypages": "Ilipat an mga pahina kan kategorya",
        "action-movefile": "ibalyo ining sagunson",
        "action-upload": "ikarga ining mga sagunson",
        "action-reupload": "sampawan ining eksistidong sagunson",
        "action-delete": "puraon ining pahina",
        "action-deleterevision": "puraon ining pagbabago",
        "action-deletedhistory": "tanawon an pinagpurang historiya kaining pahina",
+       "action-deletedtext": "hilingon an pinura na mga rebisyon",
        "action-browsearchive": "hanapon an pinagpurang mga pahina",
        "action-undelete": "dae pagpuraon ining pahina",
        "action-suppressrevision": "hilngon otro asin ibalik ining pinagtagong pagbabago",
        "action-userrights-interwiki": "liwaton an paragamit na mga karapatan kan mga paragamit nin ibang wikis",
        "action-siteadmin": "ikandado o dae ikandado an datos-sarayan",
        "action-sendemail": "magpadara nin mga e-koreo",
+       "action-editmyoptions": "liwaton an sadiri mong mga kamuyahan",
        "action-editmywatchlist": "liwaton an saimong bantay-listahan",
        "action-viewmywatchlist": "tanawon an saimong bantay-listahan",
        "action-viewmyprivateinfo": "tanawon an saimong pribadong impormasyon",
        "action-editmyprivateinfo": "liwaton an saimong pribadong impormasyon",
+       "action-purge": "tanggalon an pahinang ini",
+       "action-apihighlimits": "maggamit nin harahalangkaw na sagkodan sa mga kahaputan kan API",
+       "action-autoconfirmed": "Dai magin apektado sa paagi kan rata na nakabase sa IP na mga sagkodan",
+       "action-bigdelete": "Puraon an mga pahina na igwang darakulang mga historiya",
+       "action-blockemail": "bagaton an paragamit na makapagpadara nin e-koreo",
+       "action-bot": "Pagtratuhon bilang awtomatikong proseso",
+       "action-editprotected": "Liwaton an mga pahina na protektado bilang \"{{int:protect-level-sysop}}\"",
+       "action-editsemiprotected": "Liwaton an mga pahina na protektado bilang \"{{int:protect-level-autoconfirmed}}\"",
+       "action-editinterface": "Liwaton an interface kan paragamit",
+       "action-editusercss": "Liwaton an CSS na mga sagunson kan ibang mga paragamit",
+       "action-edituserjson": "Liwaton an JSON na mga sagunson kan ibang mga paragamit",
+       "action-edituserjs": "Liwaton an JavaScript na mga sagunson kan ibang mga paragamit",
+       "action-editsitecss": "Liwaton an buong CSS",
+       "action-editsitejson": "Liwaton an buong JSON",
+       "action-editsitejs": "Liwaton an buong JavaScript",
+       "action-editmyusercss": "Liwaton an saimong sadireng paragamit na sagunson sa CSS",
+       "action-editmyuserjson": "Liwaton an saimong sadiring paragamit na sagunson sa JSON",
+       "action-editmyuserjs": "Liwaton an saimong sadiring paragamit na sagunson sa JavaScript",
+       "action-viewsuppressed": "Hilingon an mga rebisyon na nakatago dawa na isay na paragamit",
+       "action-hideuser": "Kubkubon an pangaran nin paragamit, itago ini sa publiko",
+       "action-ipblock-exempt": "Sampawan an pangubkob kan IP, awtomatikong-kubkob asin panhalawig na kubkob",
+       "action-unblockself": "Halion an bagat sa sadiri",
+       "action-noratelimit": "Dae magin apektado sa paagi kan rata nin mga sagkodan",
+       "action-reupload-own": "Patungan an eksistido nang mga pahina na ipinagkarga sa paagi mo",
+       "action-nominornewtalk": "Dae gayod nagkaigwa nin menor na pagliwat sa mga pahina nin orolayan minasulpang nin bunyaw kan bagong mga mensahe",
+       "action-markbotedits": "Markahan an pinagbalik na mga niliwat bilang bot na panliwat",
+       "action-patrolmarks": "Tanawon an pinakahuring mga pagbabago na markadong patrol",
+       "action-override-export-depth": "Eksportaron an mga pahina kabali na an pinagkilyawan na mga pahina sagkod sa rarom na 5",
+       "action-suppressredirect": "Dae tabi magmukna nin paotrong direksyon gikan sa ginikanang mga pahina kunsoarin magbabalyo nin mga pahina",
        "nchanges": "$1 {{PLURAL:$1|kaliwatan|mga kaliwatan}}",
        "enhancedrc-since-last-visit": "$1 {{PLURAL:$1|poon kaidtong huring bisita}}",
        "enhancedrc-history": "historiya",
        "recentchanges-legend-heading": "<strong>Kabalaynan:</strong>",
        "recentchanges-legend-newpage": "{{int:recentchanges-label-newpage}} (hilngon man [[Special:NewPages|listahan kan mga baguhong pahina]])",
        "recentchanges-legend-plusminus": "(''±saro-duwa-tolo'')",
+       "recentchanges-submit": "Ipahiling",
+       "rcfilters-tag-remove": "Halion '$1'",
+       "rcfilters-group-results-by-page": "Grupong resulta ayon sa pahina",
+       "rcfilters-activefilters": "Mga aktibong pangsara",
+       "rcfilters-activefilters-hide": "Itago",
+       "rcfilters-activefilters-show": "Ipahiling",
+       "rcfilters-days-show-days": "$1 {{PLURAL:$1|aldaw|mga aldaw}}",
+       "rcfilters-days-show-hours": "$1 {{PLURAL:$1|oras|mga oras}}",
+       "rcfilters-quickfilters": "Itagama an saraan",
+       "rcfilters-quickfilters-placeholder-title": "Mayo pang pangsara na nakatagama",
+       "rcfilters-savedqueries-defaultlabel": "Itagama an saraan",
+       "rcfilters-savedqueries-rename": "Pangaranan liwat",
+       "rcfilters-savedqueries-remove": "Puraon",
+       "rcfilters-savedqueries-new-name-label": "Pangaran",
+       "rcfilters-savedqueries-apply-label": "Magmukna nin pangsara",
+       "rcfilters-savedqueries-cancel-label": "Kanselaron",
+       "rcfilters-clear-all-filters": "Klaradohon an mga saraan",
+       "rcfilters-search-placeholder-mobile": "Mga saraan",
+       "rcfilters-filterlist-title": "Mga pangsara",
+       "rcfilters-highlightmenu-title": "Magpili nin kulay",
+       "rcfilters-filterlist-noresults": "Mayong pangsara na nahiling",
+       "rcfilters-filter-editsbyself-label": "Mga kaliwatan mo",
+       "rcfilters-filter-editsbyself-description": "An saimong mga kaarambagan",
+       "rcfilters-filter-editsbyother-label": "Mga kaliwatan nin iba",
+       "rcfilters-filter-editsbyother-description": "Gabos na kaliwatan pwera kan saimo.",
+       "rcfilters-filter-user-experience-level-registered-label": "Rehistrado",
+       "rcfilters-filter-user-experience-level-unregistered-label": "Dai rehistrado",
+       "rcfilters-filter-user-experience-level-experienced-label": "Mga eksperyensyadong paragamit",
+       "rcfilters-filter-bots-label": "Bot",
+       "rcfilters-filter-reviewstatus-unpatrolled-label": "Dai patrolado",
+       "rcfilters-filter-minor-label": "Sadit na kaliwatan",
+       "rcfilters-filtergroup-watchlist": "Mga binabantayan na mga pahina",
+       "rcfilters-filter-watchlist-watched-label": "Nasa bantay-listahan",
+       "rcfilters-filter-watchlist-notwatched-label": "Mayo sa bantay-listahan",
+       "rcfilters-filter-watchlistactivity-unseen-label": "Dai nahiling na pagbabago",
+       "rcfilters-filtergroup-changetype": "Klase kan pagbago",
+       "rcfilters-filter-pageedits-label": "Pagliwat sa pahina",
+       "rcfilters-filter-newpages-label": "Pagmukna kan pahina",
+       "rcfilters-filter-newpages-description": "Mga kaliwatan na nagibo nin bagong pahina",
+       "rcfilters-filter-categorization-label": "Pagbago sa kategorya",
+       "rcfilters-filtergroup-lastrevision": "Sa ngunyan na rebisyon",
+       "rcfilters-filter-lastrevision-label": "Sa ngunyan na rebisyon",
+       "rcfilters-filter-previousrevision-label": "Bako an pinakabagong rebisyon",
        "rcnotefrom": "Sa ibaba {{PLURAL:$5|iyo an kaliwatan|an mga kaliwatan}} poon kan <strong>$3, $4</strong> (sagkod <strong>$1</strong> an pinapahiling).",
+       "rclistfromreset": "Liwaton an pagpili kan petsa",
        "rclistfrom": "Ipahiling an baguhon na mga kaliwatan magpoon kan $3 $2",
        "rcshowhideminor": "$1 saradit na mga pagliwat",
        "rcshowhideminor-show": "Ipatanaw",
        "uploaderror": "Salâ an pagkarga",
        "upload-recreate-warning": "'''Patanid tabi: An sagunson sa pangaran kaini pinagpura o pinagbalyo na tabi.'''\n\nAn talaan kan pagkapura asin pagkabalyo para sa pahinang ini yaon digde para sa saimong konbenyensiya:",
        "uploadtext": "Gamita an porma sa ibaba tanganing makapagkarga nin mga sagunson.\nPara hilngon o hanapon an dati nang pinagkargang mga sagunson, magduman tabi sa [[Special:FileList|listahan kan pinagkargang mga sagunson]], mga pagkarga asin pagkarga otro pinagtala man sa [[Special:Log/upload|talaan nin pagkakarga]], mga pinagpura na yaon sa [[Special:Log/delete|talaan nin pagkapura]].\n\nSa pagbali nin sarong sagunson sa sarong pahina, gamita tabi an takod kan saro sa mga minasunod na mga porma:\n* '''<code><nowiki>[[</nowiki>{{ns:file}}<nowiki>:File.jpg]]</nowiki></code>''' sa paggamit kan bilog na bersyon kan sagunson\n* '''<code><nowiki>[[</nowiki>{{ns:file}}<nowiki>:File.png|200px|thumb|left|alt text]]</nowiki></code>''' sa paggamit kan 200 pixel na lawig kan pagkakua sa sarong kahon na yaon sa parteng wala nin gaygayan na yaon an 'alt text' bilang deskripsyon\n* '''<code><nowiki>[[</nowiki>{{ns:media}}<nowiki>:File.ogg]]</nowiki></code>''' para sa direktang nakakatakod sa sagunson na dae pinagpapahiling na sarong sagunson",
-       "upload-permitted": "Pinagtutugutang mga tipo nin sagunson: $1",
-       "upload-preferred": "Pinagpapaurog na mga tipo nin sagunson: $1",
-       "upload-prohibited": "Pinagbabawal na mga tipo nin sagunson: $1.",
+       "upload-permitted": "Pinagtutugutang mga tipo nin sagunson {{PLURAL:$2|type|types}}: $1",
+       "upload-preferred": "Pinagpapaurog na mga tipo nin sagunson {{PLURAL:$2|type|types}}: $1",
+       "upload-prohibited": "Pinagbabawal na mga tipo nin sagunson {{PLURAL:$2|type|types}}: $1.",
        "uploadlogpage": "Ikarga an katalaanan",
        "uploadlogpagetext": "Yaon sa ibaba an sarong listahan kan dae pa sanang nahahaloy na pinagkargang mga sagunson.\nHilngon tabi an [[Special:NewFiles|galleriya kan mga bagong sagunson]] para sa mas biswal na lantawon.",
        "filename": "Pangaran kan dokumento",
        "largefileserver": "Mas dakula an ''file'' sa pigtotogotan na sokol kan ''server''.",
        "emptyfile": "Garo mayong laog an ''file'' na kinarga mo. Pwede ser na salâ ining tipo nin ''filename''. Isegurado tabî kun talagang boot mong ikarga ining ''file''.",
        "windows-nonascii-filename": "Ining wiki dae tabi nagsusuporta kan mga pangaran kan sagunson na igwang espesyal na mga karakter.",
-       "fileexists": "Igwa nang ''file'' na may parehong pangaran sa ini, sosogon tabî an <strong>[[:$1]]</strong> kun dai ka seguradong ribayan ini.\n[[$1|thumb]]",
+       "fileexists": "Igwa nang ''file'' na may parehong pangaran sa ini, sosogon tabî an <strong>[[:$1]]</strong> kun {{GENDER:|ika}} dai ka seguradong ribayan ini.\n[[$1|thumb]]",
        "filepageexists": "An pahinang pandeskripsyon kaining sagunson pinagmukna na tabi sa <strong>[[:$1]]</strong>, alagad mayong sagunson na igwa kaining pangaran sa ngunyan nag-eeksister.\nAn sumaryong na saimong ipinaglaog dae minaluwas sa pahina kan deskription.\nTanganing gibohon na an saimong sumaryo magluwas duman, kaipohan mong manwal na pagliliwat kaini.\n[[$1|thumb]]",
        "fileexists-extension": "May ''file'' na may parehong pangaran: [[$2|thumb]]\n* Pangaran kan pigkakargang ''file'': <strong>[[:$1]]</strong>\n* Pangaran kan yaon nang ''file'': <strong>[[:$2]]</strong>\nMagpili tabî nin ibang pangaran.",
        "fileexists-thumbnail-yes": "An ''file'' garo ladawan kan pinasadit ''(thumbnail)''. [[$1|thumb]]\nSosogon tabî an ''file'' <strong>[[:$1]]</strong>.\nKun an sinosog na ''file'' iyo an parehong ladawan na nasa dating sokol, dai na kaipuhan magkarga nin iba pang retratito.",
        "upload-too-many-redirects": "An kilyawan nagkaigwa nin kadakol na mga kaliwatan",
        "upload-http-error": "Sarong HTTP na kasalaan an nangyari: $1",
        "upload-copy-upload-invalid-domain": "Pangungupkop nin kopya bakong puwede gikan sa kinasakupan kaini.",
+       "upload-form-label-infoform-date": "Petsa",
        "backend-fail-stream": "Dae maipakupsit an sagunson $1.",
        "backend-fail-backup": "Dae makapagtago nin saro pang kopya an sagunson $1.",
        "backend-fail-notexists": "An sagunson na $1 bakong eksistido.",
        "backend-fail-read": "Dae makakabasa nin sagunson $1.",
        "backend-fail-create": "Dae makakapagsurat nin sagunson $1.",
        "backend-fail-maxsize": "Dae makakapagsuat nin sagunson $1 nin huli ta ini grabe kadakula nin {{PLURAL:$2|sarong byte|$2 bytes}}.",
-       "backend-fail-readonly": "An sarayan na panampad \"$1\" yaon sa estado na basahon-sana. An rason na pinagtao iyo na: \"''$2''\"",
+       "backend-fail-readonly": "An sarayan na panampad \"$1\" yaon sa estado na basahon-sana. An rason na pinagtao iyo na: <em>$2</em>",
        "backend-fail-synced": "An sagunson \"$1\" yaon sa estado na bakong konsistido sa laog kan mga panampad na sarayan",
        "backend-fail-connect": "Dae nakakapagsugpon sa panampad na sarayan \"$1\".",
        "backend-fail-internal": "Sarong bakong bistadong kasalaan an nangyari sa panampad na sarayan \"$1\".",
        "uploadstash-summary": "An pahinang ini minatao nin agihan pasiring sa mga sagunson na ikinarga na (o baya yaon pa sa proseso nin pagkakarga) alagad dae pa naipublisa sa wiki. An mga sagunson na ini bakong hiling sa kiisay man kundi sa paragamit na nagkarga kan mga ini.",
        "uploadstash-clear": "Pinaglinigan na makantidad na mga sagunson",
        "uploadstash-nofiles": "Ika mayo nin mahalagang mga sagunson.",
-       "uploadstash-badtoken": "An paggibo kan aksyon na yan bakong matrayumpo, baka nin huli ta an saimong kredensiyal sa pagliliwat nagpaso na.",
+       "uploadstash-badtoken": "An paggibo kan aksyon na yan bakong matrayumpo, baka nin huli ta an saimong kredensiyal sa pagliliwat nagpaso na. Uliton giraray.",
        "uploadstash-errclear": "An paglilinig kan mga sagunson bakong matrayumpo.",
        "uploadstash-refresh": "Papreskoha otro an listahan kan mga sagunson",
        "invalid-chunk-offset": "Imbalidong tagpas na pampahale",
        "license-header": "Paglisensiya",
        "nolicense": "Mayong pigpilî",
        "license-nopreview": "(Mayong patânaw)",
-       "upload_source_url": " (sarong tama, na bukas sa publikong URL)",
-       "upload_source_file": " (sarong ''file'' sa kompyuter mo)",
+       "upload_source_url": "(an sagunson napili tama, asin bukas sa publikong URL)",
+       "upload_source_file": "(sarong ''file'' sa kompyuter mo)",
        "listfiles-summary": "Ining espesyal na pahina minapahiling kan gabos na ipinagkargang mga sagunson.",
        "listfiles_search_for": "Hanápon an pangaran kan retrato:",
        "imgfile": "dokumento",
        "statistics-files": "Pinagkargang mga sagunson",
        "statistics-edits": "Mga pagliwat sa pahina magpoon pa na an {{SITENAME}} pinagmukna.",
        "statistics-edits-average": "Katahaw kan mga pagliliwat sa kada pahina",
-       "statistics-users": "Rehistrado [[Special:ListUsers|users]]",
+       "statistics-users": "Rehistradong mga paragamit",
        "statistics-users-active": "Mga Aktibong Paragamit",
        "statistics-users-active-desc": "Mga paragamit na may ginibong aksyon sa nakaaging {{PLURAL:$1|aldaw|$1 mga aldaw}}",
        "pageswithprop": "Mga pahina na igwang pahina nin kagrugaring",
        "mostrevisions": "Mga artikulong may pinakadakol na pagpakarháy",
        "prefixindex": "Gabos na mga pahina na igwa nin enotang panigmit",
        "prefixindex-namespace": "Gabos na mga pahina na igwa nin enotang panigmit ($1 espasyong ngaran)",
-       "prefixindex-strip": "Waknison an pangenot na panigmit na yaon sa listahan",
+       "prefixindex-strip": "Itago an panginot na panigmit na yaon sa listahan",
        "shortpages": "Haralìpot na pahina",
        "longpages": "Mga halabang pahina",
        "deadendpages": "Mga pahinang mayong luwasan",
        "protectedpages-indef": "Daeng sagkod na proteksyon sana",
        "protectedpages-cascade": "Mga pasurunod na proteksyon sana",
        "protectedpagesempty": "Mayong pang páhina an napoprotehiran kaining mga parametros.",
+       "protectedpages-performer": "Nagpoprotektang paragamit",
+       "protectedpages-unknown-performer": "Dai bistong paragamit",
        "protectedtitles": "Protektadong mga titulo",
        "protectedtitlesempty": "Mayong mga titulo sa presente an protektado kaining mga parametro.",
        "listusers": "Lista nin paragamit",
        "listusers-editsonly": "Ipahiling sana an mga paragamit na igwang mga pinagliwat",
+       "listusers-temporarygroupsonly": "Ipahiling sana an paragamit na nasa temporaryong grupo kan mga paragamit",
        "listusers-creationsort": "Salansanon sa paagi kan petsa nin pagmukna",
        "listusers-desc": "Salansanon sa paibabang pasurunod",
        "usereditcount": "$1 {{PLURAL:$1|pigliwat|mga pigliwat}}",
        "querypage-disabled": "Ining espesyal na pahina pinagpundo nin huli sa kaggibohang mga rason.",
        "apisandbox": "Kahong-buhangin kan API",
        "apisandbox-api-disabled": "An API dae pinagpagana sa sityong ini.",
-       "apisandbox-intro": "Gamitong ining pahina sa pag-eksperimento kan '''MediaWiki web service API'''.\nKonsultaron an  [https://www.mediawiki.org/wiki/API:Main_page the API documentation] para sa iba pang mga detalye sa paggamit kan API. Ehemplo: [https://www.mediawiki.org/wiki/API#A_simple_example kuahon an laman kan Pangenot na Pahina]. Magpili nin aksyon tanganing hilngon an mga kadagdagan na mga ehemplo.",
+       "apisandbox-intro": "Gamitong ining pahina sa pag-eksperimento kan '''MediaWiki web service API'''.\nKonsultaron an [[mw:API:Main page|the API documentation]] para sa iba pang mga detalye sa paggamit kan API. Ehemplo: [https://www.mediawiki.org/wiki/API#A_simple_example kuahon an laman kan Pangenot na Pahina]. Magpili nin aksyon tanganing hilngon an mga kadagdagan na mga ehemplo.",
        "apisandbox-submit": "Maghimo nin kahagadan",
        "apisandbox-reset": "Klaro",
        "apisandbox-examples": "Ehemplo",
        "apisandbox-results": "Resulta",
        "apisandbox-request-url-label": "Hagad URL:",
-       "apisandbox-request-time": "Hagad oras:$1",
+       "apisandbox-request-time": "Hagad oras: {{PLURAL:$1|$1 ms}}",
        "booksources": "Mga Ginikanan kan libro",
        "booksources-search-legend": "Maghanap para sa mga ginikanang libro",
        "booksources-search": "Hanápon",
        "cachedspecial-viewing-cached-ts": "Ika nakahiling sa sarong pinagsaray na bersyon kaining pahina, na mapuwedeng bakong aktuwal na kumpleto talaga.",
        "cachedspecial-refresh-now": "Hilngon an pinakahuri.",
        "categories": "Mga Kategoriya",
-       "categoriespagetext": "An minasunod {{PLURAL:$1|kategorya na may laog na|mga kategorya na may laog na}} mga pahina o midya.\n[[Special:UnusedCategories|Dae ginamit na mga kategorya]] dae ipinapahiling digde.\nAsin man hilnga an [[Special:WantedCategories|kinakaipong mga kategorya]].",
+       "categoriespagetext": "An minasunod {{PLURAL:$1|kategorya na may laog na|mga kategorya na may laog na}} mga pahina o midya.\nAsin hilnga an [[Special:WantedCategories|kinakaipong mga kategorya]].",
        "categoriesfrom": "Pahilnga an mga kategorya magpoon sa:",
        "deletedcontributions": "Parâon an mga kontribusyon kan parágamit",
        "deletedcontributions-title": "Parâon an mga kontribusyon kan parágamit",
        "emailccsubject": "Kopya kan saimong mensahe sa $1: $2",
        "emailsent": "Naipadará na an e-surat",
        "emailsenttext": "Naipadará na su e-surat mo.",
-       "emailuserfooter": "Ining e-surat ipinadara sa paagi nin $1 pasiring ki $2 kan \"E-surat na paragamit\" na punksyon kan {{SITENAME}}.",
+       "emailuserfooter": "Ining e-surat {{GENDER:$1|ipinadara}} sa paagi nin $1 pasiring ki {{GENDER:$2|$2}} kan \"{{int:emailuser}}\"  na punksyon kan {{SITENAME}}. Kun {{GENDER:$2|ika}} magsimbag sa e-surat, {{GENDER:$2|ang saimong}} e-surat ipapadara sa {{GENDER:$1|orihinal na tagapagpadara}}, ipapahiling an {{GENDER:$2|saimong}} e-surat sa {{GENDER:$1|pinadarahan}}.",
        "usermessage-summary": "Magwawalat nin pansistemang mensahe.",
        "usermessage-editor": "Pansistemang mensahero",
        "watchlist": "Bantay-listahan",
        "watchlistanontext": "Pakipalaog tabi tanganing makapaghiling o makapagliwat sa mga aytem na yaon sa saimong bantay-listahan.",
        "watchnologin": "Mayô sa laog",
        "addwatch": "Idagdag sa bantay-listahan",
-       "addedwatchtext": "Ining pahina \"[[:$1]]\" dinadagdag sa saimong mga [[Special:Watchlist|Bantay-listahan]].\nAn maabot na mga pagbabâgo sa páhinang ini asin sa asosyadong páhina nin olay paglilistahon duman.",
+       "addedwatchtext": "Ining pahina \"[[:$1]]\" dinadagdag sa saimong mga [[Special:Watchlist|Bantay-listahan]].",
        "removewatch": "Halion gikan sa bantay-listahan",
        "removedwatchtext": "An pahina \"[[:$1]]\" pinaghale gikan sa [[Special:Watchlist|saimong bantay-listahan]].",
        "watch": "Bantayán",
        "wlheader-showupdated": "Mga pahina na pinagriliwat poon kaidtong huri kang nagbisita sainda ipinapatanaw na '''mahîbog'''",
        "wlnote": "Sa ibaba kan {{PLURAL:$1|huring pagbabago|mga huring <strong>$1</strong> pagbabago}} sa nakalihis na {{PLURAL:$2|oras|'''$2''' mga oras}}, magpoon pa kan $3, $4.",
        "wlshowlast": "Ipahilíng an nakalihis na $1 na mga oras mga $2 na mga aldaw",
+       "wlshowhideliu": "rehistradong mga paragamit",
        "watchlist-options": "Bantay-listahan na mga pagpipilian",
        "watching": "Pigbabantayan...",
        "unwatching": "Dai pigbabantayan...",
        "deletepage": "Paraon an pahina",
        "confirm": "Kompermaron",
        "excontent": "Ini an dating laog: '$1'",
-       "excontentauthor": "ini an dating laog: '$1' (asin an unikong kontribuidor si '[[Special:Contributions/$2|$2]]')",
+       "excontentauthor": "ini an dating laog: '$1' (asin an unikong kontribuidor si '[[Special:Contributions/$2|$2]]')([[User talk:$2|talk]])",
        "exbeforeblank": "Ini an dating laog bagô blinankohán: '$1'",
        "delete-confirm": "Puraon \"$1\"",
        "delete-legend": "Paraon",
        "delete-edit-reasonlist": "Pagliwat kan mga rason nin pagpupura",
        "delete-toobig": "Ining pahina igwa nin dakulaong historiya sa pagliwat, minasobrang $1 {{PLURAL:$1|rebisyon|mga rebisyon}}.\nAn pagpupura kan nasambit na mga pahina dae pinagtutugot tanganing maiwasan an aksidenteng pagka-antala kan {{SITENAME}}.",
        "delete-warning-toobig": "Ining pahina igwa nin dakulaong historiya sa pagliwat, minasobrang $1 {{PLURAL:$1|rebisyon|mga rebisyon}}.\nAn pagpupura kaini mapuwedeng makapag-antala sa mga operasyon kan datos-sarayan kan {{SITENAME}}; magpadagos tabi na igwang pag-iingat.",
-       "deleting-backlinks-warning": "'''Patanid:''' An ibang mga pahina nakatakod sa pahina na muya mong pagpupuraon.",
+       "deleting-backlinks-warning": "'''Patanid:''' [[Special:WhatLinksHere/{{FULLPAGENAME}}|An ibang mga pahina]] nakatakod sa pahina na muya mong pagpupuraon.",
        "rollback": "Mga paghihira na pabalík",
        "rollbacklink": "pabalikwaton",
        "rollbacklinkcount": "ibalik $1 {{PLURAL:$1|pagliwat|mga pagliwat}}",
        "editcomment": "An sumaryo kan pagliwat: <em>$1</em>.",
        "revertpage": "Ibinalik na mga pagliwat ni [[Special:Contributions/$2|$2]] ([[User talk:$2|talk]]) sagkod sa huring rebisyon ni [[User:$1|$1]]",
        "revertpage-nouser": "Binalikwat na mga pagliliwat kan sarong nakatagong paragamit sa huring rebisyon ni {{GENDER:$1|[[User:$1|$1]]}}",
-       "rollback-success": "Binawî na mga paghirá ni $1; pigbalik sa dating bersyón ni $2.",
+       "rollback-success": "Binawî na mga paghirá ni {{GENDER:$3|$1}};; pigbalik sa dating bersyón ni {{GENDER:$4|$2}}.",
        "sessionfailure-title": "Nagpalyang sesyon",
        "sessionfailure": "Garo may problema sa paglaog mo;\nkinanselár ining aksyón bilang sarong paglikay kontra sa ''session hijacking''.\nPindotón tabî an \"back\" asin ikarga giraray an páhinang ginikanan mo, dangan probarán giraray.",
        "protectlogpage": "Katalaanan nin proteksyon",
        "undeletepagetext": "An minasunod na {{PLURAL:$1|pahina pinagpura na alagad yaon|$1 mga pahina pinagpura na alagad yaraon }} pa man sa arkibo asin puwedeng maipagbalik.\nAn arkibo mapupuwedeng peryodikal na paglilinigan.",
        "undelete-fieldset-title": "Ibalik an mga rebisyon",
        "undeleteextrahelp": "Tanganing maibalik an enterong historiya kan pahina, pabayae na an gabos na mga kahon nin tsek dae pagkaagan asin i-klik mo an '''''{{int:undeletebtn}}'''''.\nTanganing gibohon an piniling restorasyon, i-tsek mo an mga kahon na kinatangudan kan mga rebisyon na ipagbabalik, asin i-klik an '''''{{int:undeletebtn}}'''''.",
-       "undeleterevisions": "$1 {{PLURAL:$1|na pagriribay|na mga pagriribay}} na nakaarchibo",
+       "undeleterevisions": "$1 {{PLURAL:$1|na pagriribay|na mga pagriribay}} na napura",
        "undeletehistory": "Kun saimong ipagbalik an pahina, an gabos nga mga rebisyon ipagbabalik sa historiya.\nKun an baguhon na pahina na igwang kaparehas na ngaran naimukna na poon kan puraon, an ipinagbalik na mga rebisyon minaluwas sa nakaagi nang historiya.",
        "undeleterevdel": "An dae pagpupura dae paggigibohon kun ini magreresulta sa kaibabawan kan pahina o rebisyon kan sagunson bilang parsiyal na pinagpura.\nSa arog na mga kaso, kaipuhan mong haleon an tsek o tagoon an pinakabaguhon na pinagpurang rebisyon.",
        "undeletehistorynoadmin": "Pigparâ na ining péhina. Mahihiling an rason sa epitome sa babâ, kasabay sa mga detalye kan mga parágamit na naghira kaining páhina bago pigparâ. Sa mga administrador sana maipapahiling an mga pagribay sa mismong tekstong ini.",
        "tooltip-invert": "I-tsek ining kahon tanganing tagoon an mga pagbabago sa mga pahina na yaon sa laog kan pinagpiling espasyong-ngaran (asin an asosyado na espasyong-ngaran kun may tsek)",
        "namespace_association": "Asosyado na espasyong-ngaran",
        "tooltip-namespace_association": "I-tsek ining kahon tangani man ibali an olay o subheto na espasyong-ngaran na asosyado sa pinagpili na espasyong-ngaran",
-       "blanknamespace": "(Pangenot)",
+       "blanknamespace": "(Panginot)",
        "contributions": "{{GENDER:$1|Paragamit}} na mga kaambagan",
        "contributions-title": "Mga kontribusyon kan paragamit para sa $1",
        "mycontris": "Mga Kaarambagan",
        "anoncontribs": "Mga Kaarambagan",
        "contribsub2": "Para ki {{GENDER:$3|$1}} ($2)",
+       "contributions-userdoesnotexist": "Paragamit na panindog \"$1\" bako tabing rehistrado.",
        "nocontribs": "Mayong mga pagbabago na nahanap na kapadis sa ining mga criteria.",
        "uctop": "sa ngunyan",
        "month": "Poon bulan (asin mas amay):",
        "year": "Poon taon (asin mas amay):",
+       "date": "Poon bulan (asin mas amay):",
        "sp-contributions-newbies": "Ipahiling an mga kaarambagan kan mga baguhong panindog sana",
        "sp-contributions-newbies-sub": "Para sa mga bàgong account",
        "sp-contributions-newbies-title": "Mga kontribusyon kan paragamit para sa baguhon an mga panindog",
        "sp-contributions-blocklog": "Bagáton an katalaanan",
-       "sp-contributions-deleted": "pinagpurang mga kontribusyon kan paragamit",
+       "sp-contributions-suppresslog": "pinagpurang mga kontribusyon kan {{GENDER:$1|paragamit}}",
+       "sp-contributions-deleted": "pinagpurang mga kontribusyon kan {{GENDER:$1|paragamit}}",
        "sp-contributions-uploads": "mga ikinarga",
        "sp-contributions-logs": "mga tinalaan",
        "sp-contributions-talk": "olayan",
-       "sp-contributions-userrights": "manihamento sa mga karapatan kan paragamit",
+       "sp-contributions-userrights": "manihamento sa mga karapatan kan {{GENDER:$1|paragamit}}",
        "sp-contributions-blocked-notice": "Ining paragamit sa presente pinagbarahan.\nAn pinakahuring entrada sa talaan nin pagbara nakahaya sa ibaba bilang reperensiya:",
        "sp-contributions-blocked-notice-anon": "Ining IP adres sa presente pinagbarahan.\nAn pinakahuring entrada sa talaan nin pagbara nakahaya sa ibaba bilang reperensiya:",
        "sp-contributions-search": "Maghanap nin mga kaarambagan",
        "block": "Barahon an paragamit",
        "unblock": "Haleon an bara kan paragamit",
        "blockip": "Bagáton {{KASARIAN:$1|paragamit}}",
-       "blockiptext": "Gamiton an pormularyo sa babâ para bagaton an pagsurat kan sarong espesipikong IP o ngaran nin parágamit.\nDapat gibohon sana ini para maibitaran vandalismo, asin kompirmi sa [[{{MediaWiki:Policy-url}}|palakaw]].\nMagkaag nin espisipikong rason (halimbawa, magtao nin ehemplo kan mga páhinang rinaot).",
+       "blockiptext": "Gamiton an pormularyo sa babâ para bagaton an pagsurat kan sarong espesipikong IP o ngaran nin parágamit.\nDapat gibohon sana ini para maibitaran vandalismo, asin kompirmi sa [[{{MediaWiki:Policy-url}}|palakaw]].\nMagkaag nin espisipikong rason (halimbawa, magtao nin ehemplo kan mga páhinang rinaot).\nPwede mong bagaton an IP gamit an [https://en.wikipedia.org/wiki/Classless_Inter-Domain_Routing CIDR] syntax;an pinakadakulang sakop na tinutugutan iyo an /$1 para sa IPv4 asin /$2 para sa IPv6.",
        "ipaddressorusername": "direksyon nin IP o gahâ:",
        "ipbreason": "Rason:",
        "ipbreason-dropdown": "*Mga komon na rason sa pagbagat\n** Nagkakaag nin salang impormasyon\n** Naghahalî nin mga laog kan páhina\n** Nagkakaag nin mga takod na ''spam'' kan mga panluwas na ''site''\n** Nagkakaag nin kalokohan/ringaw sa mga pahina\n** Gawî-gawing makatakót/makauyám\n** Nag-aabuso nin mga lain-lain na ''account''\n** Dai akong ngaran nin parágamit",
        "ipb-hardblock": "Pugulan an yaon sa laog na mga paragamit na magliliwat gikan kaining IP adres",
-       "ipbcreateaccount": "Pugulon an pagibo nin kuenta.",
-       "ipbemailban": "Pugolan ining paragamit na magpadara nin e-surat",
+       "ipbcreateaccount": "Magmukna nin panindog",
+       "ipbemailban": "Nagpapadara nin e-surat",
        "ipbenableautoblock": "Enseguidang bagaton an huring direccion nin  IP na ginamit kaining paragamit, asin kon ano pang ibang IP na proprobaran nindang gamiton",
        "ipbsubmit": "Bagáton ining parágamit",
        "ipbother": "Ibang oras:",
        "ipboptions": "2ng oras:2 hours,1ng aldaw:1 day,3ng aldaw:3 days,1ng semana:1 week,2ng semana:2 weeks,1ng bulan:1 month,3ng bulan:3 months,6 na bulan:6 months,1ng taon:1 year,daeng kasagkoran:infinite",
        "ipbhidename": "Tagoon an ngaran nin paragamit gikan sa mga pagliliwat asin mga listahan",
        "ipbwatchuser": "Bantayi ining gamit kan paragamit asin mga pahina nin olayan",
-       "ipb-disableusertalk": "Pugulan ining paragamit na magliliwat kan saiyang sadireng pahina nin olayan habang ini barado",
+       "ipb-disableusertalk": "Magliwat kan saiyang sadiring pahina nin olayan",
        "ipb-change-block": "Barahan-otro an paragamit na igwa kaining mga panuytoy",
        "ipb-confirm": "Kumpirmaron an pagbara",
        "badipaddress": "Dai pwede ining IP",
        "emailblock": "binagát an e-surat",
        "blocklist-nousertalk": "dae makakaliwat kan sadireng pahina nin olayan",
        "ipblocklist-empty": "Mayong laog an lista nin mga binagat.",
-       "ipblocklist-no-results": "Dai nabagat an hinagad na direccion nin IP o ngaran nin paragamit.",
+       "ipblocklist-no-results": "Mayong bagat na nahiling sa IP o ngaran nin paragamit.",
        "blocklink": "bagáton",
        "unblocklink": "haleon an bagat",
        "change-blocklink": "ribayan an bagat",
        "contribslink": "mga ambág",
        "emaillink": "ipadara an e-surat",
-       "autoblocker": "Enseguidang binagat an saimong direccion nin IP ta kaaaging ginamit ini ni \"[[User:$1|$1]]\". An rason nin pagbagat ni $1: \"$2\"",
+       "autoblocker": "Enseguidang binagat an saimong  IP ta kaaaging ginamit ini ni \"[[User:$1|$1]]\". An rason nin pagbagat ni $1: \"$2\"",
        "blocklogpage": "Katalaanan nin bagat",
        "blocklog-showlog": "Ining paragamit dati nang pinagbarahan.\nAn talaan nin pagbara nakahaya sa ibaba bilang reperensiya:",
        "blocklog-showsuppresslog": "Ining paragamit pinagkubkob asin dati nang ipinagtago.\nAn talaan nin pagpaunlok ipinagtao sa ibaba para hilingan.",
        "range_block_disabled": "Pigpopondo an abilidad kan sysop na maggibo nin bagat na hilera.",
        "ipb_expiry_invalid": "Dai pwede ini bilang oras kan pagpasó.",
        "ipb_expiry_temp": "Itinagong pangaran nin paragamit na nagkukubkob dapat na magin permanente.",
-       "ipb_hide_invalid": "Dae nakayanan na untukon ining panindog; ini gayod nagkaigwa nin kadakulon na mga pagliliwat.",
+       "ipb_hide_invalid": "Dae nakayanan na untukon ining panindog; ini gayod nagkaigwa nin kadakulon na mga {{PLURAL:$1|one edit|$1 pagliliwat}}.",
        "ipb_already_blocked": "An \"$1\" pinagkubkob na",
        "ipb-needreblock": "An $1 pinagkubkob na. Gusto mong liwaton an mga panuytoy?",
        "ipb-otherblocks-header": "An ibang {{PLURAL:$1|kubkob|mga kubkob}}",
        "movenotallowedfile": "Ika mayo nin permiso na magbabalyo nin mga sagunson.",
        "cant-move-user-page": "Ika mayo nin permiso na magbabalyo nin mga pahina nin paragamit (laen pa sa mga sub-pahina).",
        "cant-move-to-user-page": "Ika mayo nin permiso na magbabalyo nin pahina paduman sa sa sarong pahina nin paragamit (laen pa sa sub-pahina nin paragamit).",
-       "newtitle": "Sa bàgong titulong:",
+       "newtitle": "Sa bàgong titulo:",
        "move-watch": "Bantayán ining pahina",
        "movepagebtn": "Ibalyó an pahina",
        "pagemovedsub": "Naibalyó na",
        "movenosubpage": "Ining pahina mayo nin mga sub-pahina.",
        "movereason": "Rason:",
        "revertmove": "balikon",
-       "delete_and_move_text": "==Kaipuhan na parâon==\n\nIgwa nang páhina na \"[[:$1]]\". Gusto mong parâon ini tangarig maibalyó?",
+       "delete_and_move_text": "Igwa nang páhina na \"[[:$1]]\". Gusto mong parâon ini tangarig maibalyó?",
        "delete_and_move_confirm": "Iyo, parâon an pahina",
        "delete_and_move_reason": "Pinagpura sa paghimo nin dalan para maibalyo gikan sa \"[[$1]]\"",
        "selfmove": "Pareho an páhinang ginikanan asin destinasyon; dai pwedeng ibalyó an sarong páhina sa sadiri.",
        "move-leave-redirect": "Walaton an sarong panlikwat sa likod",
        "protectedpagemovewarning": "'''Patanid:''' Ining pahina protektado tangani na an mga paragamit sana na igwang administrador na mga pribilihiyo an makakapagbalyo kaini.\nAn pinakahuring entrada sa talaan pinagtao sa ibaba para sa reperensiya:",
        "semiprotectedpagemovewarning": "'''Giromdomon:''' Ining pahina protektado tanganing an mga rehistradong paragamit sana an makakabalyo kaini. \nAn pinakahuring entrada sa talaan pinagtao sa ibaba para sa reperensiya:",
-       "move-over-sharedrepo": "== Yaon nang Sagunson ==\n[[:$1]] yaon na sa pinagheras na repositoryo. An pagbabalyo nin sagunson paduman kaining titulo masalambaw sa pinagheras na sagunson.",
+       "move-over-sharedrepo": "[[:$1]] yaon na sa pinagheras na repositoryo. An pagbabalyo nin sagunson paduman kaining titulo masalambaw sa pinagheras na sagunson.",
        "file-exists-sharedrepo": "An pangaran nin saguson na pinili ginagamit na sa pinagheras na repositoryo.\nPakipili kan ibang pangaran.",
        "export": "Paluwason an mga pahina",
        "exporttext": "Pwede mong ipadara an teksto asin historya nin paghirá kan sarong partikular na páhina o grupo nin mga páhina na nakapatos sa ibang XML. Pwede ining ipadara sa ibang wiki gamit an MediaWiki sa paagi kan [[Special:Import|pagpadara nin páhina]].\n\nPara makapadara nin mga páhina, ilaag an mga titulo sa kahon para sa teksto sa babâ, sarong titulo kada linya, dangan pilîon kun boot mo presenteng bersyón asin dating bersyón, na may mga linya kan historya, o an presenteng bersyón sana na may impormasyon manonongod sa huring hirá.\n\nSa kaso kan huri, pwede ka man na maggamit nin takod, arog kan [[{{#Special:Export}}/{{MediaWiki:Mainpage}}]] para sa páhinang \"[[{{MediaWiki:Mainpage}}]]\".",
        "thumbnail_gd-library": "Bakong kumpleto an kasalansanan kan kalibrohang GD: Nawawara an trabaho kan $1",
        "thumbnail_image-missing": "An sagunson garo baga nawawara: $1",
        "import": "Ilaog an mga páhina",
-       "importinterwiki": "Ipadara an Transwiki",
+       "importinterwiki": "importaron an mga pahina gikan sa ibang wiki",
        "import-interwiki-text": "Pumili nin sarong wiki asin titulo kan pahina na importaron.\nMga petsa nin kaliwatan asin pangaran kan mga paraliwat pagpepreserbaron.\nGabos na aksyon nin importa sa transwiki nakatala sa [[Special:Log/import|talaan nin importa]].",
        "import-interwiki-sourcepage": "Gikanang pahina:",
        "import-interwiki-history": "Kopyahon an gabos na mga bersyón para sa páhinang ini",
        "importuploaderrortemp": "Pagkarga kan ini-importang sagunson nagpalya.\nAn temporaryong polder nawawara.",
        "import-parse-failure": "XML importang panabot puminalya",
        "import-noarticle": "Mayong pahina na maiimporta!",
-       "import-nonewrevisions": "An gabos na mga rebisyon dati nang importado.",
+       "import-nonewrevisions": "Mayong rebisyon na pig-import (dati nang yaun o dai binali huli sa mga sala)",
        "xml-error-string": "$1 sa linya $2, kol $3 (bayta $4): $5",
        "import-upload": "Ikarga an XML na datos",
-       "import-token-mismatch": "Nawara an datos kan sesyon.\nPaki-otro giraray.",
+       "import-token-mismatch": "Pagkawara nin datos kan sesyon.\n\nProbaran tabì giraray. Kun dai man giraray magibo, probaran [[Special:UserLogout|na magluwas]] dangan maglaog giraray asin siguraduhon na nagtutugot an browser sa mga cookies hali sa site na ini.",
        "import-invalid-interwiki": "Dae makakapag-importa gikan sa pinagsambit na wiki.",
        "import-error-edit": "An pahina \"$1\" bakong importado nin huli ta ika dae tinutugutan na magliliwat kaini.",
        "import-error-create": "An pahina \"$1\" bakong importado nin huli ta ika dae tinutugutan na magmumukna kaini.",
        "import-rootpage-nosubpage": "Espasyong-ngaran \"$1\" kan ugat na pahina dae minatugot nin pan-irarom na mga pahina.",
        "importlogpage": "Usip nin pagpalaog",
        "importlogpagetext": "Administratibong mga importadong pahina na igwang historiya nin pagliliwat gikan sa ibang wikis.",
-       "import-logentry-upload-detail": "$1 {{PLURAL:$1|rebisyon|mga rebisyon}}",
+       "import-logentry-upload-detail": "$1 {{PLURAL:$1|rebisyon|mga rebisyon}} na importado",
        "import-logentry-interwiki-detail": "$1 {{PLURAL:$1|rebisyon|mga rebisyon}} gikan sa $2",
        "javascripttest": "Testing sa JavaScript",
        "javascripttest-qunit-intro": "Hilngon [$1 dokumentasyon sa pagtesting] sa mediawiki.org.",
        "tooltip-search": "Hanápon an {{SITENAME}}",
        "tooltip-search-go": "Magduman sa pahina na igwa kaining eksaktong pangaran kun eksistido",
        "tooltip-search-fulltext": "Hanápon an mga pahina para kaining teksto",
-       "tooltip-p-logo": "Bisitahon an Pangenot na Pahina",
+       "tooltip-p-logo": "Bisitahon an Panginot na Pahina",
        "tooltip-n-mainpage": "Bisitahon an Pangenot na Pahina",
        "tooltip-n-mainpage-description": "Bisitahon an Pangenot na Pahina",
        "tooltip-n-portal": "Manunungod sa proyekto, ano an saimong maginibo, saen makanumpong nin mga bagay",
        "anonymous": "Bako-bistadong {{PLURAL:$1|paragamit|mga paragamit}} kan {{SITENAME}}",
        "siteuser": "Paragamit kan {{SITENAME}} na si $1",
        "anonuser": "{{SITENAME}} bako-bistadong paragamit $1",
-       "lastmodifiedatby": "Ining páhina huring binago sa $2, $1 ni $3.",
+       "lastmodifiedatby": "Ining pahina huring binago sa $2, $1 ni $3.",
        "othercontribs": "Binase ini sa trabaho ni $1.",
        "others": "iba pa",
-       "siteusers": "{{SITENAME}} {{PLURAL:$2|paragamit|mga paragamit}} $1",
+       "siteusers": "{{SITENAME}}{{PLURAL:$2|paragamit|mga paragamit}}$1",
        "anonusers": "{{SITENAME}} bako-bistadong {{PLURAL:$2|paragamit|mga paragamit}} $1",
        "creditspage": "Mga krédito nin páhina",
        "nocredits": "Mayong talastas kan kredito para sa ining pahina.",
        "scarytranscludefailed-httpstatus": "[An paghigkos kan panguyog nagpalya para sa $1: HTTP $2]",
        "scarytranscludetoolong": "[An kilyawan grabe kahalaba]",
        "deletedwhileediting": "'''Patanid tabi''': Ining pahina pinagpura matapos na ika nagpoon na magliliwat!",
-       "confirmrecreate": "Si [[User:$1|$1]] ([[User talk:$1|olay]]) pigparâ ining páhina pagkatapos mong magpoon kan paghira ta:\n: ''$2''\nIkonpirmar tabi na talagang gusto mong gibohon giraray ining pahina.",
-       "confirmrecreate-noreason": "Paragamit [[User:$1|$1]] ([[User talk:$1|Olay]]) an nagpura kaining pahina matapos na ika nagpoon na magliliwat. Pakikumpirma tabi na ika boot na muknaon otro ining pahina.",
+       "confirmrecreate": "Si [[User:$1|$1]] ([[User talk:$1|olay]]){{GENDER:$1|pigpara}} ining páhina pagkatapos mong magpoon kan paghira ta:\n: <em>$2</em>\nIkonpirmar tabi na talagang gusto mong gibohon giraray ining pahina.",
+       "confirmrecreate-noreason": "Paragamit [[User:$1|$1]] ([[User talk:$1|Olay]]){{GENDER:$1|an nagpura}} kaining pahina matapos na ika nagpoon na magliliwat. Pakikumpirma tabi na ika boot na muknaon otro ining pahina.",
        "recreate": "Gibohón giraray",
        "confirm_purge_button": "Sige",
        "confirm-purge-top": "Halîon an an aliho kaining páhina?",
        "confirm-watch-top": "Idadagdag ining pahina sa saimong bantay-listahan?",
        "confirm-unwatch-button": "OK tabi",
        "confirm-unwatch-top": "Haleon ining pahina gikan sa saimong bantay-listahan?",
+       "confirm-mcrrestore-title": "Ibalik an mga rebisyon",
        "imgmultipageprev": "← nakaaging pahina",
        "imgmultipagenext": "sunod na pahina →",
        "imgmultigo": "Dumanán!",
        "tags-active-no": "Dae",
        "tags-edit": "liwatón",
        "tags-hitcount": "$1 {{PLURAL:$1|kaliwatan|mga kaliwatan}}",
+       "tags-edit-nooldid-title": "Imbalidong target nin pagbabago",
        "comparepages": "Ikumpara an mga pahina",
        "compare-page1": "Pahina 1",
        "compare-page2": "Pahina 2",
        "compare-revision-not-exists": "An rebisyon na saimong pinagsambit bakong eksistido.",
        "dberr-problems": "Sori! Ining sityo igwang naeksperiyensiyahan na mga kakundian sa teknikal.",
        "dberr-again": "Prubaring maghalat tabi nin nagkapirang minutos asin otrohon ikarga.",
-       "dberr-info": "(Dae makakontak sa serbidor kan datos-sarayan: $1)",
+       "dberr-info": "(Dae makakontak sa serbidor kan datos-sarayan:$1)",
        "dberr-info-hidden": "(Dae makakontak sa serbidor kan datos-sarayan)",
        "htmlform-invalid-input": "Igwa nin mga problema an iba sa saimong pinaglaog",
        "htmlform-select-badoption": "An halaga na saimong pinagsambit bakong saro sa balidong pagpipilian.",
        "logentry-newusers-create2": "An panindog kan paragamit $3 {{GENDER:$2|pinagmukna}} na ni $1",
        "logentry-newusers-byemail": "An panindog kan paragamit $3 {{GENDER:$2|pinagmukna}} ni $1 asin an sekretong panlaog ipinadara na sa paagi nin e-surat",
        "logentry-newusers-autocreate": "An panindong kan paragamit $1 awtomatikong {{GENDER:$2|pinagmukna}} na",
-       "logentry-rights-rights": "$1 {{GENDER:$2|pinagliwat}} kan pangrupong pagkamiyembro para sa $3 gikan sa $4 pasiring sa $5",
+       "logentry-rights-rights": "$1 {{GENDER:$2|pinagliwat}} kan pangrupong pagkamiyembro para sa {{GENDER:$6|$3}} gikan sa $4 pasiring sa $5",
        "logentry-rights-rights-legacy": "$1 {{GENDER:$2|nagliwat}} kan pangrupong pagkamiyembro para sa $3",
        "logentry-rights-autopromote": "$1 awtomatikong {{GENDER:$2|pinagpalangkaw}} gikan sa $4 pasiring sa $5",
        "logentry-upload-upload": "$1 {{GENDER:$2|pig-upload}} $3",
        "feedback-subject": "Subheto",
        "feedback-submit": "Isumite",
        "feedback-thanks": "Salamat! An saimong balik-simbag pinagposte sa pahina \"[$2 $1]\".",
+       "feedback-useragent": "Ahente nin paragamit:",
        "searchsuggest-search": "{{SITENAME}}",
        "searchsuggest-containing": "may laog na...",
        "api-error-badtoken": "Panlaog na kasalaan: Raot na pangilip",
        "limitreport-expansiondepth": "Kinatugmadan kan pinakahalangkaw na kahiwasan",
        "limitreport-expensivefunctioncount": "Bilang kan hiro nin mamahalon na parabangay",
        "expandtemplates": "Bigwakon an mga panguyog",
-       "expand_templates_intro": "Ining espesyal na pahina minakua nin teksto asin minabigwak kan gabos na mga panguyog na yaon kaini na paoro-otro.\nIni man minabigwak sa punksyon kan suportadong parabangay na arog kan <code><nowiki>{{</nowiki>#language:…}}</code> asin mga kapilyangan arog kan <code><nowiki>{{</nowiki>CURRENTDAY}}</code>.\nSa katunayan, ini minapabigwak kan gabos na bagay na yaon sa mga dobleng panandayan.",
+       "expand_templates_intro": "Ining espesyal na pahina minakua nin teksto asin minabigwak kan gabos na mga panguyog na yaon kaini na paoro-otro.\nIni man minabigwak sa punksyon kan suportadong parabangay na arog kan \n<code><nowiki>{{</nowiki>#language:…}}</code> asin mga kapilyangan arog kan \n<code><nowiki>{{</nowiki>CURRENTDAY}}</code>.\nSa katunayan, ini minapabigwak kan gabos na bagay na yaon sa mga dobleng panandayan.",
        "expand_templates_title": "Kontekstong titulo, para sa {{FULLPAGENAME}}, ibpa.:",
        "expand_templates_input": "Ikaag an teksto.",
        "expand_templates_output": "Resulta",
        "expand_templates_generate_xml": "Ipahiling an panlunhay na kahoy nin XML",
        "expand_templates_preview": "Patânaw",
        "mw-widgets-abandonedit": "Nakakaseguro ka na gusto mong bumalik sa kamugtakan nin pagtanaw na dae nagtatagamang enot?",
+       "mw-widgets-dateinput-no-date": "Mayong pigpilî",
        "mw-widgets-dateinput-placeholder-day": "TTTT-BB-AA",
        "mw-widgets-dateinput-placeholder-month": "TTTT-BB",
-       "randomrootpage": "Purakan na ugat nin pahina"
+       "date-range-from": "Poon na petsa:",
+       "randomrootpage": "Purakan na ugat nin pahina",
+       "log-action-filter-delete-revision": "Pagpura kan mga pagbabago",
+       "revid": "rebisyon $1"
 }
index dbc65ab..cd0cacf 100644 (file)
@@ -77,7 +77,7 @@
        "tog-watchlisthideliu": "অ্যাকাউন্টে প্রবেশকৃত ব্যবহারকারীদের সম্পাদনাগুলি নজরতালিকায় লুকিয়ে রাখা হোক",
        "tog-watchlistreloadautomatically": "প্রতিবার একটি ছাঁকনি পরিবর্তন হওয়া মাত্রই স্বয়ংক্রিয়ভাবে নজরতালিকাটি পুনঃলোড করা হোক (জাভাস্ক্রিপ্ট প্রয়োজন)",
        "tog-watchlistunwatchlinks": "পরিবর্তনসহ দেখা পাতাগুলিতে সরাসরি দেখা/না দেখার ({{int:Watchlist-unwatch}}/{{int:Watchlist-unwatch-undo}}) চিহ্ন যুক্ত করুন (এই কার্যকারিতা জন্য জাভাস্ক্রিপ্ট প্রয়োজন)",
-       "tog-watchlisthideanons": "বà§\87নামà§\80 à¦¬à§\8dযবহারà¦\95ারà§\80দà§\87র à¦¸à¦®à§\8dপাদনাà¦\97à§\81লি à¦¨à¦\9cরতালিà¦\95ায় আড়ালে রাখা হোক",
+       "tog-watchlisthideanons": "নà¦\9cরতালিà¦\95ায় à¦¬à§\87নামà§\80 à¦¬à§\8dযবহারà¦\95ারà§\80দà§\87র à¦¸à¦®à§\8dপাদনাà¦\97à§\81লি আড়ালে রাখা হোক",
        "tog-watchlisthidepatrolled": "পরীক্ষিত সম্পাদনাগুলি নজরতালিকায় লুকিয়ে রাখা হোক",
        "tog-watchlisthidecategorization": "পাতার শ্রেণীবদ্ধকরণ লুকিয়ে রাখা হোক",
        "tog-ccmeonemails": "অন্য ব্যবহারকারীর কাছে আমার পাঠানো ইমেইলের একটি প্রতিলিপি আমাকে পাঠানো হোক",
        "history_short": "ইতিহাস",
        "history_small": "ইতিহাস",
        "updatedmarker": "আপনার শেষ পরিদর্শনের পর থেকে হালনাগাদকৃত",
-       "printableversion": "à¦\9bাপার যোগ্য সংস্করণ",
+       "printableversion": "মà§\81দà§\8dরণযোগ্য সংস্করণ",
        "permalink": "স্থায়ী সংযোগ",
        "print": "মুদ্রণ",
        "view": "দেখুন",
        "otherlanguages": "অন্যান্য ভাষাসমূহ",
        "redirectedfrom": "($1 থেকে পুনর্নির্দেশিত)",
        "redirectpagesub": "পুনর্নির্দেশ পাতা",
-       "redirectto": "পুননির্দেশিত হয়েছে:",
+       "redirectto": "পà§\81নরà§\8dনিরà§\8dদà§\87শিত à¦¹à¦¯à¦¼à§\87à¦\9bà§\87:",
        "lastmodifiedat": "এই পাতা শেষ সম্পাদিত হয়েছে $2টার সময়, $1 তারিখে।",
        "viewcount": "এই পাতাটি {{PLURAL:$1|একবার|$1 বার}} দেখা হয়েছে।",
        "protectedpage": "সুরক্ষিত পাতা",
        "virus-scanfailed": "স্ক্যান করা যাচ্ছে না (কোড $1)",
        "virus-unknownscanner": "অজানা এন্টিভাইরাস:",
        "logouttext": "'''আপনি এখন আপনার অ্যাকাউন্ট থেকে প্রস্থান করেছেন।'''\n\nনোট করুন যে কিছু পাতায় আপনাকে এখনও প্রবেশ অবস্থায় দেখাবে, যতক্ষণ না আপনি ব্রাউজার ক্যাশ পরিষ্কার করছেন।",
+       "logging-out-notify": "আপনাকে প্রস্থান করানো হচ্ছে, দয়া করে অপেক্ষা করুন।",
+       "logout-failed": "এখন প্রস্থান করা যাবে না: $1",
        "cannotlogoutnow-title": "এখন প্রস্থান করা যাবে না",
        "cannotlogoutnow-text": "$1 ব্যবহার করার সময় প্রস্থান করা সম্ভব নয়।",
        "welcomeuser": "স্বাগতম, $1!",
        "autoblockedtext": "আপনার আইপি ঠিকানাটিকে স্বয়ংক্রিয়ভাবে সম্পাদনায় বাধাদান করা হয়েছে কারণ এমন আরেকজন ব্যবহারকারী এটি ব্যবহার করেছেন, যাকে $1 বাধা দিয়েছেন।\nযে কারণে বাধা দেওয়া হয়েছে সেটি হল:\n\n:<em>$2</em>\n\n* বাধা শুরুর সময়: $8\n* বাধা শেষের সময়: $6\n* যাকে বাধাদান করা হয়েছে: $7\n\nআপনি $1-এর সাথে কিংবা অন্য যেকোন [[{{MediaWiki:Grouppage-sysop}}|প্রশাসকের]] সাথে যোগাযোগ করে এই বাধা সংক্রান্ত বিষয়ে আলোচনা করতে পারেন।\n\nলক্ষ্য করুন, আপনি \"{{int:emailuser}}\" বৈশিষ্ট্যটি ব্যবহার করতে পারবেন না যদি না আপনার [[Special:Preferences|অ্যাকাউন্টের পছন্দসমূহে]] একটি বৈধ ইমেইল ঠিকানা নিবন্ধিত না থাকে এবং আপনাকে এটি ব্যবহার করা থেকে অবরুদ্ধ না করা হয়ে থাকে।\n\nআপনার বর্তমান আইপি ঠিকানা হচ্ছে $3, এবং বাধা নং হল #$5।\nদয়া করে আপনার যেকোন জিজ্ঞাসাতে উপরের সমস্ত বিবরণ অন্তর্ভুক্ত করুন।",
        "systemblockedtext": "আপনার ব্যবহারকারী নাম অথবা আইপি ঠিকানাটিকে স্বয়ংক্রিয়ভাবে মিডিয়াউইকি দ্বারা বাধাদান করা হয়েছে। যে কারণটি দেওয়া হয়েছে, সেটি হল:\n\n:<em>$2</em>\n\n* বাধা শুরুর সময়: $8\n* বাধা উঠিয়ে নেয়ার সময়: $6\n* যাকে বাধাদান করা হয়েছে: $7\n\nআপনার বর্তমান আইপি ঠিকানাটি হল $3।\nদয়া করে আপনার যেকোন জিজ্ঞাসাতে উপরের সমস্ত বিবরণ অন্তর্ভুক্ত করুন।",
        "blockednoreason": "কোন কারণ দেওয়া হয়নি",
+       "blockedtext-composite": "<strong>আপনার ব্যবহারকারী নাম অথবা আইপি ঠিকানাটিকে বাধা দেয়া হয়েছে।</strong>\n\nযে কারণটি দেওয়া হয়েছে, সেটি হল:\n\n:<em>$2</em>\n\n* বাধা শুরুর সময়: $8\n* বাধা উঠিয়ে নেয়ার সময়: $6\n\n* $5\n\nআপনার বর্তমান আইপি ঠিকানাটি হল $3।\nদয়া করে আপনার যেকোন জিজ্ঞাসাতে উপরের সমস্ত বিবরণ অন্তর্ভুক্ত করুন।",
        "whitelistedittext": "পাতায় সম্পাদনা করতে অনুগ্রহ করে $1 করুন।",
        "confirmedittext": "কোন সম্পাদনা করার আগে আপনার ই-মেইল ঠিকানাটি অবশ্যই নিশ্চিত করতে হবে। দয়া করে আপনার ই-মেইল ঠিকানাটি [[Special:Preferences|ব্যবহারকারীর পছন্দতালিকায়]] ঠিকমত দিন।",
        "nosuchsectiontitle": "অনুচ্ছেদ পাওয়া যায়নি",
        "accmailtext": "[[User talk:$1|$1]] এর জন্য দৈব ভাবে উৎপন্ন শব্দ চাবি $2 এ পাঠানো হয়েছে।\nলগ-ইন করার পর ''[[Special:ChangePassword|পাসওয়ার্ড পরিবর্তন]]'' পাতা থেকে এটি পরিবর্তন করা যাব।",
        "newarticle": "(নতুন)",
        "newarticletext": "আপনি এমন একটি পাতার সংযোগ অনুসরণ করছেন, যার অস্তিত্ব নেই।\nপাতাটি তৈরি করতে, নিচের বাক্সে তা টাইপ করা শুরু করুন (আরও তথ্য জানতে [$1 সহায়িকা পাতা] দেখুন)।\nআপনি যদি ভুল করে এখানে এসে থাকেন, তাহলে আপনার ব্রাউজারের <strong>পিছন</strong> বোতামে ক্লিক করুন।",
-       "anontalkpagetext": "----\n<em>à¦\8fà¦\9fি à¦\8fà¦\95à¦\9fি à¦¬à§\87নামà§\80 à¦¬à§\8dযবহারà¦\95ারà§\80র à¦\86লাপà§\87র à¦ªà¦¾à¦¤à¦¾, à¦¯à¦¿à¦¨à¦¿ à¦\8fà¦\96নà¦\93 à¦\95à§\8bন à¦\85à§\8dযাà¦\95াà¦\89নà§\8dà¦\9f à¦¤à§\88রি à¦\95রà§\87ননি, à¦\95িà¦\82বা à¦¤à¦¿à¦¨à¦¿ à¦\85à§\8dযাà¦\95াà¦\89নà§\8dà¦\9fà¦\9fি à¦¬à§\8dযবহার à¦\95রà¦\9bà§\87ন à¦¨à¦¾à¥¤</em>\nà¦\86মরা à¦¤à¦¾à¦\87 à¦¸à¦¾à¦\82à¦\96à§\8dযিà¦\95 à¦\86à¦\87পি à¦ à¦¿à¦\95ানা à¦¬à§\8dযবহার à¦\95রà§\87 à¦¤à¦¾à¦\81দà§\87র à¦¶à¦¨à¦¾à¦\95à§\8dত à¦\95রà¦\9bি।\nà¦\8fà¦\95াধিà¦\95 à¦¬à§\8dযবহারà¦\95ারà§\80 à¦\8fরà¦\95ম à¦\8fà¦\95à¦\9fি à¦\86à¦\87পি à¦ à¦¿à¦\95ানা à¦¬à§\8dযবহার à¦\95রতà§\87 à¦ªà¦¾à¦°à§\87ন।\nà¦\86পনি à¦¯à¦¦à¦¿ à¦\8fà¦\95à¦\9cন à¦¬à§\87নামà§\80 à¦¬à§\8dযবহারà¦\95ারà§\80 à¦¹à¦¯à¦¼à§\87 à¦¥à¦¾à¦\95à§\87ন à¦\8fবà¦\82 à¦¯à¦¦à¦¿ à¦\85নà§\81ভব à¦\95রà§\87ন à¦¯à§\87 à¦\86পনার à¦ªà§\8dরতি à¦\85পà§\8dরাসà¦\99à§\8dà¦\97িà¦\95 à¦®à¦¨à§\8dতবà§\8dয à¦\95রা à¦¹à¦¯à¦¼à§\87à¦\9bà§\87, à¦¤à¦¾à¦¹à¦²à§\87 à¦\85নà§\8dযানà§\8dয à¦¬à§\87নামà§\80 ব্যবহারকারীর সাথে ভবিষ্যতে বিভ্রান্তি এড়াতে অনুগ্রহ করে [[Special:CreateAccount|একটি অ্যাকাউন্ট তৈরি করুন]] অথবা  [[Special:UserLogin|অ্যাকাউন্টে প্রবেশ করুন]]।",
+       "anontalkpagetext": "----\n<em>à¦\8fà¦\9fি à¦\8fà¦\95à¦\9cন à¦¬à§\87নামà§\80 à¦¬à§\8dযবহারà¦\95ারà§\80র à¦\86লাপà§\87র à¦ªà¦¾à¦¤à¦¾, à¦¯à¦¿à¦¨à¦¿ à¦\8fà¦\96নà¦\93 à¦\95à§\8bন à¦\85à§\8dযাà¦\95াà¦\89নà§\8dà¦\9f à¦¤à§\88রি à¦\95রà§\87ননি, à¦\95িà¦\82বা à¦¤à¦¿à¦¨à¦¿ à¦\85à§\8dযাà¦\95াà¦\89নà§\8dà¦\9fà¦\9fি à¦¬à§\8dযবহার à¦\95রà¦\9bà§\87ন à¦¨à¦¾à¥¤</em>\nà¦\86মরা à¦¤à¦¾à¦\87 à¦¸à¦¾à¦\82à¦\96à§\8dযিà¦\95 à¦\86à¦\87পি à¦ à¦¿à¦\95ানা à¦¬à§\8dযবহার à¦\95রà§\87 à¦¤à¦¾à¦\81দà§\87র à¦¶à¦¨à¦¾à¦\95à§\8dত à¦\95রà¦\9bি।\nà¦\8fà¦\95াধিà¦\95 à¦¬à§\8dযবহারà¦\95ারà§\80 à¦\8fরà¦\95ম à¦\8fà¦\95à¦\9fি à¦\86à¦\87পি à¦ à¦¿à¦\95ানা à¦¬à§\8dযবহার à¦\95রতà§\87 à¦ªà¦¾à¦°à§\87ন।\nà¦\86পনি à¦¯à¦¦à¦¿ à¦\8fà¦\95à¦\9cন à¦¨à¦¾à¦®à¦¹à§\80ন à¦¬à§\8dযবহারà¦\95ারà§\80 à¦¹à¦¯à¦¼à§\87 à¦¥à¦¾à¦\95à§\87ন à¦\8fবà¦\82 à¦¯à¦¦à¦¿ à¦\85নà§\81ভব à¦\95রà§\87ন à¦¯à§\87 à¦\86পনার à¦ªà§\8dরতি à¦\85পà§\8dরাসà¦\99à§\8dà¦\97িà¦\95 à¦®à¦¨à§\8dতবà§\8dয à¦\95রা à¦¹à¦¯à¦¼à§\87à¦\9bà§\87, à¦¤à¦¾à¦¹à¦²à§\87 à¦\85নà§\8dযানà§\8dয à¦¨à¦¾à¦®à¦¹à§\80ন ব্যবহারকারীর সাথে ভবিষ্যতে বিভ্রান্তি এড়াতে অনুগ্রহ করে [[Special:CreateAccount|একটি অ্যাকাউন্ট তৈরি করুন]] অথবা  [[Special:UserLogin|অ্যাকাউন্টে প্রবেশ করুন]]।",
        "noarticletext": "বর্তমানে এই পাতায় কোন লেখা নেই।\nআপনি চাইলে অন্যান্য পাতায় [[Special:Search/{{PAGENAME}}| এই শিরোনামটি অনুসন্ধান করতে পারেন]],\n<span class=\"plainlinks\">[{{fullurl:{{#Special:Log}}|page={{FULLPAGENAMEE}}}} এ সম্পর্কিত লগ অনুসন্ধান করতে পারেন], \nকিংবা [{{fullurl:{{FULLPAGENAME}}|action=edit}} এই পাতাটি তৈরি করতে পারেন]</span>।",
        "noarticletext-nopermission": "বর্তমানে এই পাতায় কোন লেখা নেই।\nআপনি চাইলে অন্য পাতায় [[Special:Search/{{PAGENAME}}| শিরোনামটি অনুসন্ধান করতে পারেন]], অথবা <span class=\"plainlinks\">[{{fullurl:{{#Special:Log}}|page={{FULLPAGENAMEE}}}} সম্পর্কিত লগ অনুসন্ধান করতে পারেন]</span>, কিন্তু আপনার এই পাতাটি তৈরী করার অনুমতি নেই।",
        "missing-revision": "\"{{FULLPAGENAME}}\" এর #$1তম সংস্করণটি প্রদর্শন সম্ভব নয়।\n\nসাধারণত মুছে ফেলা হয়েছে এমন পাতার মেয়াদ উত্তীর্ণ ইতিহাসের সংযোগ অনুসরণ করার কারণে এটি হতে পারে। \n[{{fullurl:{{#Special:Log}}/delete|page={{FULLPAGENAMEE}}}} অপসারণ লগে] বিস্তারিত তথ্য জানা যাবে।",
        "previewconflict": "এই প্রাকদর্শনটি সম্পাদনা ক্ষেত্রের উপরের অংশটির টেক্সট সংরক্ষণ করলে যেরকম দেখাবে, তা দেখাচ্ছে।",
        "session_fail_preview": "দুঃখিত! সেশন ডাটা হারিয়ে যাওয়ার কারণে আপনার সম্পাদনাটি সংরক্ষণ করা সম্ভব হয়নি।\n\nআপনি সম্ভবত সংযোগ হারিয়েছন। <strong>দয়া করে যাচাই করুন যে আপনি এখনও প্রবেশরত রয়েছেন এবং আবার চেষ্টা করুন</strong>। যদি এটি এখনও কাজ না করে, তাহলে দয়া করে [[Special:UserLogout|অ্যাকাউন্ট থেকে প্রস্থান করুন]] এবং আবার অ্যাকাউন্টে প্রবেশ করে চেষ্টা করুন এবং এবং পরীক্ষা করুন যে আপনার ব্রাউজার এই সাইটে কুকি ব্যবহারের অনুমতি দেয়।",
        "session_fail_preview_html": "দুঃখিত! সেশনের উপাত্ত হারিয়ে যাওয়ার কারণে আমরা আপনার সম্পাদনাটি প্রক্রিয়াভুক্ত করতে পারিনি।\n\n<em>{{SITENAME}}-এ raw HTML সক্রিয় আছে বলে জাভাস্ক্রিপ্ট ভিত্তিক আক্রমণ থেকে প্রতিরক্ষার জন্য প্রাকদর্শনটি দেখানো হচ্ছে না।</em>\n\n<strong>যদি এটি সম্পাদনার একটি বৈধ প্রচেষ্টা হয়, তবে অনুগ্রহ করে আবার চেষ্টা করুন।</strong>\nযদি তারপরেও কাজ না হয়, তবে অ্যাকাউন্ট থেকে [[Special:UserLogout|বেরিয়ে গিয়ে]] আবার প্রবেশ করুন, এবং পরীক্ষা করে দেখুন যে আপনার ব্রাউজারে এই সাইট থেকে কুকি রাখার অনুমতি আছে কি না।",
-       "token_suffix_mismatch": "'''আপনার সম্পাদনাটি প্রত্যাখ্যান করা হয়েছে, কারণ আপনার ক্লায়েন্ট প্রোগ্রামটি সম্পাদনা টেক্সটের বিরামচিহ্নগুলি গুলিয়ে ফেলেছে। পাতাটির টেক্সটে যাতে ক্ষতি না হয় সেজন্য সম্পাদনাটি প্রত্যাখ্যান করা হয়েছে। আপনি কোন ত্রুটিপূর্ণ ওয়েব-ভিত্তিক বেনামী প্রক্সি সেবা ব্যবহার করলে এরকম হতে পারে।'''",
+       "token_suffix_mismatch": "<strong>আপনার সম্পাদনাটি প্রত্যাখ্যান করা হয়েছে, কারণ আপনার ক্লায়েন্ট প্রোগ্রামটি সম্পাদনা পাঠ্যের বিরামচিহ্নগুলি গুলিয়ে ফেলেছে।</strong>\nপাতাটির পাঠ্যে যাতে ক্ষতি না হয় সেজন্য সম্পাদনাটি প্রত্যাখ্যান করা হয়েছে।\nআপনি কোন ত্রুটিপূর্ণ ওয়েব-ভিত্তিক বেনামী প্রক্সি সেবা ব্যবহার করলে এরকম হতে পারে।",
        "edit_form_incomplete": "'''আপনার সম্পাদনার কিছু অংশ সার্ভারে পৌছায় নি; আপনার সম্পাদনা সম্পূর্ণরুপে আছে কিনা নিশ্চিত হয়ে আবার চেষ্টা করুন'''",
        "editing": "সম্পাদনা করছেন: $1",
        "creating": "$1 পাতাটি তৈরি করছেন",
        "rcfilters-clear-all-filters": "সব ছাঁকনি পরিষ্কার করুন",
        "rcfilters-show-new-changes": "$1 থেকে নতুনতর পরিবর্তনসমূহ দেখুন",
        "rcfilters-search-placeholder": "সাম্প্রতিক পরিবর্তনসমূহ ছাঁকুন (ব্রাউজ বা টাইপ করা শুরু করুন)",
+       "rcfilters-search-placeholder-mobile": "ছাঁকনি",
        "rcfilters-invalid-filter": "অকার্যকর ছাঁকনি",
        "rcfilters-empty-filter": "কোনো সক্রিয় ফিল্টার নেই। সমস্ত অবদান দেখানো হয়েছে।",
        "rcfilters-filterlist-title": "ছাঁকনি",
        "ipblocklist-otherblocks": "অন্যান্য {{PLURAL:$1|বাধাঁ|বাধাঁসমূহ}}",
        "infiniteblock": "অসীম",
        "expiringblock": "$1 তারিখের $2 এ মেয়াদোত্তীর্ণ হবে",
-       "anononlyblock": "শুধু বেনামীদের",
+       "anononlyblock": "শুধুমাত্র বেনামীগণ",
        "noautoblockblock": "স্বয়ংক্রিয় বাধাদান নিষ্ক্রিয় করা হয়েছে",
        "createaccountblock": "অ্যাকাউন্ট সৃষ্টিতে বাধা দেওয়া হয়েছে",
        "emailblock": "ই-মেইল বাধা দেয়া হয়েছে",
        "reblock-logentry": "[[$1]]-এর বাধাদান সেটিং পরিবর্তন করেছেন যেটি শেষ হবার মেয়াদ $2 $3",
        "blocklogtext": "এটি ব্যবহারকারীদেরকে বাধা দানের বা বাধা তুলে নেওয়ার লগ।\nস্বয়ংক্রিয়ভাবে বাধাদানকৃত আইপি ঠিকানাগুলি এখানে তালিকাবদ্ধ করা হয়নি।\nবর্তমানে সক্রিয় নিষিদ্ধকরণ ও বাধাদানের তালিকার জন্য [[Special:BlockList| বাধাদান তালিকা]] দেখুন।",
        "unblocklogentry": "$1-এর উপর বাধা তুলে নেয়া হয়েছে",
-       "block-log-flags-anononly": "à¦\95à§\87বল à¦¬à§\87নামà§\80 à¦¬à§\8dযবহারà¦\95ারà§\80রা",
+       "block-log-flags-anononly": "শà§\81ধà§\81মাতà§\8dর à¦¬à§\87নামà§\80 à¦¬à§\8dযবহারà¦\95ারà§\80à¦\97ণ",
        "block-log-flags-nocreate": "অ্যাকাউন্ট সৃষ্টি নিষ্ক্রিয় করা হয়েছে",
        "block-log-flags-noautoblock": "স্বয়ংক্রিয় বাধাদান নিষ্ক্রিয়",
        "block-log-flags-noemail": "ই-মেইলে বাধা আছে",
        "block-log-flags-nousertalk": "নিজের আলাপের পাতা সম্পাদনা করতে পারবে না",
-       "block-log-flags-angry-autoblock": "à¦\89নà§\8dনত à¦\85à¦\9fà§\8bবà§\8dলà¦\95 সক্রিয়",
+       "block-log-flags-angry-autoblock": "à¦\89নà§\8dনত à¦¸à§\8dবয়à¦\82à¦\95à§\8dরিয় à¦¬à¦¾à¦§à¦¾à¦¦à¦¾à¦¨ সক্রিয়",
        "block-log-flags-hiddenname": "ব্যবহারকারী নাম লুক্কায়িত",
        "range_block_disabled": "প্রশাসকের পক্ষে আইপি ঠিকানার শ্রেণী বাধাদানের ক্ষমতা নিষ্ক্রিয় আছে।",
        "ipb_expiry_invalid": "মেয়াদোত্তীর্ণকাল অবৈধ।",
        "proxyblockreason": "আপনার আইপি ঠিকানাকে বাধা দেয়া হয়েছে কারণ এটি একটি উন্মুক্ত প্রক্সি। অনুগ্রহ করে আপনার ইন্টারনেট সেবা প্রদানকারী কোম্পানির সাথে কারিগরি সহায়তার ব্যাপারে যোগাযোগ করুন এবং এই গুরুত্বপূর্ণ নিরাপত্তা সমস্যার ব্যাপারে তাদেরকে অবহিত করুন।",
        "sorbsreason": "আপনার আইপি ঠিকানাটি {{SITENAME}}-এর ব্যবহার করা DNSBL-এ উন্মুক্ত প্রক্সি হিসেবে তালিকাভুক্ত আছে।",
        "sorbs_create_account_reason": "আপনার আইপি ঠিকানাটি {{SITENAME}}-এর ব্যবহার করা DNSBL-এ উন্মুক্ত প্রক্সি হিসেবে তালিকাভুক্ত আছে। আপনি কোন অ্যাকাউন্ট সৃষ্টি করতে পারবেন না।",
-       "softblockrangesreason": "বà§\87নামà§\80 à¦\85বদান à¦\86পনার à¦\86à¦\87পি à¦ à¦¿à¦\95ানা à¦¥à§\87à¦\95à§\87 অনুমোদিত নয় ($1)। দয়া করে প্রবেশ করুন।",
+       "softblockrangesreason": "à¦\86পনার à¦\86à¦\87পি à¦ à¦¿à¦\95ানা à¦¥à§\87à¦\95à§\87 à¦¬à§\87নামà§\87 à¦\85বদান à¦°à¦¾à¦\96া অনুমোদিত নয় ($1)। দয়া করে প্রবেশ করুন।",
        "xffblockreason": "X-Forwarded-For হেডারে থাকা আইপি ঠিকানাটি ব্লক করা হয়েছে, হয় এটি আপনার নিজের অথবা আপনার ব্যবহৃত প্রক্সি সার্ভারের আইপি ঠিকানা। ব্লক করার কারণ হল: $1",
        "cant-see-hidden-user": "আপনি যে ব্যবহারকারীকে ব্লক বা লুকিয়ে রাখতে চাচ্ছেন তাকে আগে থেকেই ব্লক বা লুকিয়ে রাখা হয়েছে। এছাড়া আপনার Hideuser অধিকার নেই, তাই আপনি ব্যবহারকারীর অবস্থা পরিবর্তন করতে পারবেন না।",
        "ipbblocked": "আপনি অন্য কোন ব্যবহারকরীকে বাধাদান বা বাধা বাতিল করতে পারবেন না, কারণ আপনি নিজেই অবরুদ্ধ রয়েছেন।",
        "group-bot.js": "/* এখানে সন্নিবেশিত জাভাস্ক্রিপ্ট শুধু বটের জন্য লোড হবে */",
        "group-sysop.js": "/* এখানে সন্নিবেশিত জাভাস্ক্রিপ্ট শুধু প্রশাসকদের জন্য লোড হবে */",
        "group-bureaucrat.js": "/* এখানে সন্নিবেশিত জাভাস্ক্রিপ্ট শুধু ব্যুরোক্র্যাটদের জন্য লোড হবে */",
-       "anonymous": "{{SITENAME}} এর বেনামী {{PLURAL:$1|ব্যবহারকারী|ব্যবহারকারীবৃন্দ}}",
+       "anonymous": "{{SITENAME}}-এর বেনামী {{PLURAL:$1|ব্যবহারকারী|ব্যবহারকারীগণ}}",
        "siteuser": "{{SITENAME}} ব্যবহারকারী $1",
        "anonuser": "{{SITENAME}} বেনামী ব্যবহারকারী $1",
        "lastmodifiedatby": "$3 কর্তৃক $2, $1 তারিখে এই পাতাটি শেষ সম্পাদিত হয়েছিল।",
        "mw-widgets-abandonedit-discard": "সম্পাদনা বাতিল করুন",
        "mw-widgets-abandonedit-keep": "সম্পাদনা অব্যাহত রাখুন",
        "mw-widgets-abandonedit-title": "আপনি কি নিশ্চিত?",
+       "mw-widgets-copytextlayout-copy": "অনুলিপি",
+       "mw-widgets-copytextlayout-copy-fail": "ক্লিপবোর্ডে অনুলিপি করা ব্যর্থ হয়েছে।",
+       "mw-widgets-copytextlayout-copy-success": "ক্লিপবোর্ডে অনুলিপি করা হয়েছে।",
        "mw-widgets-dateinput-no-date": "কোন তারিখ নির্বাচন করা হয়নি",
        "mw-widgets-dateinput-placeholder-day": "বববব-মম-দদ",
        "mw-widgets-dateinput-placeholder-month": "বববব-মম",
        "restrictionsfield-help": "লাইন প্রতি একটি আইপি ঠিকানা বা CIDR পরিসীমা। সবকিছু সক্রিয় করতে ব্যবহার করুন: :<pre>0.0.0.0/0\n::/0</pre>",
        "edit-error-short": "ত্রুটি: $1",
        "edit-error-long": "ত্রুটিসমূহ:\n\n$1",
+       "specialmute": "নিঃশব্দ",
        "specialmute-submit": "নিশ্চিত করুন",
        "revid": "সংশোধন $1",
        "pageid": "পাতার আইডি $1",
index bbaf678..42b937e 100644 (file)
@@ -20,7 +20,8 @@
                        "Matma Rex",
                        "Trizek (WMF)",
                        "Dishual",
-                       "Fitoschido"
+                       "Fitoschido",
+                       "Huñvreüs"
                ]
        },
        "tog-underline": "Liammoù islinennet",
        "history": "Istor ar bajenn",
        "history_short": "Istor",
        "history_small": "istor",
-       "updatedmarker": "kemmet abaoe ma zaol-sell diwezhañ",
+       "updatedmarker": "kemmet abaoe ho kwel diwezhañ",
        "printableversion": "Stumm da voullañ",
        "permalink": "Chomlec'h ar stumm-mañ",
        "print": "Moullañ",
        "defaultmessagetext": "Testenn dre ziouer",
        "content-failed-to-parse": "C'hwitet eo dielfennadur endalc'had $2 evit ar patrom $1: $3",
        "invalid-content-data": "n'eo ket mat roadennoù an endalc'had",
-       "content-not-allowed-here": "N'eo ket aotreet an endalc'had \"$1\" er bajenn [[:$2]]",
+       "content-not-allowed-here": "N'eo ket aotreet an endalc'had \"$1\" er bajenn [[:$2]] e lec’h \"$3\"",
        "editwarning-warning": "Mar kuitait ar bajenn-mañ e c'hallit koll ar c'hemmoù degaset ganeoc'h.\nMa'z oc'h kevreet e c'hallit diweredekaat ar c'hemenn-diwall-mañ e rann \"{{int:prefs-editing}}\" en ho penndibaboù.",
        "editpage-invalidcontentmodel-title": "N'eo ket skoret ar patrom danvez",
        "editpage-invalidcontentmodel-text": "N'eo ket skoret ar patrom danvez \"$1\".",
        "stub-threshold-disabled": "Diweredekaet",
        "recentchangesdays": "Niver a zevezhioù da ziskouez er c'hemmoù diwezhañ :",
        "recentchangesdays-max": "D'ar muiañ $1 {{PLURAL:$1|deiz|deiz}}",
-       "recentchangescount": "Niver a gemmoù da ziskouez dre ziouer",
-       "prefs-help-recentchangescount": "Kemer a ra an dra-mañ e kont ar c'hemmoù diwezhañ, istor ar pajennoù hag ar marilhoù.",
+       "recentchangescount": "Niver a gemmoù da ziskouez e kemmoù diwezhañ, istorioù ha marilhoù, dre ziouer :",
+       "prefs-help-recentchangescount": "Niver brasañ : 1000",
        "prefs-help-watchlist-token2": "Homañ zo an alc'hwez kuzh d'ho roll-evezhiañ davit boued war ar web. Forzh piv a anavez anezhañ a c'hall lenn ho roll-evezhiañ, setu na lavarit grit diwar e benn. M'ho pez ezhomm, e c'hallit [[Special:ResetTokens|assevel anezhañ]].",
        "savedprefs": "Enrollet eo bet ar penndibaboù.",
        "savedrights": "Enrollet eo bet strolladoù implijer {{GENDER:$1|$1}}.",
        "timezoneregion-europe": "Europa",
        "timezoneregion-indian": "Meurvor Indez",
        "timezoneregion-pacific": "Meurvor Habask",
-       "allowemail": "Aotren ar posteloù a-berzh implijerien all",
+       "allowemail": "Aotreañ implijerien all da gas din posteloù",
        "email-allow-new-users-label": "Aotreañ e-bostoù a-berzh an implijerien nevez",
        "email-blacklist-label": "Difenn d'an implijerien-se da gas un e-bost din:",
        "prefs-searchoptions": "Klask",
        "grouppage-autoconfirmed": "{{ns:project}}: Implijerien bet kadarnaet ent emgefre",
        "grouppage-bot": "{{ns:project}}:Botoù",
        "grouppage-sysop": "{{ns:project}}:Merourien",
-       "grouppage-interface-admin": "{{ns:project}}:Merourien etrefas",
+       "grouppage-interface-admin": "{{ns:project}}:Merourien an etrefas",
        "grouppage-bureaucrat": "{{ns:project}}: Burevidi",
        "grouppage-suppress": "{{ns:project}}: Diverkerien",
        "right-read": "Lenn ar pajennoù",
        "recentchanges-legend-plusminus": "(<em>±123</em>)",
        "recentchanges-submit": "Diskouez",
        "rcfilters-tag-remove": "Dilemel '$1'",
+       "rcfilters-legend-heading": "<strong>Roll ar berradurioù&nbsp;:</strong>",
        "rcfilters-other-review-tools": "Ostilhoù reizhañ all",
        "rcfilters-group-results-by-page": "Strollañ an disoc'hoù dre bajenn",
        "rcfilters-activefilters": "Siloù oberiant",
        "rcfilters-activefilters-show": "Diskouez",
        "rcfilters-advancedfilters": "Siloù araokaet",
        "rcfilters-limit-title": "Kemmoù da vezañ diskouezet",
-       "rcfilters-limit-and-date-label": "$1 {{PLURAL:$1|kemm|kemmoù}}, $2",
+       "rcfilters-limit-and-date-label": "$1 kemm, $2",
+       "rcfilters-date-popup-title": "Prantad amzer da glask",
        "rcfilters-days-title": "Deizioù paseet",
        "rcfilters-hours-title": "Eurioù paseet",
        "rcfilters-days-show-days": "($1 {{PLURAL:$1|deiz}})",
        "rcfilters-clear-all-filters": "Riñsañ an holl siloù",
        "rcfilters-show-new-changes": "Gwelet ar c'hemmoù diwezhañ",
        "rcfilters-search-placeholder": "Silañ ar c'hemmoù diwezhañ (merdeiñ pe kregiñ da skrivañ)",
+       "rcfilters-search-placeholder-mobile": "Siloù",
        "rcfilters-invalid-filter": "Sil direizh",
        "rcfilters-empty-filter": "Sil oberiant ebet. War wel emañ an holl gemmoù.",
        "rcfilters-filterlist-title": "Siloù",
        "rcfilters-exclude-button-off": "Skarzhañ ar re dibabet",
        "rcfilters-exclude-button-on": "Re dibabet skarzhet",
        "rcfilters-view-tags": "Kemmoù merket",
+       "rcfilters-liveupdates-button": "Nevesaat war-eeun",
        "rcfilters-filter-showlinkedfrom-label": "Diskouez ar c'hemmoù war ar bajennoù liammet",
        "rcfilters-filter-showlinkedto-label": "Diskouez ar c'hemmoù war ar bajennoù liammet",
        "rcfilters-target-page-placeholder": "Skrivañ anv ar bajenn (pe rummad)",
        "deletecomment": "Abeg :",
        "deleteotherreason": "Abegoù/traoù all :",
        "deletereasonotherlist": "Abeg all",
-       "deletereason-dropdown": "*Abegoù diverkañ boazetañ\n** Strob\n** Vandalerezh\n** Gaou ouzh ar gwirioù perc'hennañ\n** Goulenn gant saver ar pennad\n** Liamm torret",
+       "deletereason-dropdown": "*Abegoù diverkañ boazetañ\n** Strob\n** Vandalerezh\n** Gaou ouzh ar gwirioù perc'hennañ\n** Goulenn gant saver ar pennad\n** Adkas torret",
        "delete-edit-reasonlist": "Kemmañ a ra an abegoù diverkañ",
        "delete-toobig": "Bras eo istor ar bajenn-mañ, ouzhpenn $1 {{PLURAL:$1|stumm|stumm}} zo. Bevennet eo bet an diverkañ pajennoù a-seurt-se kuit da zegas reuz war {{SITENAME}} dre fazi .",
        "delete-warning-toobig": "Bras eo istor ar bajenn-mañ, ouzhpenn {{PLURAL:$1|stumm|stumm}} zo.\nDiverkañ anezhi a c'hallo degas reuz war mont en-dro diaz titouroù {{SITENAME}};\ntaolit evezh bras.",
        "ipbreason": "Abeg :",
        "ipbreason-dropdown": "*Abegoù stankañ boutinañ\n** Degas titouroù faos\n** Tennañ danvez eus ar pajennoù\n** Degas liammoù Strobus war-du lec'hiennoù diavaez\n** Degas danvez diboell/dizoare er pajennoù\n** Emzalc'h hegazus/handeus betek re\n** Mont re bell gant implij meur a gont\n** Anv implijer n'eo ket aotreet",
        "ipb-hardblock": "Mirout ouzh an implijerien kevreet da zegas kemmoù adalek ar chomlec'h IP-mañ",
-       "ipbcreateaccount": "Mirout ouzh an implijer da grouiñ kontoù",
-       "ipbemailban": "Mirout ouzh an implijer da gas posteloù",
+       "ipbcreateaccount": "Krouiñ kontoù",
+       "ipbemailban": "Kas posteloù",
        "ipbenableautoblock": "Stankañ war-eeun ar chomlec'h IP diwezhañ implijet gant an den-mañ hag an holl chomlec'hioù a c'hallfe klask kemmañ traoù drezo diwezhatoc'h",
        "ipbsubmit": "Stankañ an implijer-mañ",
        "ipbother": "Prantad all",
        "ipboptions": "2 eurvezh:2 hours,1 devezh:1 day,3 devezh:3 days,1 sizhunvezh:1 week,2 sizhunvezh:2 weeks,1 mizvezh:1 month,3 mizvezh:3 months,6 mizvezh:6 months,1 bloaz:1 year,da viken:infinite",
        "ipbhidename": "Kuzhat anv an implijer er rolloù hag er c'hemmoù",
        "ipbwatchuser": "Evezhiañ pajennoù implijer ha kaozeal an implijer-mañ",
-       "ipb-disableusertalk": "Mirout ouzh an implijer-mañ da implijout e bajenn gaozeal dezhañ e-unan e-keit hag emañ stanket",
+       "ipb-disableusertalk": "Kemmañ e bajenn gaozeal dezhañ",
        "ipb-change-block": "Adstankañ an implijer-mañ gant an hevelep arventennoù",
        "ipb-confirm": "Kadarnaat ar stankadenn",
        "ipb-pages-label": "Pajennoù",
        "import-nonewrevisions": "N'eus bet enporzhiet degasadenn ebet (aze e oant dija, pe distaolet e oant bet abalamour da fazioù).",
        "xml-error-string": "$1 war al linenn $2, bann $3 (okted $4) : $5",
        "import-upload": "Enporzhiañ roadennoù XML",
-       "import-token-mismatch": "Kollet eo bet roadennoù an dalc'h.\n\nMarteze oc'h bet digevreet. <strong>Gwiriit emaoc'h mat kevreet ha klaskit en-dro</strong>.\nMa ne'z a ket en-dro c'hoazh, klaskit [[Special:UserLogout|digevreañ]] hag adkevreañ en-dro, ha gwiriit mat ec'h asant ho merdeer degemer toupinoù digant al lec'hienn-mañ.",
+       "import-token-mismatch": "Kollet eo bet roadennoù an dalc'h.\n\nMarteze oc'h bet digevreet. '''Gwiriit emaoc'h kevreet mat ha klaskit en-dro'''.\nMa ne'z a ket en-dro c'hoazh, klaskit [[Special:UserLogout|digevreañ]] hag adkevreañ, ha gwiriit mat ec'h asant ho merdeer degemer toupinoù digant al lec'hienn-mañ.",
        "import-invalid-interwiki": "Dibosupl enporzhiañ adal ar wiki spisaet.",
        "import-error-edit": "N'eo ket bet enporzhiet ar bajenn \"$1\" peogwir n'oc'h ket aotreet da zegas kemmoù enni.",
        "import-error-create": "N'eo ket bet enporzhiet ar bajenn \"$1\" peogwir n'oc'h ket aotreet da grouiñ anezhi.",
        "logentry-import-interwiki": "{{GENDER:$2|Enporzhiet}} eo bet $3 gant $1 adalek ur wiki all",
        "logentry-import-interwiki-details": "{{GENDER:$2|Enporzhiet}} eo bet $3 adalek $5 gant $1 ($4 {{PLURAL:$4|adweladenn}})",
        "logentry-merge-merge": "$1 {{GENDER:$2|en deus lakaet|he deus lakaet}} $3 da gendeuziñ e $4 (adweladennoù betek $5)",
-       "logentry-move-move": "$1 en deus kaset ar bajenn $3 da $4",
-       "logentry-move-move-noredirect": "kaset ar bajenn $3 da $4 gant $1 hep adkas",
-       "logentry-move-move_redir": "kaset ar bajenn $3 da $4 gant $1 dreist un adkas",
-       "logentry-move-move_redir-noredirect": "kaset ar bajenn $3 da $4 gant $1 dreist un adkas hep lezel un adkas",
+       "logentry-move-move": "$1 {{GENDER:$2|en|he|en}} deus kaset ar bajenn $3 da $4",
+       "logentry-move-move-noredirect": "Kaset eo bet ar bajenn $3 da $4 gant $1 hep lezel un adkas",
+       "logentry-move-move_redir": "Kaset eo bet ar bajenn $3 da $4 gant $1 dreist un adkas",
+       "logentry-move-move_redir-noredirect": "Kaset eo bet ar bajenn $3 da $4 gant $1 dreist un adkas hep lezel un adkas",
        "logentry-patrol-patrol": "{{GENDER:$2|Merket}} eo bet an adweladenn $4 eus ar bajenn $3 evel gwiriet gant $1",
        "logentry-patrol-patrol-auto": "{{GENDER:$2|Merket}} eo bet ent emgefre an adweladenn $4 eus ar bajenn $3 evel gwiriet gant $1",
        "logentry-newusers-newusers": "{{GENDER:$2|Krouet}} eo bet ar gont implijer $1",
        "authmanager-email-help": "Chomlec'h postel",
        "authmanager-realname-label": "Anv gwir",
        "authmanager-realname-help": "Anv gwir an implijer",
-       "authmanager-provider-password": "Gwiriekadur diazezet war ur ger-termen",
+       "authmanager-provider-password": "Gwiriekadur diazezet war ur ger-tremen",
        "authmanager-provider-password-domain": "Dilesadur diazezet war ur ger-tremen hag an domani",
        "authmanager-provider-temporarypassword": "Ger-tremen da c'hortoz",
        "authprovider-confirmlink-message": "Diwar an taolioù-esae diwezhañ graet ganeoc'h evit kevreañ e c'haller liammañ ar c'hontoù da heul ouzh ho kont wiki. Ur wech liammet e c'hallit kevreañ dre ar c'hontoù-se. Diuzit ar re zo da vezañ liammet.",
        "undelete-cantcreate": "N'hallit ket diziverkañ ar bajenn-mañ rak n'eus pajenn ebet gant an anv-mañ ha n'oc'h ket aotreet da grouiñ ar bajenn-mañ.",
        "pagedata-title": "Roadennoù ar bajenn",
        "pagedata-bad-title": "Titl direizh : $1.",
+       "passwordpolicies": "Reolennoù e keñver ar ger-tremen",
        "passwordpolicies-group": "Strollad",
        "passwordpolicies-policies": "Politikerezh",
        "passwordpolicies-policy-passwordcannotmatchusername": "Ar ger-kuzh ne c'hell ket bezañ an anv implijer"
index 65ab290..839a5dc 100644 (file)
        "autoblockedtext": "La vostra adreça IP ha estat blocada automàticament perquè va ser usada per un usuari actualment blocat. Aquest usuari va ser blocat per l'{{GENDER:$1|administrador|administradora}} $1. El motiu donat per al blocatge és aquest:\n\n:<em>$2</em>\n\n* Inici del blocatge: $8\n* Final del blocatge: $6\n* Usuari blocat: $7\n\nPodeu contactar l'usuari $1 o algun altre dels [[{{MediaWiki:Grouppage-sysop}}|administradors]] per a discutir el blocatge.\n\nRecordeu que per a poder usar l'opció «{{int:emailuser}}» haureu d'haver validat una adreça de correu electrònic a les vostres [[Special:Preferences|preferències]].\n\nEl número d'identificació de la vostra adreça IP és $3, i l'ID del blocatge és #$5. Si us plau, incloeu aquestes dades en totes les consultes que feu.",
        "systemblockedtext": "El vostre nom d'usuari o adreça IP ha estat blocada automàticament pel MediaWiki.\nEl motiu donat és:\n\n:<em>$2</em>\n\n* Inici del blocatge: $8\n* Caducitat del blocatge: $6\n* Destinatari del blocatge: $7\n\nLa vostra adreça IP actual és $3.\nAfegiu les dades de més amunt en qualsevol consulta que feu al respecte.",
        "blockednoreason": "no s'ha donat cap motiu",
+       "blockedtext-composite-no-ids": "La vostra adreça IP apareix en diferents llistes negres",
        "blockedtext-composite-reason": "Hi ha diferents blocatges associats al vostre compte i/o adreça IP",
        "whitelistedittext": "Heu de $1 per modificar pàgines.",
        "confirmedittext": "Heu de confirmar la vostra adreça electrònica abans de poder modificar les pàgines. Definiu i valideu la vostra adreça electrònica a través de les vostres [[Special:Preferences|preferències d'usuari]].",
        "edit-no-change": "S'ha ignorat la vostra modificació perquè no feia cap canvi al text.",
        "edit-slots-cannot-add": "{{PLURAL:$1|L'espai no està|Els espais no estan}} implementats aquí: $2.",
        "edit-slots-cannot-remove": "{{PLURAL:$1|L'espai següent és necessari i no es pot suprimir|Els espais següents són necessaris i no es poden suprimir}}: $2.",
+       "edit-slots-missing": "Hi {{PLURAL:$1|manca l'espai següent|manquen els espais següents}}: $2.",
        "postedit-confirmation-created": "S'ha creat la pàgina.",
        "postedit-confirmation-restored": "S'ha restaurat la pàgina.",
        "postedit-confirmation-saved": "S'ha desat la modificació.",
        "grant-editmycssjs": "Modifiqueu el vostre CSS/JSON/JavaScript d'usuari",
        "grant-editmyoptions": "Editeu les vostres preferències d'usuari i la configuració JSON",
        "grant-editmywatchlist": "Modifica la llista de seguiment",
+       "grant-editsiteconfig": "Modificar els CSS/JS del lloc web i d'usuari",
        "grant-editpage": "Modifica les pàgines existents",
        "grant-editprotected": "Modifica pàgines protegides",
        "grant-highvolume": "Edició d'alt volum",
        "action-changetags": "afegeix i elimina etiquetes a les revisions i les entrades de registre individuals",
        "action-deletechangetags": "eliminar etiquetes des de la base de dades",
        "action-purge": "purga la pàgina",
+       "action-apihighlimits": "utilitza límits majors en les consultes API",
+       "action-bigdelete": "eliminar les pàgines amb grans historials",
        "action-blockemail": "blocar un usuari per tal que no enviï correu",
        "action-bot": "ser tractat com un procés automatitzat",
        "action-editprotected": "modificar pàgines protegides com «{{int:protect-level-sysop}}»",
        "action-viewsuppressed": "mostrar revisions amagades de qualsevol usuari",
        "action-hideuser": "blocar un nom d'usuari, amagant-lo del públic",
        "action-unblockself": "desblocar-se un mateix",
+       "action-reupload-own": "sobreescriure fitxers existents que hàgiu carregat",
+       "action-suppressredirect": "no crear redireccions de les pàgines d'origen en reanomenar-les",
        "nchanges": "$1 {{PLURAL:$1|canvi|canvis}}",
        "enhancedrc-since-last-visit": "$1 {{PLURAL:$1|des de la darrera visita}}",
        "enhancedrc-history": "historial",
        "rcfilters-clear-all-filters": "Esborra tots els filtres",
        "rcfilters-show-new-changes": "Mostra els canvis nous des del $1",
        "rcfilters-search-placeholder": "Filtra els canvis recents (utilitzeu el menú o cerqueu el nom del filtre)",
+       "rcfilters-search-placeholder-mobile": "Filtres",
        "rcfilters-invalid-filter": "Filtre no vàlid",
        "rcfilters-empty-filter": "No hi ha cap filtre actiu. Es mostren totes les contribucions.",
        "rcfilters-filterlist-title": "Filtres",
        "ipb-confirm": "Confirma el blocatge",
        "ipb-sitewide": "A tot el lloc web",
        "ipb-partial": "Parcial",
+       "ipb-sitewide-help": "Cada pàgina en el wiki i totes les altres accions de contribució.",
        "ipb-partial-help": "Pàgines específiques o espais de noms.",
        "ipb-pages-label": "Pàgines",
        "ipb-namespaces-label": "Espais de noms",
        "move-watch": "Vigila aquesta pàgina",
        "movepagebtn": "Reanomena la pàgina",
        "pagemovedsub": "Reanomenament amb èxit",
+       "cannotmove": "No s'ha pogut reanomenar la pàgina {{PLURAL:$1|pel motiu següent|pels motius següents}}:",
        "movepage-moved": "'''«$1» s'ha mogut a «$2»'''",
        "movepage-moved-redirect": "S'ha creat una redirecció.",
        "movepage-moved-noredirect": "La creació d'una redirecció s'ha suprimit.",
        "confirm-mcrrestore-title": "Restaura una revisió",
        "confirm-mcrundo-title": "Desfés un canvi",
        "mcrundofailed": "Ha fallat el desfer",
+       "mcrundo-missingparam": "Manquen paràmetres obligatoris en la sol·licitud.",
        "mcrundo-changed": "La pàgina ha canviat d'ençà que heu vist la diferència. Reviseu el nou canvi.",
        "mcrundo-parse-failed": "No s'ha pogut analitzar la revisió nova: $1",
        "colon-separator": ":&#32;",
        "htmlform-date-invalid": "El valor que heu especificat no és una data reconeguda. Proveu d'utilitzar el format AAAA-MM-DD.",
        "htmlform-time-invalid": "El valor que heu especificat no és una hora reconeguda. Proveu d'utilitzar el format HH:MM:SS.",
        "htmlform-datetime-invalid": "El valor que heu especificat no és una data i hora reconeguda. Proveu d'utilitzar el format AAAA-MM-DD HH:MM:SS.",
+       "htmlform-time-toohigh": "El valor que heu especificat és posterior a la darrera hora permesa de $1.",
        "htmlform-datetime-toolow": "El valor que heu especificat és anterior a la data i hora més antiga permeses, $1.",
        "htmlform-datetime-toohigh": "El valor que heu especificat és posterior a la data i hora permeses, $1.",
        "htmlform-title-badnamespace": "[[:$1]] no es troba en l'espai de noms \"{{ns:$2}}\".",
        "logentry-suppress-block": "$1 {{GENDER:$2|ha blocat}} {{GENDER:$4|$3}} per un temps de $5 $6",
        "logentry-suppress-reblock": "$1 {{GENDER:$2|ha canviat}} la configuració de blocatge de {{GENDER:$4|$3}} per un temps de $5 $6",
        "logentry-import-upload": "$1 {{GENDER:$2|va importar}} $3 a través de càrrega de fitxer",
+       "logentry-import-upload-details": "$1 {{GENDER:$2|ha importat}} $3 per càrrega de fitxer ($4 {{PLURAL:$4|revisió|revisions}})",
        "logentry-import-interwiki": "$1 {{GENDER:$2|va importar}} $3 d'un altre wiki",
+       "logentry-import-interwiki-details": "$1 {{GENDER:$2|ha importat}} $3 de $5 ($4 {{PLURAL:$4|revisió|revisions}})",
        "logentry-merge-merge": "$1 {{GENDER:$2|ha fusionat}} $3 en $4 (revisions fins a $5)",
        "logentry-move-move": "$1 ha mogut $3 a $4",
        "logentry-move-move-noredirect": "$1 ha mogut $3 a $4 sense deixar una redirecció",
        "logentry-managetags-activate": "$1 {{GENDER:$2|ha activat}} l'etiqueta \"$4\" per a ser utilitzada en usuaris i bots",
        "logentry-managetags-deactivate": "$1 {{GENDER:$2|ha desactivat}} l'etiqueta \"$4\" per a ser utilitzada en usuaris i bots",
        "log-name-tag": "Registre d'etiquetes",
+       "logentry-tag-update-add-revision": "$1 {{GENDER:$2|ha afegit}} {{PLURAL:$7|l'etiqueta|les etiquetes}} $6 a la revisió $4 de la pàgina $3",
        "rightsnone": "(cap)",
        "rightslogentry-temporary-group": "$1 (temporal, fins a $2)",
        "feedback-adding": "S'està afegint el comentari a la pàgina...",
        "edit-error-short": "Error: $1",
        "edit-error-long": "Errors:\n\n$1",
        "specialmute": "Silencia",
+       "specialmute-success": "S'han actualitzat les vostres preferències de silenciament. Vegeu tots els usuaris silenciats a les [[Special:Preferences|your preferències]].",
        "specialmute-submit": "Confirma",
        "specialmute-label-mute-email": "Silencia el correu electrònic d'aquest usuari",
+       "specialmute-header": "Seleccioneu les vostres preferències de silenciament de l'usuari <b>{{BIDI:[[User:$1|$1]]}}</b>.",
        "specialmute-error-invalid-user": "No s’ha trobat el nom d’usuari que heu indicat.",
        "specialmute-login-required": "Inicieu una sessió per canviar les preferències de silenciament.",
+       "mute-preferences": "Preferències de silenciament",
        "revid": "revisió $1",
        "pageid": "ID de pàgina $1",
        "rawhtml-notallowed": "No és possible fer servir les etiquetes &lt;html&gt; fora de les pàgines normals.",
index 88e5bc1..7081e67 100644 (file)
        "lockmanager-fail-acquirelock": "«$1» блоктоха цатарло.",
        "lockmanager-fail-openlock": "Блоктохаран «$1» файл схьаелла цаелира.",
        "lockmanager-fail-releaselock": "\"$1\" блокдӀаяккха цаелира.",
+       "zip-file-open-error": "Архивна талламбархьама файл схьаеллаш гӀалат даьлла.",
+       "zip-wrong-format": "Билгалйина файл — ZIP-архив яц.",
+       "zip-unsupported": "ХӀокху ZIP-файло лелош ю таронаш, MediaWiki ловш йоцу.\nИза кхочуш талла еш яц.",
        "uploadstash": "Къайлаха чуяккхар",
        "uploadstash-clear": "ДӀацӀанъян къайла йолу файлаш",
        "uploadstash-nofiles": "Хьан къайла файлаш яц.",
        "allmessages-filter-unmodified": "Хийцан йоцурш",
        "allmessages-filter-all": "Массо",
        "allmessages-filter-modified": "Хийцнарш",
-       "allmessages-prefix": "Ð\9bÑ\83Ñ\8cÑ\82Ñ\82Ñ\83Ñ\80г Ð¾Ñ\86Ñ\83 Ð´ÐµÑ\88аÑ\85Ñ\8cалÑ\85е:",
+       "allmessages-prefix": "Ð\9bÑ\83Ñ\8cÑ\82Ñ\82Ñ\83Ñ\80г Ð¿Ñ\80еÑ\84икÑ\81Ñ\86а:",
        "allmessages-language": "Мотт:",
        "allmessages-filter-submit": "Дехьа гӀо",
        "allmessages-filter-translate": "Гочйе",
index 3bb077b..4da930e 100644 (file)
        "rcfilters-clear-all-filters": "Zrušit všechny filtry",
        "rcfilters-show-new-changes": "Zobrazit změny od $1",
        "rcfilters-search-placeholder": "Filtrovat poslední změny (použijte menu nebo vyhledejte název filtru)",
+       "rcfilters-search-placeholder-mobile": "Filtry",
        "rcfilters-invalid-filter": "Neplatný filtr",
        "rcfilters-empty-filter": "Žádné aktivní filtry. Zobrazeny jsou všechny příspěvky.",
        "rcfilters-filterlist-title": "Filtry",
        "move-page-legend": "Přesunout stránku",
        "movepagetext": "Použitím tohoto formuláře změníte název stránky a přesunete i celou její historii na nový název.\nPůvodní název se stane přesměrováním na nový název.\nPřesměrování na původní název můžete nechat aktualizovat automaticky.\nPokud nenecháte, nezapomeňte poté zkontrolovat [[Special:DoubleRedirects|dvojitá]] nebo [[Special:BrokenRedirects|přerušená]] přesměrování.\nJe vaší zodpovědností zajistit, aby odkazy stále vedly tam, kam mají.\n\nStránku <strong>není možné</strong> přejmenovat, pokud pod cílovým názvem již nějaká stránka existuje, s výjimkou situace, kdy je cílová stránka přesměrováním na tuto stránku a nemá žádnou historii editací.\nTo znamená, že stránku můžete přesunout zpět na původní název, pokud uděláte chybu, a že nemůžete přepsat existující stránku.\n\n<strong>Poznámka:</strong>\nPřejmenování oblíbené stránky může být drastická a nečekaná změna;\npředtím, než změnu provedete, se ujistěte, že chápete důsledky svého kroku.",
        "movepagetext-noredirectfixer": "Použitím tohoto formuláře změníte název stránky a přesunete i celou její historii na nový název.\nPůvodní název se stane přesměrováním na nový název.\nNezapomeňte poté zkontrolovat [[Special:DoubleRedirects|dvojitá]] nebo [[Special:BrokenRedirects|přerušená]] přesměrování.\nJe vaší zodpovědností zajistit, aby odkazy stále vedly tam, kam mají.\n\nStránku <strong>není možno</strong> přejmenovat, pokud pod cílovým názvem již nějaká stránka existuje, s výjimkou situace, kdy je cílová stránka prázdná nebo je přesměrováním na tuto stránku a nemá žádnou historii editací.\nTo znamená, že stránku můžete přesunout zpět na původní název, pokud uděláte chybu, a že nemůžete přepsat existující stránku.\n\n<strong>Poznámka:</strong>\nPřejmenování oblíbené stránky může být drastická a nečekaná změna; předtím, než změnu provedete, se prosím ujistěte, že chápete důsledky svého kroku.",
+       "movepagetext-noredirectsupport": "Použitím tohoto formuláře změníte název stránky a přesunete i celou její historii na nový název.\nJe vaší zodpovědností zajistit, aby odkazy stále vedly tam, kam mají.\n\nStránku <strong>není možné</strong> přejmenovat, pokud pod cílovým názvem již nějaká stránka existuje.\nTo znamená, že stránku můžete přesunout zpět na původní název, pokud uděláte chybu, a že nemůžete přepsat existující stránku.\n\n<strong>Poznámka:</strong>\nPřejmenování oblíbené stránky může být drastická a nečekaná změna;\npředtím, než změnu provedete, se ujistěte, že chápete důsledky svého kroku.",
        "movepagetalktext": "Pokud zaškrtnete toto pole, přidružená diskusní stránka bude automaticky přesunuta na nový název, leda by tam již neprázdná diskusní stránka existovala.\n\nV takovém případě musíte stránky přesunout nebo sloučit ručně, přejete-li si to.",
        "moveuserpage-warning": "'''Upozornění:''' Chystáte se přesunout uživatelskou stránku. Uvědomte si prosím, že bude přesunuta pouze tato stránka, ale uživatel ''nebude'' přejmenován.",
        "movecategorypage-warning": "<strong>Upozornění:</strong> Chystáte se přesunout stránku kategorie. Uvědomte si, že bude přesunuta pouze tato stránka a že žádné stránky v původní kategorii <em>nebudou</em> do nové překategorizovány.",
index e468acc..55157d9 100644 (file)
@@ -99,7 +99,8 @@
                        "Johanna Strodt (WMDE)",
                        "Andi-3",
                        "1233qwer1234qwer4",
-                       "MarkusRost"
+                       "MarkusRost",
+                       "Mcandri13"
                ]
        },
        "tog-underline": "Links unterstreichen:",
        "systemblockedtext": "Dein Benutzername oder deine IP-Adresse wurde von MediaWiki automatisch gesperrt.\nDer angegebene Grund ist:\n\n:<em>$2</em>\n\n* Beginn der Sperre: $8\n* Ablauf der Sperre: $6\n* Sperre betrifft: $7\n\nDeine aktuelle IP-Adresse ist $3.\nBitte gib alle oben stehenden Details in jeder Anfrage an.",
        "blockednoreason": "keine Begründung angegeben",
        "blockedtext-composite": "<strong>Dein Benutzername oder deine IP-Adresse wurde gesperrt.</strong>\n\nDer Angegebene Grund ist:\n\n:<em>$2</em>\n\n* Beginn der Sperre: $8\n* Ablauf der längsten Sperre: $6\n\n* $5\n\nDeine aktuelle IP-Adresse ist $3.\nBitte gib alle oben stehenden Details in jeder Anfrage an.",
+       "blockedtext-composite-no-ids": "Deine IP-Adresse taucht in mehreren Sperrlisten auf",
        "blockedtext-composite-reason": "Es gibt mehrere Sperren gegen dein Benutzerkonto und/oder deine IP-Adresse",
        "whitelistedittext": "Du musst dich $1, um Seiten bearbeiten zu können.",
        "confirmedittext": "Du musst deine E-Mail-Adresse erst bestätigen, bevor du Bearbeitungen durchführen kannst. Bitte ergänze und bestätige deine E-Mail in den [[Special:Preferences|Einstellungen]].",
index 01e1007..a509502 100644 (file)
        "rcfilters-clear-all-filters": "Clear all filters",
        "rcfilters-show-new-changes": "View new changes since $1",
        "rcfilters-search-placeholder": "Filter changes (use menu or search for filter name)",
+       "rcfilters-search-placeholder-mobile": "Filters",
        "rcfilters-invalid-filter": "Invalid filter",
        "rcfilters-empty-filter": "No active filters. All contributions are shown.",
        "rcfilters-filterlist-title": "Filters",
        "changecontentmodel" : "Change content model of a page",
        "changecontentmodel-legend": "Change content model",
        "changecontentmodel-title-label": "Page title",
+       "changecontentmodel-current-label": "Current content model:",
        "changecontentmodel-model-label": "New content model",
        "changecontentmodel-reason-label": "Reason:",
        "changecontentmodel-submit": "Change",
        "block-log-flags-angry-autoblock": "enhanced autoblock enabled",
        "block-log-flags-hiddenname": "username hidden",
        "range_block_disabled": "The administrator ability to create range blocks is disabled.",
+       "ipb-prevent-user-talk-edit": "Editing their own talk page must be allowed for a partial block, unless it includes a restriction on the User Talk namespace.",
        "ipb_expiry_invalid": "Expiry time invalid.",
        "ipb_expiry_old": "Expiry time is in the past.",
        "ipb_expiry_temp": "Hidden username blocks must be permanent.",
        "move-page-legend": "Move page",
        "movepagetext": "Using the form below will rename a page, moving all of its history to the new name.\nThe old title will become a redirect page to the new title.\nYou can update redirects that point to the original title automatically.\nIf you choose not to, be sure to check for [[Special:DoubleRedirects|double]] or [[Special:BrokenRedirects|broken redirects]].\nYou are responsible for making sure that links continue to point where they are supposed to go.\n\nNote that the page will <strong>not</strong> be moved if there is already a page at the new title, unless the latter is a redirect and has no past edit history.\nThis means that you can rename a page back to where it was renamed from if you make a mistake, and you cannot overwrite an existing page.\n\n<strong>Note:</strong>\nThis can be a drastic and unexpected change for a popular page;\nplease be sure you understand the consequences of this before proceeding.",
        "movepagetext-noredirectfixer": "Using the form below will rename a page, moving all of its history to the new name.\nThe old title will become a redirect page to the new title.\nBe sure to check for [[Special:DoubleRedirects|double]] or [[Special:BrokenRedirects|broken redirects]].\nYou are responsible for making sure that links continue to point where they are supposed to go.\n\nNote that the page will <strong>not</strong> be moved if there is already a page at the new title, unless it is a redirect and has no past edit history.\nThis means that you can rename a page back to where it was renamed from if you make a mistake, and you cannot overwrite an existing page.\n\n<strong>Note:</strong>\nThis can be a drastic and unexpected change for a popular page;\nplease be sure you understand the consequences of this before proceeding.",
+       "movepagetext-noredirectsupport": "Using the form below will rename a page, moving all of its history to the new name.\nYou are responsible for making sure that links continue to point where they are supposed to go.\n\nNote that the page will <strong>not</strong> be moved if there is already a page at the new title.\nThis means that you can rename a page back to where it was renamed from if you make a mistake, and you cannot overwrite an existing page.\n\n<strong>Note:</strong>\nThis can be a drastic and unexpected change for a popular page;\nplease be sure you understand the consequences of this before proceeding.",
        "movepagetalktext": "If you check this box, the associated talk page will be automatically moved to new title, unless a non-empty talk page already exists there.\n\nIn this case, you will have to move or merge the page manually if desired.",
        "moveuserpage-warning": "<strong>Warning:</strong> You are about to move a user page. Please note that only the page will be moved and the user will <em>not</em> be renamed.",
        "movecategorypage-warning": "<strong>Warning:</strong> You are about to move a category page. Please note that only the page will be moved and any pages in the old category will <em>not</em> be recategorized into the new one.",
        "linkaccounts": "Link accounts",
        "linkaccounts-success-text": "The account was linked.",
        "linkaccounts-submit": "Link accounts",
+       "cannotunlink-no-provider-title": "There are no linked accounts to unlink",
+       "cannotunlink-no-provider": "There are no linked accounts that can be unlinked.",
        "unlinkaccounts": "Unlink accounts",
        "unlinkaccounts-success": "The account was unlinked.",
        "authenticationdatachange-ignored": "The authentication data change was not handled. Maybe no provider was configured?",
index ee1d97a..93c5f93 100644 (file)
        "search-interwiki-more": "(plu)",
        "search-interwiki-more-results": "pliaj rezultoj",
        "search-relatedarticle": "Relataj",
+       "search-invalid-sort-order": "Ordiga metodo de $1 ne estas rekonata; implicita ordiga metodo uziĝos. Jen validaj ordigaj metodoj: $2",
+       "search-unknown-profile": "Serĉa profilo de $1 ne estas rekonata; implicita serĉa profilo uziĝos.",
        "searchrelated": "rilataj",
        "searchall": "ĉiuj",
        "showingresults": "Montras {{PLURAL:$1|'''1''' trovitan|'''$1''' trovitajn}} ekde la #'''$2'''-a.",
        "right-editmyusercss": "Redakti viajn proprajn CSS-dosierojn",
        "right-editmyuserjson": "Redakti viajn proprajn JSON-dosierojn",
        "right-editmyuserjs": "Redakti viajn proprajn JavaScript-dosierojn",
+       "right-editmyuserjsredirect": "Redakti viajn proprajn JavaScript-dosierojn kiuj estas alidirektiloj",
        "right-viewmywatchlist": "Vidi vian propran atentaron",
        "right-editmywatchlist": "Redakti vian propran atentaron. Notu, ke kelkaj agoj ankoraŭ ebligas aldoni paĝojn sen ĉi tiu rajto.",
        "right-viewmyprivateinfo": "Vidi viajn proprajn privatajn informojn (ekz. retpoŝtan adreson, veran nomon)",
        "action-editmyusercss": "redakti proprajn CSS-dosierojn",
        "action-editmyuserjson": "redakti proprajn JSON-dosierojn",
        "action-editmyuserjs": "redakti proprajn JavaScript-dosierojn",
+       "action-editmyuserjsredirect": "redakti viajn proprajn JavaScript-dosierojn kiuj estas alidirektiloj",
        "action-viewsuppressed": "vidi versiojn kaŝitajn for de ajna uzanto",
        "action-hideuser": "forbari uzantnomon, kaŝante ĝin de la publiko",
        "action-ipblock-exempt": "preterpasi forbarojn de IP-adresoj, aŭtomatajn forbarojn, kaj forbarojn de IP-adresaj intervaloj",
        "rcfilters-clear-all-filters": "Nuligi ĉiujn filtrilojn",
        "rcfilters-show-new-changes": "Vidi la novajn ŝanĝojn ekde $1",
        "rcfilters-search-placeholder": "Filtri lastajn ŝanĝojn (vi povas elekti aŭ ekskribi)",
+       "rcfilters-search-placeholder-mobile": "Filtriloj",
        "rcfilters-invalid-filter": "Nevalida filtrilo",
        "rcfilters-empty-filter": "Ekzistas neniuj aktivaj filtriloj. Ĉiuj kontribuaĵoj estas montritaj.",
        "rcfilters-filterlist-title": "Filtriloj",
        "block-log-flags-angry-autoblock": "progresa aŭtoforbaro ebliĝis",
        "block-log-flags-hiddenname": "salutnomo kaŝita",
        "range_block_disabled": "La ebleco de administranto krei forbaritajn intervalojn da IP-adresoj estas malebligita.",
+       "ipb-prevent-user-talk-edit": "Redaktado de onia propra diskutpaĝo devas esti permesata pri parta forbaro, krom se ĝi inkluzivas malpermeson pri la uzanto-diskutpaĝa nomspaco.",
        "ipb_expiry_invalid": "Nevalida blokdaŭro.",
        "ipb_expiry_old": "Limdato antaŭas la nuntempon.",
        "ipb_expiry_temp": "Kaŝitaj salutnomaj blokoj estu daŭraj.",
        "move-page-legend": "Alinomi paĝon",
        "movepagetext": "Per la jena formulo vi povas ŝanĝi la nomon de iu paĝo, kunportante ĝian historion de redaktoj al la nova nomo.\nLa antaŭa titolo fariĝos alidirektilo al la nova titolo.\nVi povas ĝisdatigi alidirektilojn kiu indikas la originalan titolon aŭtomate.\nSe vi elektas ĝisdatigi permane, bonvolu kontroli [[Special:DoubleRedirects|duoblajn]] aŭ [[Special:BrokenRedirects|rompitajn alidirektilojn]].\nVi estas responsa por certigi ke ligilojn direktas fidinde.\n\nNotu, ke la paĝo '''ne''' estos movita se jam ekzistas paĝo ĉe la nova titolo, krom se tiu loko estas malplena aŭ alidirektilo al ĉi tiu paĝo, kaj sen antaŭa redaktohistorio.\nPro tio, vi ja povos removi la paĝon je la antaŭa titolo se vi mistajpus, kaj ne povas forviŝi ekzistantan paĝon per movo.\n\n'''Note:'''\nTio povas esti drasta kaj neatendita ŝanĝo por populara paĝo;\nbonvolu certigi vin, ke vi komprenas ties konsekvencojn antaŭ ol vi antaŭeniru.",
        "movepagetext-noredirectfixer": "Per jena formularo vi povas alinomigi paĝon, kaj movi tutan ĝian redaktohistorion al la nova nomo. \nLa antaŭa titolo alidirektigos onin al la nova titolo.\nKontrolu pri [[Special:DoubleRedirects|duoblajn]] aŭ [[Special:BrokenRedirects|nefunkciantajn alidirektilojn]].\nVi respondecas pri tio ke ligoj restas montrantaj ĝustadirekten.\n\nKonsciu ke la paĝo <strong>ne</strong> estas movota se jam ekzistas paĝo havanta la novan titolon, krom se ĝi estas alidirektilo sen antaŭa redaktohistorio.\nTio ĉi signifas ke vi povas alinomigi paĝon reen al antaŭa nomo se vi eraras, kaj vi ke vi ne povas anstataŭigi ekzistantan paĝon.\n\n<strong>Rimarko:</strong>\nEblas ke tio ĉi estas drasta kaj neatendita ŝanĝo de populara paĝo;\nAntaŭ daŭrigi, bonvolu certiĝi, ke vi komprenas la konsekvencojn de tiuj ĉi ŝanĝo.",
+       "movepagetext-noredirectsupport": "La jena formularo renomas paĝon, movante ĉiom da ĝia historio al la nova nomo.\nVi respondecas pri certigo ke ligiloj daŭre indikas la ĝustajn celojn.\n\nNotu ke la paĝo <strong>ne</strong> estos movita se jam ekzistas paĝo ĉe la nova titolo.\nTial, vi povas renomi paĝon reen al la originala titolo se vi misrenomis, kaj vi ne povas anstataŭigi ekzistantan paĝon.\n\n<strong>Noto:</strong>\nĈi tio estas eble drasta kaj neatendata ŝanĝo pri populara paĝo;\nbonvolu certigu ke vi komprenas la konsekvencon de tio, antaŭ ol fari tion.",
        "movepagetalktext": "Se vi validas tiun elektobutono, la asociata diskutpaĝo estos aŭtomate alinomita al nova titolo, krom se malplena diskutpaĝo jam ekzistas.\n\nTiujokaze, vi alinomigendos aŭ kunfandendos malaŭtomate la paĝon se vi tion deziras.",
        "moveuserpage-warning": "<strong>Averto:</strong> Vi preskaŭ alinomas paĝon de uzanto. Bonvolu noti ke nur la paĝo estos alinomita kaj la uzanto mem <em>ne</em> estos alinomita.",
        "movecategorypage-warning": "<strong>Averto:</strong> Vi baldaŭ movos kategorian paĝon. Bonvolu noti ke, nur la paĝo estos movita, kaj la paĝoj en la malnova kategorio <em>ne</em> transiros en la novan kategorion.",
        "linkaccounts": "Ligi kontojn",
        "linkaccounts-success-text": "La konto estis ligita.",
        "linkaccounts-submit": "Ligi kontojn",
+       "cannotunlink-no-provider-title": "Ne estas ligitaj kontoj por malligi",
+       "cannotunlink-no-provider": "Ne estas ligitaj kontoj, kiuj povas esti malligotaj.",
        "unlinkaccounts": "Malligi kontojn",
        "unlinkaccounts-success": "La konto estis malligita.",
        "authenticationdatachange-ignored": "La ŝanĝo de datumoj pri aŭtentigado ne estis traktita. Eble neniu provizanto estis agordita?",
        "specialmute-label-mute-email": "Kaŝi retmesaĝojn el ĉi tiu uzanto",
        "specialmute-header": "Bonvolu elekti viajn preferojn pri kaŝado de mesaĝoj el uzanto <b>{{BIDI:[[User:$1]]}}</b>.",
        "specialmute-error-invalid-user": "La petita uzantnomo ne troviĝis.",
-       "specialmute-email-footer": "[$1 Administri preferojn pri retpoŝto por {{BIDI:$2}}.]",
+       "specialmute-error-no-options": "Silentiga funkcio ne estas havebla. Tio esta eble ĉar: vi ne jam konfirmis vian retpoŝtan adreson, aŭ la vikia administranto malfunkciigis retpoŝtajn funkciojn kaj/aŭ retpoŝtan forbarliston pri ĉi tiu vikio.",
+       "specialmute-email-footer": "Por administri preferojn pri retpoŝto por la uzanto {{BIDI:$2}}, bonvolu viziti la paĝon <$1>.",
        "specialmute-login-required": "Bonvolu ensaluti por konservi vian preferon pri kaŝado de mesaĝoj.",
        "mute-preferences": "Agordoj pri silentigo",
        "revid": "revizio $1",
        "passwordpolicies-policy-passwordnotinlargeblacklist": "Pasvorto ne povas esti unu el la cent mil plej popularaj pasvortoj",
        "passwordpolicies-policyflag-forcechange": "devas ŝanĝi dum ensaluto",
        "passwordpolicies-policyflag-suggestchangeonlogin": "sugesti ŝanĝadon dum ensaluto",
+       "mycustomjsredirectprotected": "Vi ne havas permeson redakti ĉi tiun JavaScript-paĝon, ĉar ĝi estas alidirektilo, kies celo ne estas en via uzantospaco.",
        "easydeflate-invaliddeflate": "Provizita enhavo ne estas ĝuste densigita",
        "unprotected-js": "Pro sekurecaj kialoj, JavaScript ne povas esti ŝargata el neprotektataj paĝoj. Bonvolu nur krei JavaScript en la nomspaco MediaWiki: aŭ kiel subpaĝo de Uzanto.",
        "userlogout-continue": "Ĉu vi volas elsaluti?"
index 70add63..ebc2405 100644 (file)
        "systemblockedtext": "Tu nombre de usuario o dirección IP ha sido bloqueado automáticamente por el software MediaWiki.\nLa razón dada es:\n\n:<em>$2</em>\n\n* Inicio del bloqueo: $8\n* Caducidad de bloqueo: $6\n* Destinatario del bloqueo: $7\n\nTu dirección IP actual es $3.\nPor favor, incluye todos los datos aquí mostrados en cualquier consulta que hagas.",
        "blockednoreason": "no se ha especificado el motivo",
        "blockedtext-composite": "<strong>Tu nombre de usuario o dirección IP han sido bloqueados.</strong>\n\nLa razón es:\n\n:<em>$2</em>.\n\n* Inicio del bloqueo: $8\n* Vencimiento del bloqueo más largo: $6\n\n* $5\n\nTu dirección IP actual es $3.\nPor favor, incluye todos los detalles anteriores en cualquier consulta que realices.",
+       "blockedtext-composite-ids": "Identificadores de bloqueo relevantes: $1 (tu dirección IP puede encontrarse también en alguna lista negra)",
        "blockedtext-composite-no-ids": "Tu dirección IP figura en varias listas negras",
-       "blockedtext-composite-reason": "Hay múltiples bloques contra tu cuenta y/o dirección IP.",
+       "blockedtext-composite-reason": "Existen múltiples bloqueos contra tu cuenta y/o dirección IP",
        "whitelistedittext": "Tienes que $1 para editar páginas.",
        "confirmedittext": "Debes confirmar tu dirección de correo electrónico antes de poder editar páginas. Por favor, configura y confirma tu dirección de correo a través de tus [[Special:Preferences|preferencias de usuario]].",
        "nosuchsectiontitle": "Sección no encontrada",
        "right-block": "Bloquear a otros usuarios para que no editen",
        "right-blockemail": "Bloquear a un usuario para que no pueda mandar correos electrónicos",
        "right-hideuser": "Bloquear un nombre de usuario, haciéndolo invisible",
-       "right-ipblock-exempt": "Evitar bloqueos a IP, automáticos y por intervalos",
+       "right-ipblock-exempt": "Evitar bloqueos de IP, bloqueos automáticos y bloqueos de rango",
        "right-unblockself": "Desbloquearse a sí mismo",
        "right-protect": "Cambiar niveles de protección y editar páginas protegidas en cascada",
        "right-editprotected": "Editar páginas protegidas como «{{int:protect-level-sysop}}»",
        "right-editmyusercss": "Editar tus archivos CSS",
        "right-editmyuserjson": "Editar tus propias páginas en formato JSON",
        "right-editmyuserjs": "Editar tus archivos JavaScript",
+       "right-editmyuserjsredirect": "Editar tus propias páginas de usuario en formato JavaScript cuando sean redirecciones",
        "right-viewmywatchlist": "Ver su propia lista de seguimiento",
        "right-editmywatchlist": "Editar su propia lista de seguimiento (algunas acciones seguirán añadiendo páginas aun sin este permiso).",
        "right-viewmyprivateinfo": "Ver su propia información privada (ej.: correo electrónico, nombre real)",
        "action-editmyusercss": "editar tus propios archivos CSS",
        "action-editmyuserjson": "editar tus propios archivos JSON",
        "action-editmyuserjs": "editar tus propios archivos JavaScript",
+       "action-editmyuserjsredirect": "editar tus propias páginas de usuario en formato JavaScript cuando sean redirecciones",
        "action-viewsuppressed": "ver revisiones ocultas de cualquier usuario",
        "action-hideuser": "bloquear un nombre de usuario, haciéndolo invisible",
-       "action-ipblock-exempt": "evitar bloques de IP, autobloqueos y bloqueos a distancia",
+       "action-ipblock-exempt": "evitar bloqueos de IP, bloqueos automáticos y bloqueos de rango",
        "action-unblockself": "desbloquearse a sí mismo",
        "action-noratelimit": "no resultar afectado por los límites de frecuencia de edición",
        "action-reupload-own": "sobrescribir archivos existentes subidos por uno mismo",
        "rcfilters-clear-all-filters": "Borrar todos los filtros",
        "rcfilters-show-new-changes": "Ver cambios nuevos desde $1",
        "rcfilters-search-placeholder": "Filtrar cambios (utiliza el menú o busca el nombre de un filtro)",
+       "rcfilters-search-placeholder-mobile": "Filtros",
        "rcfilters-invalid-filter": "Filtro no válido",
        "rcfilters-empty-filter": "No hay filtros activos. Se muestran todas las contribuciones.",
        "rcfilters-filterlist-title": "Filtros",
        "linkaccounts": "Vincular cuentas",
        "linkaccounts-success-text": "La cuenta fue vinculada.",
        "linkaccounts-submit": "Vincular cuentas",
+       "cannotunlink-no-provider-title": "No hay cuentas enlazadas que desenlazar",
+       "cannotunlink-no-provider": "No hay cuentas enlazadas que puedan desenlazarse.",
        "unlinkaccounts": "Desvincular cuentas",
        "unlinkaccounts-success": "Se ha desvinculado la cuenta.",
        "authenticationdatachange-ignored": "El cambio den los datos de autentificacion no fue realizado. ¿Tal vez, no se configuró un proveedor?",
index 3245986..876b037 100644 (file)
        "systemblockedtext": "MediaWiki tarkvara on sinu kasutajanime või IP-aadressi automaatselt blokeerinud.\nToodud on järgmine põhjus:\n\n:<em>$2</em>\n\n* Blokeerimisaeg: $8\n* Blokeeringu aegumistähtaeg: $6\n* Sooviti blokeerida: $7\n\nSinu praegune IP-aadress on $3.\nLisa need andmed kõigile järelepärimistele, mida kavatsed teha.",
        "blockednoreason": "põhjendust ei ole kirja pandud",
        "blockedtext-composite": "<strong>Sinu kasutajanimi või IP-aadress on blokeeritud.</strong>\n\nÄra on toodud järgmine põhjus:\n\n:<em>$2</em>.\n\n* Blokeeringu algus: $8\n* Pikima blokeeringu aegumistähtaeg: $6\n\n* $5\n\nSinu praegune IP-aadress on $3.\nPalun lisa need andmed kõigile järelepärimistele, mida kavatsed teha.",
+       "blockedtext-composite-ids": "Asjakohaste blokeeringute identifikaatorid: $1 (sinu IP-aadress võib olla samuti mustas nimekirjas)",
+       "blockedtext-composite-no-ids": "Sinu IP-aadress on mitmes mustas nimekirjas",
        "blockedtext-composite-reason": "Sinu IP-aadressi ja/või kasutajanime suhtes on kehtestatud mitu blokeeringut",
        "whitelistedittext": "Lehekülgede toimetamiseks pead $1.",
        "confirmedittext": "Lehekülgi ei saa toimetada enne e-posti aadressi kinnitamist.\nPalun määra ja kinnita e-posti aadress [[Special:Preferences|eelistuste leheküljel]].",
        "search-interwiki-more": "(veel)",
        "search-interwiki-more-results": "veel tulemusi",
        "search-relatedarticle": "Seotud",
+       "search-invalid-sort-order": "Järjestus \"$1\" on tundmatu, kasutatakse vaikejärjestust. Sobivad järjestused: $2",
+       "search-unknown-profile": "Otsingu profiil $1 on tundmatu, kasutatakse otsingu vaikeprofiili.",
        "searchrelated": "seotud",
        "searchall": "kõik",
        "showingresults": "Allpool näidatakse '''{{PLURAL:$1|ühte|$1}}''' tulemust alates '''$2'''. tulemusest.",
        "right-editmyusercss": "Redigeerida enda CSS-faile",
        "right-editmyuserjson": "Redigeerida enda JSON-faile",
        "right-editmyuserjs": "Redigeerida enda JS-faile",
+       "right-editmyuserjsredirect": "Redigeerida enda JS-faile, mis on ümbersuunamised",
        "right-viewmywatchlist": "Vaadata oma jälgimisloendit",
        "right-editmywatchlist": "Redigeerida oma jälgimisloendit. Pane tähele, et mõne toiminguga lisatakse lehekülgi siiski ka ilma selle õiguseta.",
        "right-viewmyprivateinfo": "Vaadata oma eraandmeid (nt e-posti aadress, pärisnimi)",
        "action-editmyusercss": "redigeerida enda CSS-faile",
        "action-editmyuserjson": "redigeerida enda JSON-faile",
        "action-editmyuserjs": "redigeerida enda JS-faile",
+       "action-editmyuserjsredirect": "redigeerida enda JS-faile, mis on ümbersuunamised",
        "action-viewsuppressed": "vaadata kõigi kasutajate eest varjatud redaktsioone",
        "action-hideuser": "blokeerida kasutajanime, peites selle avalikkuse eest",
        "action-ipblock-exempt": "mööduda automaatsetest blokeeringutest ning aadressivahemiku- ja IP-blokeeringutest",
        "rcfilters-clear-all-filters": "Eemalda kõik filtrid",
        "rcfilters-show-new-changes": "Vaata uusi muudatusi alates: $1",
        "rcfilters-search-placeholder": "Filtreeri muudatusi (kasuta menüüd või tipi filtri nimi)",
+       "rcfilters-search-placeholder-mobile": "Filtrid",
        "rcfilters-invalid-filter": "Vigane filter",
        "rcfilters-empty-filter": "Aktiivsed filtrid puuduvad. Näidatakse kogu kaastööd.",
        "rcfilters-filterlist-title": "Filtrid",
        "linkaccounts": "Kontode linkimine",
        "linkaccounts-success-text": "Konto on lingitud.",
        "linkaccounts-submit": "Lingi kontod",
-       "unlinkaccounts": "Tühista kontode linkimine",
+       "cannotunlink-no-provider-title": "Puuduvad lingitud kontod, mille linkimine tühistada",
+       "cannotunlink-no-provider": "Puuduvad lingitud kontod, mille linkimine tühistada.",
+       "unlinkaccounts": "Kontode linkimise tühistamine",
        "unlinkaccounts-success": "Kontode linkimine on tühistatud.",
        "authenticationdatachange-ignored": "Autentimisandmete muutmine jäi rahuldamata. Võimalik, et ühtegi pakkujat polnud häälestatud.",
        "userjsispublic": "Pea silmas, et JavaScripti alamleheküljed ei tohiks sisaldada konfidentsiaalseid andmeid, kuna neid näevad teised kasutajad.",
        "restrictionsfield-help": "Üks IP-aadress või CIDR-vahemik rea kohta. Et lubada kõik, kasuta järgmist süntaksit:<pre>0.0.0.0/0\n::/0</pre>",
        "edit-error-short": "Tõrge: $1",
        "edit-error-long": "Tõrked:\n\n$1",
+       "specialmute": "Vaigistamine",
+       "specialmute-success": "Sinu vaigistamiseelistused on uuendatud. Vaata kõiki vaigistatud kasutajaid [[Special:Preferences|eelistuste leheküljel]].",
+       "specialmute-submit": "Kinnita",
+       "specialmute-label-mute-email": "Keela e-kirjad sellelt kasutajalt",
+       "specialmute-header": "Palun vali vaigistamiseelistused kasutaja <b>{{BIDI:[[User:$1|$1]]}}</b> suhtes.",
+       "specialmute-error-invalid-user": "Päritud kasutajanime ei leitud.",
+       "specialmute-error-no-options": "Vaigistamisfunktsioonid pole saadaval. Põhjused võivad olla järgmised: sa pole oma e-posti aadressi kinnitanud või viki administraator on keelanud selles vikis e-kirjade saatmise võimaluse ja/või e-posti aadresside musta nimekirja.",
+       "specialmute-email-footer": "Et hallata kasutaja {{BIDI:$2}} suhtes kehtivaid e-posti eelistusi, mine aadressile <$1>.",
+       "specialmute-login-required": "Palun logi sisse, et muuta vaigistamiseelistusi.",
+       "mute-preferences": "Vaigistamiseelistused",
        "revid": "redaktsioon $1",
        "pageid": "lehekülje identifikaator $1",
        "interfaceadmin-info": "$1\n\nSaidiüleste CSS/JS/JSON-failide redigeerimisõigused on nüüd eraldi õigusest <code>editinterface</code>. Kui sa ei saa aru, mis seda tõrketeadet näed, siis loe lehekülge [[mw:MediaWiki_1.32/interface-admin]].",
        "passwordpolicies-policy-passwordnotinlargeblacklist": "Parool ei saa olla 100&nbsp;000 kõige levinuma parooli loendis.",
        "passwordpolicies-policyflag-forcechange": "peab muutma sisselogimisel",
        "passwordpolicies-policyflag-suggestchangeonlogin": "soovita muutmist sisselogimisel",
+       "mycustomjsredirectprotected": "Sul pole õigust seda JavaScripti lehekülge redigeerida, sest see on ümbersuunamine ja see ei viita sinu kasutajaruumi.",
        "easydeflate-invaliddeflate": "Ette antud sisu ei ole õigesti vähendatud",
        "unprotected-js": "Turvalisuse huvides ei saa JavaScripti laadida kaitsmata lehekülgedelt. Palun koosta JavaScripti ainult nimeruumis MediaWiki või kasutajate nimeruumi alamleheküljel.",
        "userlogout-continue": "Kas soovid välja logida?"
index 9110c93..36303ed 100644 (file)
        "svg-long-error": "SVG fitxategi ez baliagarria: $1",
        "show-big-image": "Jatorrizko fitxategia",
        "show-big-image-preview": "Aurreikuspen honen neurria: $1.",
-       "show-big-image-preview-differ": "$2 fitxategi honen $3 aurreikuspenaren tamainia: $1.",
+       "show-big-image-preview-differ": "$2 fitxategi honen $3 aurreikuspenaren tamaina: $1.",
        "show-big-image-other": "Bestelako {{PLURAL:$2|bereizmena|bereizmenak}}: $1.",
        "show-big-image-size": "$1 × $2 pixel",
        "file-info-gif-looped": "kiribildua",
        "mediastatistics": "Media estatistikak",
        "mediastatistics-summary": "Igotako fitxategien estatistikak. Hemen ikus daitekeena fitxategiaren azken bertsioa baino ez da. Fitxategiaren bertsio zahar edo ezabatuak kanpo daude.",
        "mediastatistics-nbytes": "{{PLURAL:$1|$1 byte|$1 byte}} ($2; %$3)",
-       "mediastatistics-bytespertype": "Atal honetarako fitxategi tamainia totala: {{PLURAL:$1|$1 byte}} ($2; $3%).",
-       "mediastatistics-allbytes": "Fitxategi guztietarako fitxategi tamainia osoa: {{PLURAL:$1|$1 byte|$1 bytes}} ($2).",
+       "mediastatistics-bytespertype": "Atal honetako fitxategi tamaina, guztira: {{PLURAL:$1|$1 byte}} ($2; % $3).",
+       "mediastatistics-allbytes": "Fitxategi guztien tamaina, guztira: {{PLURAL:$1|$1 byte}} ($2).",
        "mediastatistics-table-mimetype": "MIME mota",
        "mediastatistics-table-extensions": "Luzapen posibleak",
        "mediastatistics-table-count": "Fitxategi kopurua",
index b7ad22b..715434e 100644 (file)
@@ -8,7 +8,8 @@
                        "SPQRobin",
                        "Siebrand",
                        "Sjoerddebruin",
-                       "Slomox"
+                       "Slomox",
+                       "Romaine"
                ]
        },
        "exif-imagewidth": "Breedte",
        "exif-photometricinterpretation-3": "Palet",
        "exif-photometricinterpretation-4": "Transparantiemasker",
        "exif-photometricinterpretation-5": "Gescheiden (waarschijnlijk CMYK)",
+       "exif-photometricinterpretation-8": "CIE L*a*b*",
        "exif-unknowndate": "Datum onbekend",
        "exif-orientation-1": "Normaal",
        "exif-orientation-2": "Horizontaal gespiegeld",
index 887accd..46e9750 100644 (file)
        "exif-copyright": "Infuimmazioni i' lu dirittu d'autori",
        "exif-exifversion": "Versioni di lu fuimmaddu Exif",
        "exif-colorspace": "Ippàziu di li curori",
+       "exif-pixelxdimension": "Largura di l'immàgina",
+       "exif-pixelydimension": "Althura di l'immàgina",
        "exif-usercomment": "Noti di l'utenti",
        "exif-exposuretime-format": "$1 sigundu ($2)",
        "exif-flash": "Caratterìsthiga e cundizioni di lu lampu",
        "exif-flashenergy": "Putènzia di lu lampu",
+       "exif-filesource": "Fonti di l'ischedàriu",
        "exif-contrast": "Cuntrollu cuntrasthu",
+       "exif-gpslatitude": "Latitùdini",
+       "exif-gpslongitude": "Longitùdini",
+       "exif-gpsaltitude": "Althària",
+       "exif-source": "Orìgini",
        "exif-languagecode": "Linga",
        "exif-iimcategory": "Categuria",
        "exif-orientation-1": "Noimmari",
index 17bf742..2cc372a 100644 (file)
        "exif-photometricinterpretation": "Төс моделе",
        "exif-orientation": "Кадр куелышы",
        "exif-samplesperpixel": "Төс өлешләре саны",
-       "exif-xresolution": "Ð\93оÑ\80изонÑ\82алÑ\8c Ð·Ñ\83Ñ\80лык",
-       "exif-yresolution": "Ð\92еÑ\80Ñ\82икалÑ\8c Ð·Ñ\83Ñ\80лык",
-       "exif-datetime": "Файл үзгәртүләр датасы һәм вакыты",
-       "exif-imagedescription": "РÓ\99Ñ\81емнең Ð¸Ñ\81еме",
-       "exif-make": "Камераның җитештерүчесе",
-       "exif-model": "Камераның төре",
-       "exif-software": "Ð\9fÑ\80огÑ\80аммалÑ\8b Ñ\82Ó\99Ñ\8dмин Ð¸Ñ\82елеÑ\88",
+       "exif-xresolution": "ЯÑ\82ма Ð°Ñ\87Ñ\8bклык",
+       "exif-yresolution": "Ð\90Ñ\81ма Ð°Ñ\87Ñ\8bклык",
+       "exif-datetime": "Файл үзгәреше датасы һәм вакыты",
+       "exif-imagedescription": "СÑ\83Ñ\80Ó\99Ñ\82 Ð°Ñ\82амаÑ\81Ñ\8b",
+       "exif-make": "Камера җитештерүчесе",
+       "exif-model": "Камера төре",
+       "exif-software": "Ð\9aÑ\83лланÑ\8bлган Ð¿Ñ\80огÑ\80амма",
        "exif-artist": "Автор",
-       "exif-copyright": "Автор хокуклары иясе",
+       "exif-copyright": "Авторлык хокукы иясе",
        "exif-exifversion": "Exif юрамасы",
-       "exif-flashpixversion": "FlashPix Ñ\8eÑ\80амаÑ\81Ñ\8bн Ñ\82Ó\99Ñ\8dмин Ð¸Ñ\82Ò¯",
-       "exif-colorspace": "Төсләр тирәлеге",
+       "exif-flashpixversion": "FlashPix Ñ\8fÑ\80аÑ\88лÑ\8b Ñ\8eÑ\80амаÑ\81Ñ\8b",
+       "exif-colorspace": "Төсләр киңлеге",
        "exif-componentsconfiguration": "Төсләр төзелешенең конфигурациясе",
        "exif-compressedbitsperpixel": "Кысылудан соң төснең тирәнлеге",
        "exif-pixelxdimension": "Сурәт киңлеге",
@@ -37,7 +37,7 @@
        "exif-subsectime": "Файлны үзгәртүнең өлешле секунд вакыты",
        "exif-subsectimeoriginal": "Чын ясалу вакытының өлеш секунды",
        "exif-subsectimedigitized": "Санлаштыру вакытының өлеш секунды",
-       "exif-exposuretime": "ЭкÑ\81позиÑ\86иÑ\8f Ð²Ð°ÐºÑ\8bÑ\82ы",
+       "exif-exposuretime": "ЭкÑ\81позиÑ\86иÑ\8f Ð¾Ð·Ð°ÐºÐ»Ñ\8bгы",
        "exif-exposuretime-format": "$1 с ($2)",
        "exif-fnumber": "Диафрагманың саны",
        "exif-fnumber-format": "f/$1",
        "exif-sharpness": "Ачыклыгы",
        "exif-devicesettingdescription": "Камераның көйләүләр тасвирламасы",
        "exif-subjectdistancerange": "Төшерү җисеменә кадәр ераклык",
-       "exif-imageuniqueid": "РÓ\99Ñ\81емнең саны (ID)",
+       "exif-imageuniqueid": "СÑ\83Ñ\80Ó\99Ñ\82 саны (ID)",
        "exif-gpsversionid": "GPS мәгълүматы блогының версиясе",
        "exif-gpslatituderef": "Киңлек индексы",
        "exif-gpslatitude": "Киңлек",
        "exif-gpslongituderef": "Озынлык индексы",
        "exif-gpslongitude": "Озынлык",
-       "exif-gpsaltituderef": "Ð\91иеклек индексы",
-       "exif-gpsaltitude": "Ð\91иеклек",
+       "exif-gpsaltituderef": "ЮгаÑ\80Ñ\8bлÑ\8bк индексы",
+       "exif-gpsaltitude": "ЮгаÑ\80Ñ\8bлÑ\8bк",
        "exif-gpstimestamp": "UTC буенча вакыт",
        "exif-gpssatellites": "Кулланылган иярченнәр тасвирламасы",
        "exif-gpsstatus": "Алгычның статусы һәм төшерү вакыты",
        "exif-languagecode": "Тел",
        "exif-iimversion": "IIM юрамасы",
        "exif-iimcategory": "Төркем",
+       "exif-iimsupplementalcategory": "Өстәмә төркемнәр",
        "exif-identifier": "Идентификатор",
        "exif-label": "Билгеләү",
-       "exif-copyrighted": "Автор хокуклары халәте:",
-       "exif-copyrightowner": "Автор хокуклары иясе",
+       "exif-copyrighted": "Авторлык хокукы халәте",
+       "exif-copyrightowner": "Авторлык хокукы иясе",
        "exif-usageterms": "Куллану шартлары",
        "exif-orientation-1": "Гадәти",
        "exif-orientation-3": "180° ка борылган",
-       "exif-componentsconfiguration-0": "юк",
+       "exif-componentsconfiguration-0": "барлыкта юк",
        "exif-exposureprogram-0": "Билгесез",
        "exif-exposureprogram-1": "Кулдан җайлау режимы",
        "exif-exposureprogram-2": "Программалы режим (гади)",
        "exif-meteringmode-6": "Өлешләтә",
        "exif-meteringmode-255": "Башка",
        "exif-lightsource-0": "Билгесез",
+       "exif-lightsource-1": "Көндезге яктылык",
        "exif-lightsource-4": "Яктылык",
-       "exif-lightsource-9": "Яхшы һава торышы",
+       "exif-lightsource-9": "Аяз",
+       "exif-lightsource-10": "Болытлы",
        "exif-lightsource-11": "Күләгә",
        "exif-flash-mode-3": "автоматик режим",
        "exif-focalplaneresolutionunit-2": "дюйм",
        "exif-gpsstatus-a": "Үлчәү тәмамланмаган",
        "exif-gpsstatus-v": "Мәгълүматларны җибәрүгә әзер",
        "exif-gpsspeed-k": "км/сәг",
-       "exif-gpsspeed-m": "милÑ\8f/сәг",
-       "exif-gpsspeed-n": "Төен",
+       "exif-gpsspeed-m": "милÑ\8c/сәг",
+       "exif-gpsspeed-n": "узел",
        "exif-gpsdestdistance-k": "Километр",
        "exif-gpsdestdistance-m": "Миль",
        "exif-gpsdestdistance-n": "Диңгез миле",
        "exif-gpsdop-fair": "Ярыйсы ($1)",
        "exif-gpsdop-poor": "Начар ($1)",
        "exif-dc-date": "Дата(лар)",
-       "exif-dc-publisher": "Нәшрият",
+       "exif-dc-publisher": "Нәшир",
        "exif-dc-relation": "Бәйле медиа",
        "exif-dc-rights": "Хокуклар",
        "exif-dc-source": "Чыганак медиа",
        "exif-dc-type": "Медиа төре",
        "exif-rating-rejected": "Кире кагылды",
-       "exif-isospeedratings-overflow": "65535-тән күп",
+       "exif-isospeedratings-overflow": "65535 тән күбрәк",
        "exif-iimcategory-hth": "Сәламәтлек",
        "exif-iimcategory-lab": "Хезмәт",
-       "exif-iimcategory-wea": "Һава тырышы",
+       "exif-iimcategory-wea": "Һава торышы",
        "exif-urgency-normal": "Гадәти ($1)",
        "exif-urgency-low": "Түбән ($1)",
        "exif-urgency-high": "Югары ($1)"
index 755b38f..84e08b8 100644 (file)
        "tags-edit-reason": "Syy:",
        "tags-edit-revision-submit": "Toteuta muutokset {{PLURAL:$1|tähän versioon|$1 versioon}}",
        "tags-edit-logentry-submit": "Lähetä muutoksesi {{PLURAL:$1|tähän lokimerkintään|$1 lokimerkintään}}",
-       "tags-edit-success": "Muutokset on tehty.",
+       "tags-edit-success": "Muutokset toteutettiin.",
        "tags-edit-failure": "Muutoksia ei voitu toteuttaa: $1",
        "tags-edit-nooldid-title": "Kohdeversio ei ole kelvollinen",
        "tags-edit-nooldid-text": "Et ole joko määrittänyt sitä kohdeversiota, johon tämä toimenpide kohdistuu, tai sitten määrättyä versiota ei ole olemassa.",
index 6f96360..f1c9830 100644 (file)
        "rcfilters-clear-all-filters": "Effacer tous les filtres",
        "rcfilters-show-new-changes": "Afficher les nouvelles modifications depuis $1",
        "rcfilters-search-placeholder": "Filtrer les modifications (utiliser le menu ou rechercher le nom d'un filtre)",
+       "rcfilters-search-placeholder-mobile": "Filtres",
        "rcfilters-invalid-filter": "Filtre non valide",
        "rcfilters-empty-filter": "Aucun filtre actif. Toutes les contributions sont affichées.",
        "rcfilters-filterlist-title": "Filtres",
        "block-log-flags-angry-autoblock": "autoblocage amélioré activé",
        "block-log-flags-hiddenname": "nom d’utilisateur masqué",
        "range_block_disabled": "Le droit administrateur de créer des blocages de plages IP est désactivé.",
+       "ipb-prevent-user-talk-edit": "Modifier sa propre page de discussion peut être autorisé pour un blocage partiel, sauf s’il inclut une restriction sur l’espace de noms Discussion utilisateur.",
        "ipb_expiry_invalid": "Durée d'expiration incorrecte.",
        "ipb_expiry_old": "L’heure d’expiration est passée.",
        "ipb_expiry_temp": "Les blocages de noms d'utilisateurs cachés doivent être permanents.",
        "move-page-legend": "Renommer une page",
        "movepagetext": "Utilisez le formulaire ci-dessous pour renommer une page, en déplaçant tout son historique vers le nouveau nom. L’ancien titre deviendra une page de redirection vers le nouveau titre. \nVous pouvez mettre à jour automatiquement les redirections qui pointent vers le titre original. \nSi vous choisissez de ne pas le faire, assurez-vous de vérifier toute [[Special:DoubleRedirects|double redirection]] ou [[Special:BrokenRedirects|redirection cassée]]. Vous avez la responsabilité de vous assurer que les liens continuent de pointer vers leur destination supposée.\n\nNotez que la page ne sera <strong>pas</strong> renommée s’il existe déjà une page portant le nouveau titre, sauf si cette dernière est une simple redirection avec un historique de modifications vierge. \nCela signifie que vous pouvez de nouveau renommer une page vers sa position d’origine si vous avez fait une erreur et que vous ne pouvez pas écraser une page existante.\n\n<strong>Attention !</strong>\nCeci peut provoquer un changement radical et imprévu pour une page souvent consultée ; assurez-vous d’avoir compris les conséquences de votre démarche avant de continuer.",
        "movepagetext-noredirectfixer": "Utilisez le formulaire ci-dessous pour renommer une page, en déplaçant tout son historique vers le nouveau nom.\nL’ancien titre deviendra une page de redirection vers le nouveau titre.\nVérifiez bien les [[Special:DoubleRedirects|doubles redirections]] ou les [[Special:BrokenRedirects|redirections cassées]].\nVous avez la responsabilité de vous assurer que les liens continuent de pointer vers leur destination supposée.\n\nNotez que la page ne sera <strong>pas</strong> déplacée s’il existe déjà une page avec le nouveau titre, sauf si cette dernière a un historique de modifications vierge et est soit vide, soit une simple redirection. Ceci permet de renommer une page vers sa position d’origine si le déplacement s’avère erroné, et il est impossible d’écraser une page existante.\n\n<strong>Attention !</strong>\nCeci peut provoquer un changement radical et imprévu pour une page souvent consultée ; assurez-vous d’en avoir compris les conséquences avant de continuer.",
+       "movepagetext-noredirectsupport": "Utilisez le formulaire ci-desous pour renommer une page, et déplaçer toute son historique sous le nouveau nom.\nIl vous appartient de vérifier que tous les liens continuent à pointer vers leurs endroits respectifs.\n\nNotez bien que la page <strong>ne sera pas</strong> renommée s'il existe déjà une page portant le même titre que le nouveau tire choisi.\nCela signifie que vous pouvez revenir en arrière en renommant la page sous son ancien nom si vous avez faites une erreur, et que vous ne pouvez pas écraser une page existante.\n\n<strong>Note:</strong>\nCeci peut être une modification drastique et inattendue si la page est populaire;\nveuillez vous assurer de comprendre les conséquences de cela avant de continuer.",
        "movepagetalktext": "Si vous cochez cette case, la page de discussion associée sera automatiquement renommée, à moins qu’une page de discussion non vide existe déjà sous ce nouveau nom.\n\nDans ce cas, vous devrez renommer ou fusionner cette page de discussion manuellement si vous le désirez.",
        "moveuserpage-warning": "<strong>Attention :</strong> Vous êtes sur le point de renommer une page d’utilisateur. Veuillez noter que seule la page sera renommée et que l’utilisateur <em>ne</em> sera <em>pas</em> renommé.",
        "movecategorypage-warning": "<strong>Avertissement :</strong> Vous êtes sur le point de renommer une page de catégorie. Veuillez noter que seule la catégorie sera renommée et <em>qu’aucune</em> des pages de l’ancienne catégorie ne sera transférée dans la nouvelle.",
        "linkaccounts": "Lier les comptes",
        "linkaccounts-success-text": "Le compte a été lié.",
        "linkaccounts-submit": "Lier les comptes",
+       "cannotunlink-no-provider-title": "Il n’y a pas de compte lié à délier",
+       "cannotunlink-no-provider": "Il n’y a pas de compte lié qui puisse être délié.",
        "unlinkaccounts": "Dissocier les comptes",
        "unlinkaccounts-success": "Le compte a été dissocié.",
        "authenticationdatachange-ignored": "Les modifications de données d’authentification n’ont pas été gérées. Peut-être aucun fournisseur n’a-t-il été configuré ?",
index 64f112a..4378161 100644 (file)
        "pool-queuefull": "Piha ke kiʻo kiū",
        "pool-errorunknown": "Hewa ʻikeʻole",
        "aboutsite": "No {{SITENAME}}",
-       "aboutpage": "Project:No translatewiki.net",
        "copyright": "Aia nā mealoko ma lalo o ka laikini $1 inā noka ʻole ia.",
        "copyrightpage": "{{ns:project}}:Kūleana kope",
        "currentevents": "Nūhou",
index 79f0e30..6e8339a 100644 (file)
        "search-interwiki-more": "(עוד)",
        "search-interwiki-more-results": "תוצאות נוספות",
        "search-relatedarticle": "קשור",
+       "search-invalid-sort-order": "סדר המיון של $1 אינו מוכר, יחול הסדר שמוגדר לפי ברירת המחדל. סדרי המיון התקינים הם: $2",
+       "search-unknown-profile": "פרופיל חיפוש של $1 אינו מוכר, יחול הסדר שמוגדר לפי ברירת המחדל.",
        "searchrelated": "קשור",
        "searchall": "הכול",
        "showingresults": "{{PLURAL:$1|מוצגת תוצאה <strong>אחת</strong>|מוצגות עד <strong>$1</strong> תוצאות}} החל ממספר <strong>$2</strong>:",
        "rcfilters-clear-all-filters": "מחיקת כל המסננים",
        "rcfilters-show-new-changes": "הצגת השינויים החדשים מאז $1",
        "rcfilters-search-placeholder": "סינון שינויים (ניתן להשתמש בתפריט או לחפש שם מסנן)",
+       "rcfilters-search-placeholder-mobile": "מסננים",
        "rcfilters-invalid-filter": "מסנן בלתי־תקין",
        "rcfilters-empty-filter": "אין מסננים פעילים. כל התרומות מוצגות.",
        "rcfilters-filterlist-title": "מסננים",
        "block-log-flags-angry-autoblock": "חסימה אוטומטית מתקדמת מופעלת",
        "block-log-flags-hiddenname": "שם המשתמש הוסתר",
        "range_block_disabled": "האפשרות לחסום טווח כתובות אינה פעילה.",
+       "ipb-prevent-user-talk-edit": "עריכת דף המשתמש צריכה להיות מותרת בחסימה חלקית, אלא אם כן היא כוללת הגבלה במרחב שיחת משתמש.",
        "ipb_expiry_invalid": "זמן פקיעת החסימה אינו תקין.",
        "ipb_expiry_old": "זמן הפקיעה כבר עבר.",
        "ipb_expiry_temp": "חסימות הכוללות הסתרת שם משתמש חייבות להיות לזמן בלתי מוגבל.",
        "move-page-legend": "העברת דף",
        "movepagetext": "ניתן להשתמש בטופס שלהלן כדי לשנות את השם של הדף הזה ולהעביר את כל היסטוריית העריכות שלו לשם החדש.\nהשם הישן יהפוך לדף הפניה אל השם החדש.\nבאפשרותך לעדכן באופן אוטומטי דפי הפניה שכרגע מפנים לשם הנוכחי של הדף.\nנא לוודא לאחר ההעברה שאין [[Special:DoubleRedirects|הפניות כפולות]] או [[Special:BrokenRedirects|הפניות שבורות]] (אלא אם כן בחרת לבצע את העדכון האוטומטי הנ\"ל).\nכמו כן, באחריותך לוודא שכל הקישורים ימשיכו לקשר למקומות שאליהם הם אמורים לקשר.\n\nיש לשים לב לכך שהדף <strong>לא</strong> יועבר אם כבר יש דף תחת השם החדש, אלא אם כן הדף עם השם החדש הוא הפניה ואין לו עריכות קודמות.\nזה אומר שניתן יהיה להחזיר את הדף לשם המקורי במקרה שתיעשה טעות, אבל לא ניתן \"לדרוס\" דף קיים.\n\n<strong>לתשומת לבך:</strong>\nהעברה זו עלולה להיות שינוי דרסטי ומהותי לדף פופולרי;\nיש לקחת בחשבון את התוצאות של הפעולה הזאת לפני ביצוע ההעברה.",
        "movepagetext-noredirectfixer": "ניתן להשתמש בטופס שלהלן כדי לשנות את השם של הדף הזה ולהעביר את כל היסטוריית העריכות שלו לשם החדש.\nהשם הישן יהפוך לדף הפניה אל השם החדש.\nנא לוודא לאחר ההעברה שאין [[Special:DoubleRedirects|הפניות כפולות]] או [[Special:BrokenRedirects|הפניות שבורות]].\nכמו כן, באחריותך לוודא שכל הקישורים ימשיכו לקשר למקומות שאליהם הם אמורים לקשר.\n\nיש לשים לב לכך שהדף <strong>לא</strong> יועבר אם כבר יש דף תחת השם החדש, אלא אם כן הדף עם השם החדש הוא הפניה ואין לו עריכות קודמות.\nזה אומר שניתן יהיה להחזיר את הדף לשם המקורי במקרה שתיעשה טעות, אבל לא ניתן \"לדרוס\" דף קיים.\n\n<strong>לתשומת לבך:</strong>\nהעברה זו עלולה להיות שינוי דרסטי ומהותי לדף פופולרי;\nיש לקחת בחשבון את התוצאות של הפעולה הזאת לפני ביצוע ההעברה.",
+       "movepagetext-noredirectsupport": "שימוש בטופס להלן ישנה את השם של הדף ויעביר את כל היסטוריית הגרסאות שלו לשם החדש.\nבאחריותך להבטיח שכל הקישורים אליו ימשיכו להצביע למקום שהם אמורים להגיע אליו.\n\nיש לשים לב לכך שהדף <strong>לא</strong> יועבר אם כבר יש דף בכותרת החדשה.\nזה אומר שבאפשרותך לשנות שם של דף חזרה לשם שממנו הוא הועבר במקרה של טעות, ושאי־אפשר לדרוס דף קיים.\n\n<strong>לתשומת לבך:</strong>\nזה יכול להיות שינוי קיצוני ובלתי־צפוי עבור דף פופולרי;\nנא לוודא שהבנת את ההשלכות של זה לפני המשך הפעולה.",
        "movepagetalktext": "אם האפשרות הזאת מסומנת, דף השיחה של הדף הזה יועבר אוטומטית לשם החדש, אלא אם קיים דף שיחה שאינו ריק תחת השם החדש. במקרה כזה, יש להעביר או למזג את הדפים באופן ידני, במידת הצורך.",
        "moveuserpage-warning": "<strong>אזהרה:</strong> הדף שיועבר הוא דף משתמש. חשוב לציין שרק הדף יועבר וששם המשתמש <em>לא</em> ישתנה.",
        "movecategorypage-warning": "<strong>אזהרה:</strong> הדף שיועבר הוא דף קטגוריה. חשוב לציין שרק הדף יועבר ושכל הדפים בקטגוריה הישנה <em>לא</em> יסווגו לקטגוריה החדשה.",
        "linkaccounts": "קישור חשבונות",
        "linkaccounts-success-text": "החשבון קושר.",
        "linkaccounts-submit": "קישור החשבונות",
+       "cannotunlink-no-provider-title": "אין חשבונות מקושרים שאפשר לבטל את הקישור שלהם",
+       "cannotunlink-no-provider": "אין חשבונות מקושרים שהקישור שלהם יכול להיות מבוטל.",
        "unlinkaccounts": "ביטול הקישור בין חשבונות",
        "unlinkaccounts-success": "קישור החשבון בוטל.",
        "authenticationdatachange-ignored": "השינוי בנתוני האימות לא הצליח. ייתכן שלא הוגדר ספק.",
index eaf2f0c..fdbd336 100644 (file)
        "rcfilters-clear-all-filters": "Összes szűrő kikapcsolása",
        "rcfilters-show-new-changes": "$1 óta történt friss változtatások megtekintése",
        "rcfilters-search-placeholder": "Változtatások szűrése (használd a menüt vagy keress szűrőkre)",
+       "rcfilters-search-placeholder-mobile": "Szűrők",
        "rcfilters-invalid-filter": "Érvénytelen szűrő",
        "rcfilters-empty-filter": "Nincs aktív szűrő. Minden közreműködés látható.",
        "rcfilters-filterlist-title": "Szűrők",
index 0410ad8..a8cfdb8 100644 (file)
        "search-result-category-size": "{{PLURAL:$1|անդամ}} ({{PLURAL:$2|ենթակատեգորիա}}, {{PLURAL:$3|նիշք}})",
        "search-redirect": "(վերահղում $1 էջից)",
        "search-section": "(բաժին $1)",
-       "search-category": "(կատեգորիա $1)",
+       "search-category": "(կատեգորիա  $1)",
        "search-file-match": "(համապատասխանում է նիշքի բովանդակությանը)",
        "search-suggest": "Գուցե նկատի ունե՞ք՝ $1",
        "search-interwiki-caption": "Կից նախագծեր",
index f3254dd..3e98fb7 100644 (file)
        "december-date": "$1 Դեկտեմբեր",
        "period-am": "Նախ Կէսօր",
        "period-pm": "Կէսօրէն Յետոյ",
-       "pagecategories": "{{PLURAL:$1|Կատեգորիա|Կատեգորիաներ}}",
+       "pagecategories": "{{PLURAL:$1|Ստորոգութիւն|Ստորոգութիւններ}}",
        "category_header": "«$1» ստորոգութեան մէջ էջեր",
        "subcategories": "Ենթակատեգորիաներ",
        "category-media-header": "\"$1\" ստորոգութեան հաղորդամիջոց",
        "category-empty": "<em>Այս ստորոգութիւնը ներկայիս դատարկ է։<em>",
-       "hidden-categories": "{{PLURAL:$1|Թաքցուած կատեգորիա|Թաքցուած կատեգորիաներ}}",
+       "hidden-categories": "{{PLURAL:$1|Թաքուն ստորոգութիւն|Թաքուն ստորոգութիւններ}}",
        "hidden-category-category": "Թաքցուած կատեգորիաներ",
        "category-subcat-count": "{{PLURAL:$2|Այս կատեգորիան ունի միայն հետեւեալ ենթակատեգորիան։|Այս կատեգորիան ունի հետեւեալ {{PLURAL:$1|ենթակատեգորիա|ենթակատեգորիաներ}}ը՝ ընդհանուր $2էն։}}",
        "category-subcat-count-limited": "Այս ստորոգութիւնը ունի հետեւեալ {{PLURAL:$1|ենթաստորոգութիւն|$1 ենթաստորոգութիւններ}}։",
        "nstab-mediawiki": "Հաղորդագրութիւն",
        "nstab-template": "Կաղապար",
        "nstab-help": "Օգնութեան էջ",
-       "nstab-category": "Կատեգորիա",
+       "nstab-category": "Ստորոգութիւն",
        "mainpage-nstab": "Գլխաւոր Էջ",
        "nosuchaction": "Այս գործողութիւնը չկայ",
        "nosuchspecialpage": "Այդպիսի յատուկ էջ չկայ",
        "retypenew": "Նորէն մուտքագրէք գաղտնաբառը",
        "resetpass_submit": "Հաստատեցէ՛ք անցաբառը եւ մուտք գործեցէ՛ք համակարգ",
        "changepassword-success": "Ձեր անցաբառը փոխուեցաւ։",
+       "botpasswords": "Մեքենայիկների անցաբառեր",
        "botpasswords-label-appid": "Մեքենայիկի անուն՝",
        "botpasswords-label-create": "Ստեղծել",
        "botpasswords-label-update": "Թարմացնել",
        "hiddencategories": "Այս էջը կը պատկանի հետեւեալ {{PLURAL:$1|1 թաքուն ստորոգութեան|$1 թաքուն ստորոգութիւնններուն}}.",
        "permissionserrors": "Արտօնութեան սխալ",
        "permissionserrorstext-withaction": "Արտօնութիւն չունիք $2 հետեւեալ {{PLURAL:$1|պատճառով|պատճառներով}}.",
-       "recreate-moveddeleted-warn": "<strong>Զգուշացում. Նախապէս ջնջուած էջ մը պիտի վերստեղծուի։<strong>\n\nԿը խնդրուի մտածել այս էջի խմբագրման նպատակայարմարութեան մասին։ \nՁեր դիւրութեան համար ներքեւ կը գտնէք այս էջի ջնջումին և տեղափոխումին տեղեկատետրերը։",
+       "recreate-moveddeleted-warn": "<strong>Զգուշացում. Նախապէս ջնջուած էջ մը պիտի վերստեղծուի։<strong>\n\nԿը խնդրուի մտածել այս էջի խմբագրման նպատակայարմարութեան մասին։ Ձեր դիւրութեան համար ներքեւ կը գտնէք այս էջի ջնջումին եւ տեղափոխումին տեղեկատետրերը։",
        "moveddeleted-notice": "Այս էջը ջնջուած է։\nԷջին ջնջումի, պահպանումի եւ փոխադրումի տեղեկատետրը տրամադրելի է ներքեւ որպէս տեղեկութիւն։",
        "edit-conflict": "Խմբագրման ընհարում։",
        "postedit-confirmation-created": "Էջը ստեղծուած է։",
        "rev-showdeleted": "Ցուցադրել",
        "revdelete-show-file-submit": "Այո",
        "revdelete-hide-image": "Թաքցնել նիշքին բովանդակութիւնը",
+       "revdelete-radio-set": "Թաքուն",
        "revdelete-radio-unset": "Տեսանելի",
        "revdelete-log": "Պատճառ.",
        "pagehist": "Էջի պատմութիւն",
        "search-result-category-size": "{{PLURAL:$1|1 անդամ|$1 անդամներ}} ({{PLURAL:$2|1 ենթաստորոգութիւն|$2 ենթաստորոգութիւններ}}, {{PLURAL:$3|1 նիշք|$3 նիշքեր}})",
        "search-redirect": "(Վերայղուած է $1-էն)",
        "search-section": "(բաժին $1)",
-       "search-category": "(Õ¯Õ¡Õ¿Õ¥Õ£Õ¸Ö\80Õ«Õ¡ $1)",
+       "search-category": "(Õ½Õ¿Õ¸Ö\80Õ¸Õ£Õ¸Ö\82Õ©Õ«Ö\82Õ¶ $1)",
        "search-file-match": "(համապատասխան է նիշքի բովանդակութեան)",
        "search-suggest": "$1 Նկատի ունի՞ք",
        "search-interwiki-default": "$1 արդիւնք.",
        "preferences": "Նախընտրութիւններ",
        "mypreferences": "Նախընտրութիւններ",
        "skin-preview": "Կանխաստուգել",
+       "prefs-rc": "Վերջին փոփոխութիւնները",
        "prefs-watchlist": "Հսկողութեան ցանկ",
        "prefs-editwatchlist-clear": "Մաքրել հսկողութեան ցանկը",
        "saveprefs": "Յիշել",
        "prefs-signature": "Ստորագրութիւն",
        "prefs-editor": "Խմբագրող",
        "prefs-preview": "Կանխաստուգել",
+       "userrights": "Մասնակիցների իրաւունքների կառավարում",
        "group": "Խումբ.",
        "group-user": "Մասնակիցներ",
        "group-bot": "Մեքենայիկներ",
        "rcfilters-activefilters-hide": "Թաքցնել",
        "rcfilters-activefilters-show": "Ցուցնել",
        "rcfilters-limit-title": "Ցուցադրուող արդիւնքներ",
+       "rcfilters-savedqueries-cancel-label": "Չեղարկել",
+       "rcfilters-search-placeholder-mobile": "Քամոց",
        "rcfilters-filtergroup-authorship": "Ներդրումներու հեղինակ",
        "rcfilters-filter-editsbyself-label": "Կողմէդ կատարուած փոփոխութիւնները",
        "rcfilters-filter-editsbyself-description": "Անձնական ներդրումներդ։",
        "upload-dialog-button-done": "Եղած է",
        "upload-dialog-button-save": "Յիշել",
        "upload-dialog-button-upload": "Վերբեռնել",
+       "upload-form-label-infoform-categories": "Ստորոգութիւններ",
        "license": "Արտօնագրութիւն՝",
        "license-header": "Արտօնագրում",
        "listfiles-delete": "ջնջել",
        "sharedupload-desc-here": "Այս նիշքը առնուած է $1-էն եւ կրնայ օգտագործուիլ այլ նախագիծերու մէջ։ $1-ի մէջ անոր [$2 նիշքը նկարագրող էջի]ի նկարագրութիւնը ներկայացուած է ստորեւ։",
        "filepage-nofile": "Այս անունով նիշք մը գոյութիւն չունի։",
        "upload-disallowed-here": "Այս նիշքը կարելի չէ ջնջել ու փոխարինել։",
+       "listredirects": "Վերայղումների ցանկ",
        "unusedtemplates": "Չօգտագործուող կաղապարներ",
+       "unusedtemplateswlh": "այլ յղումներ",
        "randompage": "Պատահական էջ",
+       "randomincategory-category": "Ստորոգութիւն:",
+       "randomincategory-legend": "Պատահական էջ ստորոգութեան մէջ",
+       "randomredirect": "Պատահական վերայղում",
        "statistics": "Վիճակագրութիւն",
        "statistics-header-pages": "Էջերու վիճակագրութիւն",
        "statistics-header-edits": "Խմբագրումներու վիճակագրութիւն",
        "brokenredirects-edit": "խմբագրել",
        "brokenredirects-delete": "ջնջել",
        "withoutinterwiki": "Լեզուային յղումներ չպարունակող էջեր",
+       "withoutinterwiki-summary": "Հետեւեալ էջեր չունեն լեզւական յղումներ.",
        "withoutinterwiki-submit": "Ցուցնել",
        "fewestrevisions": "Նուազ վերաքաղուած էջեր",
        "nbytes": "$1 {{PLURAL:$1|պայթ}}",
+       "ncategories": "$1 {{PLURAL:$1|ստորոգութիւն|ստորոգութիւններ}}",
+       "nlinks": "$1 {{PLURAL:$1|յղում|յղումներ}}",
        "nmembers": "$1 {{PLURAL:$1|անդամ|անդամներ}}",
        "uncategorizedpages": "Առանց ստորոգութիւններու էջեր",
        "uncategorizedcategories": "Ենթաստորոգութիւն չունեցող ստորոգութիւններ",
        "unusedcategories": "Չօգտագործուող ստորոգութիւններ",
        "unusedimages": "Չօգտագործուող նիշքեր",
        "wantedcategories": "Անհրաժեշտ ստորոգութիւններ",
+       "mostlinked": "Էջեր, որոնց շատ են յղվում",
+       "mostlinkedcategories": "Ստորոգութիւններ, որոնց շատ են յղվում",
+       "mostlinkedtemplates": "Կաղապարներ, որոնց շատ են յղվում",
        "prefixindex": "Բոլոր նախածանցներով էջերը",
        "prefixindex-submit": "Ցուցնել",
        "deadendpages": "Յղումներ չունեցող էջեր",
        "categories": "Ստորոգութիւններ",
        "categories-submit": "Ցուցնել",
        "deletedcontributions": "Մասնակիցի ջնջուած ներդրում",
+       "linksearch": "Արտաքին յղումներ",
        "linksearch-ok": "Որոնել",
        "listusers-submit": "Ցուցնել",
        "activeusers": "Աշխոյժ մասնակիցներու ցանկ",
        "namespace_association": "Առնչուած անուանատարածք",
        "tooltip-namespace_association": "Նշեցէ՛ք տուփիկը՝ ներառնելու համար տուեալ անուանատարածքին հետ կապուած քննարկումները կամ նիւթերու անուանատարածքը նոյնպէս։",
        "blanknamespace": "(Գլխաւոր)",
-       "contributions": "{{GENDER:$1|Մասնակիցի}} ներդրումները",
+       "contributions": "{{GENDER:$1|Մասնակիցի}} ներդրումներ",
        "contributions-title": "$1 մասնակիցի ներդրումը",
        "mycontris": "Ներդրումներ",
        "anoncontribs": "Ներդրումներ",
        "sp-contributions-uploads": "վերբեռնումներ",
        "sp-contributions-logs": "Տեղեկատետրեր",
        "sp-contributions-talk": "քննարկում",
+       "sp-contributions-userrights": "{{GENDER:$1|մասնակից}} իրաւունքների կառավարում",
        "sp-contributions-search": "Որոնել ներդրումները",
        "sp-contributions-username": "IP-հասցէ կամ մասնակիցի անուն.",
        "sp-contributions-toponly": "Ցոյց տալ միայն վերջին տարբերակի խմբագրումները",
        "whatlinkshere-hideredirs": "$1 վերայղումները",
        "whatlinkshere-hidetrans": "$1 ներառումները",
        "whatlinkshere-hidelinks": "$1 յղումները",
-       "whatlinkshere-hideimages": "$1 նիշքի յղումները",
+       "whatlinkshere-hideimages": "$1 նիշքի յղումներ",
        "whatlinkshere-filters": "Զտիչներ",
        "unblock": "Մասնակիցի արգելակումը վերցնել",
        "ipboptions": "2 ժամ:2 hours,1 օր:1 day,3 օր:3 days,1 շաբաթ:1 week,2 շաբաթ:2 weeks,1 ամիս:1 month,3 ամիս:3 months,6 ամիս:6 months,1 տարի:1 year,անժամկէտ:infinite",
        "ipb-pages-label": "Էջեր",
        "ipb-namespaces-label": "Անուանատարածքներ",
+       "blocklist": "Արգելափակուած մասնակիցներ",
        "infiniteblock": "Միշտ",
        "blocklist-editing-page": "էջեր",
        "blocklist-editing-ns": "անուանատարածքներ",
        "reblock-logentry": "փոխեց [[$1]] մասնակիցի արգելակումը՝ աւարտման $2 $3 ժամով",
        "block-log-flags-nocreate": "մասնակցային հաշիւի ստեղծումը արգիլուած է",
        "proxyblocker": "Փոխանորդի արգելակում",
+       "movepage-moved-redirect": "Ստեղծվել է վերայղում։",
        "movelogpage": "Տեղափոխութիւններու տեղեկատետր",
        "export": "Արտածել էջերը",
        "allmessages-language": "Լեզու.",
        "pageinfo-hidden-categories": "Թաքուն {{PLURAL:$1|խմբաւորում|խմբաւորումներ}} ($1)",
        "pageinfo-templates": "Օգտագործուած {{PLURAL:$1|կաղապար|կաղապարներ}} ($1)",
        "pageinfo-toolboxlink": "‎Էջի մասին տեղեկութիւն",
+       "pageinfo-redirectsto": "Վերայղում է դեպի",
        "pageinfo-contentpage": "Իբրեւ բովանդակութեան էջ հաշուըւած",
        "pageinfo-contentpage-yes": "Այո",
        "pageinfo-protect-cascading-yes": "Այո",
        "redirect-revision": "Էջի տարբերակներ",
        "redirect-file": "Նիշքի անունը",
        "specialpages": "Յատուկ էջեր",
+       "specialpages-note-restricted": "* Հասարակ հատուկ էջեր։\n* <span class=\"mw-specialpagerestricted\">Սահմանափակուած հատուկ էջեր։</span>",
        "specialpages-group-maintenance": "Շարունակական տեղեկագրեր",
        "tag-filter": "[[Special:Tags|Պիտակներու]] զտիչ՝",
        "tag-list-wrapper": "[[Special:Tags|{{PLURAL:$1|Պիտակ}}]]: $2",
+       "tag-mw-new-redirect": "Նոր վերայղում",
        "tags-active-yes": "Այո",
        "tags-active-no": "Ոչ",
        "tags-hitcount": "{{PLURAL:$1|փոփոխութիւն}}",
+       "permanentlink": "Մշտական յղում",
        "htmlform-no": "Ոչ",
        "htmlform-yes": "Այո",
        "logentry-delete-delete": "$1 {{GENDER:$2|ջնջեց}} $3 էջը",
index 107c22f..5db5e83 100644 (file)
@@ -65,7 +65,8 @@
                        "Pebaryan",
                        "Veracious",
                        "Mnam23",
-                       "Shirayuki"
+                       "Shirayuki",
+                       "ArlandGa"
                ]
        },
        "tog-underline": "Garis bawahi pranala:",
        "history": "Riwayat halaman",
        "history_short": "Versi terdahulu",
        "history_small": "riwayat",
-       "updatedmarker": "berubah sejak kunjungan terakhir saya",
+       "updatedmarker": "dimutakhirkan sejak kunjungan terakhir saya",
        "printableversion": "Versi cetak",
        "permalink": "Pranala permanen",
        "print": "Cetak",
        "autoblockedtext": "Alamat IP Anda telah terblokir secara otomatis karena digunakan oleh pengguna lain, yang diblokir oleh $1. Pemblokiran dilakukan dengan alasan:\n\n:<em>$2</em>\n\n* Diblokir sejak: $8\n* Blokir kedaluwarsa pada: $6\n* Sasaran pemblokiran: $7\n\nAnda dapat menghubungi $1 atau [[{{MediaWiki:Grouppage-sysop}}|pengurus]] lainnya untuk membicarakan pemblokiran ini.\n\nAnda tidak dapat menggunakan fitur \"{{int:emailuser}}\" kecuali Anda telah memasukkan alamat surel yang sah di [[Special:Preferences|preferensi akun]] Anda dan Anda tidak diblokir untuk menggunakannya.\n\nAlamat IP Anda saat ini adalah $3, dan ID pemblokiran adalah #$5.\nTolong sertakan informasi-informasi ini dalam setiap pertanyaan Anda.",
        "systemblockedtext": "Nama pengguna atau alamat IP Anda telah diblokir secara otomatis oleh MediaWiki.\nAlasan yang diberikan adalah:\n\n:<em>$2</em>\n\n* Diblokir sejak: $8\n* Blokir kedaluwarsa pada: $6\n* Sasaran pemblokiran: $7\n\nAlamat IP Anda saat ini adalah $3\nMohon sertakan semua perincian di atas dalam setiap pertanyaan yang Anda ajukan.",
        "blockednoreason": "tidak ada alasan yang diberikan",
+       "blockedtext-composite-no-ids": "Alamat IP Anda muncul dalam daftar hitam ganda",
        "blockedtext-composite-reason": "Ada pemblokiran berganda terhadap akun Anda dan/atau alamat IP Anda.",
        "whitelistedittext": "Anda harus $1 untuk dapat menyunting halaman.",
        "confirmedittext": "Anda harus mengkonfirmasikan dulu alamat surel Anda sebelum menyunting halaman.\nHarap masukkan dan validasikan alamat surel Anda melalui [[Special:Preferences|halaman preferensi pengguna]] Anda.",
        "mw-widgets-abandonedit-title": "Apakah Anda yakin?",
        "mw-widgets-copytextlayout-copy": "Salin",
        "mw-widgets-copytextlayout-copy-fail": "Gagal menyalin ke papan klip.",
-       "mw-widgets-copytextlayout-copy-success": "Salin ke papan klip.",
+       "mw-widgets-copytextlayout-copy-success": "Disalin ke papan klip.",
        "mw-widgets-dateinput-no-date": "Tanggal tidak ada yang terpilih",
        "mw-widgets-dateinput-placeholder-day": "TTTT-BB-HH",
        "mw-widgets-dateinput-placeholder-month": "TTTT-BB",
        "linkaccounts": "Tautkan akun",
        "linkaccounts-success-text": "Akun telah ditautkan.",
        "linkaccounts-submit": "Tautkan akun",
+       "cannotunlink-no-provider": "Tidak ada akun yang tertaut yang dapat dibatalkan tautannya.",
        "unlinkaccounts": "Lepastautkan akun",
        "unlinkaccounts-success": "Akun berikut telah dilepastautkan.",
        "authenticationdatachange-ignored": "Otentikasi perubahan data tidak dijalankan. Mungkin tidak ada provider yang diatur?",
index ee0890d..52f01ef 100644 (file)
        "parser-unstrip-loop-warning": "Renkontresis nefinita procedo ('loop') en la funciono \"Unstrip\"",
        "undo-success": "La redakto ne povas desfacesar.\nVerifikez adinfre per komparo inter la du versioni se to esas fakte quon vu deziras facar; pose 'salvez' la modifiki por kompletigar la redakto.",
        "undo-failure": "Ne povis nuligar la redakto pro konflikti kun intermeza redakti.",
+       "undo-norev": "Ne povis desfacar la redakto, pro ol sive efacesis, sive ne existas.",
        "undo-summary-username-hidden": "Desfacar revizo $1 facita da celita uzero",
        "cantcreateaccount-text": "La kreo di konto de ica adreso IP (<strong>$1</strong>) blokusesis da [[User:$3|$3]].\n\nLa motivo, segun $3, esas <em>$2</em>",
        "cantcreateaccount-range-text": "La kreo di konti de IP-adresi de <strong>$1</strong>, qua inkluzas vua IP-adreso (<strong>$4</strong>), blokusesis dal uzero [[User:$3|$3]].\n\nLa motivo quon $3 informis por la blokuso esis <em>$2</em>",
        "pageswithprop": "Pagini kun atributo di pagino",
        "pageswithprop-legend": "Pagini kun atributo di pagino",
        "pageswithprop-text": "Ica pagino listas pagini qui havas partikulara propraji.",
+       "pageswithprop-reverse": "Ordenar inverse",
+       "pageswithprop-sortbyvalue": "Ordenar segun valoro di proprajo",
        "pageswithprop-submit": "Irez",
        "doubleredirects": "Duopla ridirektili",
        "doubleredirectstext": "Ca pagino montras pagini qui ridirektas ad altra ridirekto-pagini.\nSingla lineo kontenas ligili al unesma e a la duesma ridirekto, ed anke la emo di la duesma ridirekto, qua ordinare esas la \"vera\" emo-pagino a quo l'unesma ridirekto mustus apuntar.\n<del>Surstrekizata ligili</del> reparesis.",
        "ncategories": "$1 {{PLURAL:$1|kategorio|kategorii}}",
        "nlinks": "$1 {{PLURAL:$1|ligilo|ligili}}",
        "nmembers": "$1 {{PLURAL:$1|membro|membri}}",
+       "nmemberschanged": "$1 → $2 {{PLURAL:$2|membro|membri}}",
        "specialpage-empty": "Existas nula rezulti por ica informo.",
        "lonelypages": "Pagini sen ligili",
        "uncategorizedpages": "Nekategorizita pagini",
        "uctop": "aktuala",
        "month": "De monato (e plu frue):",
        "year": "De yaro (e plu frue):",
+       "date": "De (ed ante) la dato:",
        "sp-contributions-newbies": "Montrez nur kontributadi di la nova uzeri",
        "sp-contributions-newbies-sub": "Dil nova uzeri",
        "sp-contributions-newbies-title": "Kontributaji dil nova uzeri",
        "thumbnail-more": "Plugrandigar",
        "thumbnail_error": "Ne sucesas krear imajeto: $1",
        "import": "Importacar pagini",
+       "import-upload-filename": "Nomo dil arkivo:",
        "import-comment": "Komento:",
        "importtext": "Voluntez exportacar l' arkivo de la fonto-wikio per [[Special:Export|exportacilo]]. Registragar ol a vua komputero ed adkargar ol hike.",
        "importfailed": "La importaco faliis: $1",
        "redirect-page": "Identigo di la pagino",
        "redirect-revision": "Revizo di la pagino",
        "redirect-file": "Arkivo-nomo",
+       "fileduplicatesearch-filename": "Nomo dil arkivo:",
        "fileduplicatesearch-submit": "Serchar",
        "specialpages": "Specala pagini",
        "specialpages-group-maintenance": "Raporti pri manteno",
        "tag-mw-replace": "Remplasita",
        "tag-mw-replace-description": "Redakturi qui removas plua kam 90% de la kontenajo di ula pagino",
        "tag-mw-rollback": "Volvar addope",
+       "tag-mw-undo": "Desfacez",
        "tags-title": "Etiketi",
        "tags-intro": "Ica pagino montras l'etiketi qui povas uzesar dal informatik-programo por markizar ula redakturo, e lia signifiko.",
        "tags-tag": "Nomo dil etiketo",
index a8c88a5..5c29cf1 100644 (file)
@@ -74,6 +74,7 @@
        "tog-norollbackdiff": "Ekki sýna breytingu eftir að endurvakning síðu hefur verið gerð.",
        "tog-useeditwarning": "Vara mig við þegar ég fer frá breytingarsíðu með óvistaðar breytingar",
        "tog-prefershttps": "Alltaf nota örugga tengingu þegar þú skráir þig inn",
+       "tog-showrollbackconfirmation": "Sýna staðfestingarglugga þegar smellt er á taka aftur tengil",
        "underline-always": "Alltaf",
        "underline-never": "Aldrei",
        "underline-default": "Skinn eða sjálfgefið í vafra",
        "modifiedarticleprotection": "breytti verndunarstigi fyrir \"[[$1]]\"",
        "unprotectedarticle": "afverndaði „[[$1]]“",
        "movedarticleprotection": "verndunarstilling hefur verið færð frá „[[$2]]“ á „[[$1]]“",
+       "protectedarticle-comment": "{{GENDER:$2|Verndaði}} „[[$1]]“",
+       "modifiedarticleprotection-comment": "{{GENDER:$2|Breytti verndunarstigi}} fyrir „[[$1]]“",
+       "unprotectedarticle-comment": "{{GENDER:$2|Fjarlægði verndun}} á „[[$1]]“",
        "protect-title": "Vernda „$1“",
        "protect-title-notallowed": "Skoða verndunarstig $1",
        "prot_1movedto2": "[[$1]] færð á [[$2]]",
        "autoblocklist": "Sjálfvirk bönn",
        "autoblocklist-submit": "Leita",
        "autoblocklist-legend": "Sýna sjálfvirk bönn",
+       "autoblocklist-localblocks": "{{PLURAL:$1|Staðbundið sjálfvirkt bann|Staðbundin sjálfvirk bönn}}",
        "autoblocklist-total-autoblocks": "Samanlagður fjöldi sjálfvirkra banna: $1",
        "autoblocklist-empty": "Listinn yfir sjálfvirk bönn er tómur.",
        "autoblocklist-otherblocks": "{{PLURAL:$1|Annað sjálfvirkt bann|Önnur sjálfvirk bönn}}",
        "tag-list-wrapper": "[[Special:Tags|{{PLURAL:$1|Merki}}]]: $2",
        "tag-mw-new-redirect": "Ný endurbeining",
        "tag-mw-removed-redirect": "Fjarlægði endurbeiningu",
+       "tag-mw-blank": "Tæming",
+       "tag-mw-blank-description": "Breytingar sem tæma síðu.",
        "tag-mw-replace": "Skipt út",
+       "tag-mw-replace-description": "Breytingar sem fjarlægja meira en 90% af innihaldi síðna.",
        "tag-mw-rollback": "Afturköllun",
+       "tag-mw-rollback-description": "Breytingar sem taka til baka fyrri breytingar með taka til baka tenglinum.",
        "tag-mw-undo": "Afturkalla",
+       "tag-mw-undo-description": "Breytingar sem taka til baka fyrri breytingar með {{int:editundo}} tenglinum",
        "tags-title": "Merki",
        "tags-intro": "Þessi síða sýnir merkin sem hugbúnaðurinn gæti merkt breytingar með, og hvað þau þýða.",
        "tags-tag": "Heiti merkis",
        "htmlform-date-placeholder": "ÁÁÁÁ-MM-DD",
        "htmlform-time-placeholder": "KK:MM:SS",
        "htmlform-datetime-placeholder": "ÁÁÁÁ-MM-DD KK:MM:SS",
+       "htmlform-date-invalid": "Uppgefið gildi er ekki þekkt. Reyndu að nota ÁÁÁÁ-MM-DD formið.",
+       "htmlform-time-invalid": "Uppgefið gildi er ekki þekktur tími. Reyndu að nota KK:MM:SS formið.",
+       "htmlform-datetime-invalid": "Uppgefið gildi er ekki þekkt dagsetning og tími. Reyndu að nota ÁÁÁÁ-MM-DD KK:MM:SS formið.",
+       "htmlform-date-toolow": "Gildið sem þú gafst upp er fyrir elstu leyfðu dagsetninguna $1.",
+       "htmlform-date-toohigh": "Gildið sem þú gafst upp er eftir síðustu leyfðu dagsetninguna $1.",
+       "htmlform-time-toolow": "Gildið sem þú gafst upp er fyrir elstu leyfðu tímasetninguna $1.",
+       "htmlform-time-toohigh": "Gildið sem þú gafst upp er eftir síðustu leyfðu tímasetninguna $1.",
+       "htmlform-datetime-toolow": "Gildið sem þú gafst upp er fyrir elstu leyfðu dagsetninguna og tímasetninguna $1.",
+       "htmlform-datetime-toohigh": "Gildið sem þú gafst upp er eftir síðustu leyfðu dagsetninguna og tímasetninguna $1.",
        "htmlform-title-badnamespace": "[[:$1]] er ekki í \"{{ns:$2}}\" nafnrýminu.",
        "htmlform-title-not-creatable": "\"$1\" er ekki hægt að nota við að búa til titil á síðu",
        "htmlform-title-not-exists": "$1 er ekki til",
        "logentry-block-block": "$1 {{GENDER:$2|bannaði}} {{GENDER:$4|$3}}, rennur út eftir $5 $6",
        "logentry-block-unblock": "$1 {{GENDER:$2|afbannaði}} {{GENDER:$4|$3}}",
        "logentry-block-reblock": "$1 {{GENDER:$2|breytti}} bann stillingum fyrir {{GENDER:$4|$3}}, rennur út $5 $6",
+       "logentry-partialblock-block": "$1 {{GENDER:$2|bannaði}} {{GENDER:$4|$3}} frá því að breyta $7, rennur út eftir $5 $6",
+       "logentry-partialblock-reblock": "$1 {{GENDER:$2|breytti}} bönnunar stillingum fyrir {{GENDER:$4|$3}} og hindraði breytingar á $7, rennur út $5 $6",
+       "logentry-non-editing-block-block": "$1 {{GENDER:$2|bannaði}} {{GENDER:$4|$3}} frá sérstökum aðgerðum öðrum en breytingum, rennur út $5 $6",
+       "logentry-non-editing-block-reblock": "$1 {{GENDER:$2|breytti}} bönnunarstillingum fyrir {{GENDER:$4|$3}} fyrir sérstakar aðgerðir aðrar en breytingar, rennur út eftir $5 $6",
        "logentry-suppress-block": "$1 {{GENDER:$2|bannaði}} {{GENDER:$4|$3}}, rennur út eftir $5 $6",
        "logentry-suppress-reblock": "$1 {{GENDER:$2|breytti}} bannstillingum fyrir {{GENDER:$4|$3}}, rennur út $5 $6",
        "logentry-import-upload": "$1 {{GENDER:$2|flutti inn}} $3 úr innsendri skrá",
index 31bd626..672027c 100644 (file)
        "rcfilters-clear-all-filters": "Pulisci tutti i filtri",
        "rcfilters-show-new-changes": "Visualizza le modifiche dal $1",
        "rcfilters-search-placeholder": "Filtra le modifiche (usa il menu o cerca il nome del filtro)",
+       "rcfilters-search-placeholder-mobile": "Filtri",
        "rcfilters-invalid-filter": "Filtro non valido",
        "rcfilters-empty-filter": "Nessun filtro attivo. Sono mostrati tutti i contributi.",
        "rcfilters-filterlist-title": "Filtri",
        "passwordpolicies-policy-passwordnotinlargeblacklist": "La password non può essere nell'elenco delle 100 000 password utilizzate più comunemente.",
        "easydeflate-invaliddeflate": "Il contenuto fornito non è compresso correttamente",
        "unprotected-js": "Per motivi di sicurezza, non è possibile caricare JavaScript da pagine non protette. Crea javascript solo nel namespace MediaWiki o come sottopagina Utente",
-       "userlogout-continue": "Vuoi uscire?"
+       "userlogout-continue": "Vuoi davvero uscire?"
 }
index 58383e9..71aebe9 100644 (file)
        "rcfilters-clear-all-filters": "すべてのフィルターをクリア",
        "rcfilters-show-new-changes": "$1 から最新の変更を表示",
        "rcfilters-search-placeholder": "絞り込みを行う(メニューから選択、またはフィルター名で検索)",
+       "rcfilters-search-placeholder-mobile": "フィルター",
        "rcfilters-invalid-filter": "無効なフィルター",
        "rcfilters-empty-filter": "絞り込みは行われていません。全ての項目が表示されます。",
        "rcfilters-filterlist-title": "フィルター",
        "mediastatistics-header-audio": "オーディオ",
        "mediastatistics-header-video": "動画",
        "mediastatistics-header-multimedia": "リッチメディア",
-       "mediastatistics-header-office": "オフィス",
+       "mediastatistics-header-office": "オフィスファイル",
        "mediastatistics-header-text": "テキスト",
        "mediastatistics-header-executable": "実行ファイル",
        "mediastatistics-header-archive": "圧縮フォーマット",
index 5d7d405..b4c04cc 100644 (file)
        "rcfilters-clear-all-filters": "필터 모두 지우기",
        "rcfilters-show-new-changes": "$1 이후 새 변경사항 보기",
        "rcfilters-search-placeholder": "필터 변경(메뉴를 사용하거나 필터 이름을 검색하세요)",
+       "rcfilters-search-placeholder-mobile": "필터",
        "rcfilters-invalid-filter": "유효하지 않은 필터",
        "rcfilters-empty-filter": "활성화된 필터가 없습니다. 모든 기여가 표시됩니다.",
        "rcfilters-filterlist-title": "필터",
        "specialmute-error-invalid-user": "요청한 사용자 이름을 찾을 수 없습니다.",
        "specialmute-email-footer": "{{BIDI:$2}}의 이메일 환경 설정을 관리하려면 <$1>을(를) 방문해 주십시오.",
        "specialmute-login-required": "알림 미표시 환경 설정을 변경하려면 로그인해 주십시오.",
+       "mute-preferences": "알림 미표시 환경 설정",
        "revid": "$1 판",
        "pageid": "페이지 ID $1",
        "interfaceadmin-info": "$1\n\n사이트 전체에 쓰이는 CSS/JS/JSON 파일의 편집 권한이 최근 <code>editinterface</code> 권한에서 분리되었습니다. 왜 이 오류가 발생하는지 이해가 되지 않는다면, [[mw:MediaWiki_1.32/interface-admin]]을 참고하십시오.",
index af92424..094b875 100644 (file)
        "thu": "Pşm",
        "fri": "În",
        "sat": "Şem",
-       "january": "rêbendan",
-       "february": "reşemî",
+       "january": "kanûna paşîn",
+       "february": "sibat",
        "march": "adar",
-       "april": "avrêl",
+       "april": "nîsan",
        "may_long": "gulan",
-       "june": "pûşper",
+       "june": "hezîran",
        "july": "tîrmeh",
-       "august": "gelawêj",
-       "september": "rezber",
-       "october": "kewçêr",
-       "november": "sermawez",
-       "december": "berfanbar",
-       "january-gen": "Rêbendan",
-       "february-gen": "Reşemî",
-       "march-gen": "Adar",
-       "april-gen": "Avrêl",
+       "august": "tebax",
+       "september": "îlon",
+       "october": "çiriya pêşîn",
+       "november": "çiriya paşîn",
+       "december": "kanûna pêşîn",
+       "january-gen": "kanûna paşîn",
+       "february-gen": "sibat",
+       "march-gen": "adar",
+       "april-gen": "nîsan",
        "may-gen": "gulan",
-       "june-gen": "pûşper",
-       "july-gen": "Tîrmeh",
-       "august-gen": "Gelawêj",
-       "september-gen": "rezber",
-       "october-gen": "Kewçêr",
-       "november-gen": "Sermawez",
-       "december-gen": "berfanbar",
-       "jan": "rêb",
-       "feb": "reş",
+       "june-gen": "hezîran",
+       "july-gen": "tîrmeh",
+       "august-gen": "tebax",
+       "september-gen": "îlon",
+       "october-gen": "çiriya pêşîn",
+       "november-gen": "çiriya paşîn",
+       "december-gen": "kanûna pêşîn",
+       "jan": "kpa",
+       "feb": "sib",
        "mar": "adr",
-       "apr": "avr",
+       "apr": "nîs",
        "may": "gln",
-       "jun": "pûş",
+       "jun": "hez",
        "jul": "tîr",
-       "aug": "Glw",
-       "sep": "rez",
-       "oct": "kew",
-       "nov": "ser",
-       "dec": "ber",
-       "january-date": "Rêbendan $1",
-       "february-date": "Reşemî $1",
+       "aug": "teb",
+       "sep": "îln",
+       "oct": "çpê",
+       "nov": "çpa",
+       "dec": "kpê",
+       "january-date": "Kanûna paşîn $1",
+       "february-date": "Sibat $1",
        "march-date": "Adar $1",
-       "april-date": "Avrêl $1",
+       "april-date": "Nîsan $1",
        "may-date": "Gulan $1",
        "june-date": "Pûşper $1",
        "july-date": "Tîrmeh $1",
-       "august-date": "Gelawêj $1",
+       "august-date": "$1 tebax",
        "september-date": "Rezber $1",
        "october-date": "Kewçêr $1",
        "november-date": "Sermawez $1",
        "deletepage": "Rûpelê jê bibe",
        "confirm": "Pesend bike",
        "excontent": "Naveroka berê: \"$1\"",
-       "excontentauthor": "Naveroka vê rûpelê ev bû: '$1' (û tenya bikarhêner '$2' bû)",
+       "excontentauthor": "Naveroka rûpelê ev bû: \"$1\", û tenê ya bikarhêner \"[[Special:Contributions/$2|$2]]\" ([[User talk:$2|talk]] bû.)",
        "exbeforeblank": "Naverok berî betalkirinê ev bû: \"$1\"",
        "delete-confirm": "Jêbirina \"$1\"",
        "delete-legend": "Jê bibe",
index d61a2e6..85f9d18 100644 (file)
        "rcfilters-clear-all-filters": "All Filteren eidelmaachen",
        "rcfilters-show-new-changes": "Nei Ännerungen zanter $1 weisen",
        "rcfilters-search-placeholder": "Ännerunge filteren (benotzt de Menü oder sicht nom Numm vum Filter)",
+       "rcfilters-search-placeholder-mobile": "Filteren",
        "rcfilters-invalid-filter": "Net valabele Filter",
        "rcfilters-empty-filter": "Keen aktive Filter. All Kontributioune gi gewisen.",
        "rcfilters-filterlist-title": "Filteren",
index cb6b134..de1f164 100644 (file)
        "printableversion": "Verscion da stanpâ",
        "permalink": "Ingancio fisso",
        "print": "Stampa",
-       "view": "Visualizza",
+       "view": "Vixoalìzza",
        "view-foreign": "Véddi in sce $1",
        "edit": "Modìfica",
        "edit-local": "Modifica descrission locale",
        "page_first": "prìmma",
        "page_last": "ùrtima",
        "histlegend": "Confronto tra verscioîn: selession-a e cascette corispondenti a-e verscioîn dexidiæ e schissa Invio oppû o pomello da basso.\n\nLegenda: (corr) = differense co-a verscion corrente, (prec) = differense co-a verscion precedente, '''m''' = modiffica minô",
-       "history-fieldset-title": "Çerca de verscioin",
+       "history-fieldset-title": "Çerca e verscioin",
        "history-show-deleted": "Solo verscioin scassæ",
        "histfirst": "prìmma",
        "histlast": "ùrtima",
        "recentchanges-page-removed-from-category": "[[:$1]] rimosso da-a categoria",
        "recentchanges-page-removed-from-category-bundled": "[[:$1]] rimossa da-a categoria, [[Special:WhatLinksHere/$1|questa pagina a l'è inclusa a l'interno di atre pagine]]",
        "autochange-username": "Modiffica aotomattica MediaWiki",
-       "upload": "Carrega 'n file",
+       "upload": "Càrega 'n file",
        "uploadbtn": "Carreghilo",
        "reuploaddesc": "Torna a-o moddulo pe-o caregamento.",
        "upload-tryagain": "Invia a descrission do file modificou",
        "imagelinks": "Utilìzzo do file",
        "linkstoimage": "{{PLURAL:$1|A segoente pàgina a contegne|E segoenti $1 pàgine contegnan}} colegaménti a-o file:",
        "linkstoimage-more": "Ciù de $1 {{PLURAL:$1|pagina aponta|pagine apontan}} a questo file.\nA seguente lista a mostra {{PLURAL:$1|a primma paggina ch'a l'aponta|e primme $1 paggine ch'apontan}} a sto file.\nL'è disponibile un [[Special:WhatLinksHere/$2|elenco completo]].",
-       "nolinkstoimage": "No gh'è nisciun-a paggina inganciâ a sto file.",
+       "nolinkstoimage": "No gh'è nisciun-a pàgina ch'a contegne sto file.",
        "morelinkstoimage": "Vixualizza [[Special:WhatLinksHere/$1|di atri inganci]] a questo file.",
        "linkstoimage-redirect": "$1 (rendriççamento file) $2",
        "duplicatesoffile": "{{PLURAL:$1|O seguente file o l'è un dupricou|I seguenti $1 file son di dupricæ}} de questo file ([[Special:FileDuplicateSearch/$2|urteioî detaggi]]):",
        "tooltip-t-contributions": "Lista de contribûssioin de {{GENDER:$1|questo|questa}} utente",
        "tooltip-t-emailuser": "Invia un messaggio email a {{GENDER:$1|questo|questa}} utente",
        "tooltip-t-info": "Urteioî informaçioin insce questa pagina",
-       "tooltip-t-upload": "Carrega di file murtimediali",
+       "tooltip-t-upload": "Càrega di files",
        "tooltip-t-specialpages": "Lista de tùtte e pagine speçiâli",
        "tooltip-t-print": "Verscion stanpabbile de sta paggina",
        "tooltip-t-permalink": "Colegaménto fisso a sta revixión da pàgina",
index 4c0e672..f24aa1d 100644 (file)
        "expansion-depth-exceeded-category-desc": "زؽر دٱسٱ سی بٱلگٱیایؽ کاْ د ڤنو پی یا ڤلٱ بیئن فرٱ پیشکرد کردٱ.",
        "expansion-depth-exceeded-warning": "بٱلگٱ د پی یا ڤلٱ بیئن پیشکرد کرد",
        "parser-unstrip-loop-warning": "گردۊلٱ د فرمونٱ Unstrip پاٛدا بیٱ",
-       "unstrip-depth-warning": "د Ø¨Ø½Ø´ØªØ±Ù\88Ù\86Ù± Ø¯ Ø³Ø±Ú\86Ø´Ù\85Ù± Ø±Ù±ØªÙ\86 Ø¯ Ø¯Ù±Ø³Ø¯Û\8aر Unstrip Ú¤Ø§Ø±Û\89تر رٱتؽتٱ($1)",
+       "unstrip-depth-warning": "د Ø¨Ø½Ø´ØªØ±Ù\88Ù\86Ù± Ø¯ Ø³Ø±Ú\86Ø´Ù\85Ù± Ø±Ù±ØªÙ\86 Ø¯ Ø¯Ù±Ø³Ø¯Û\8aر Unstrip Ú¤Ø§Ø±Û\8fتر رٱتؽتٱ($1)",
        "converter-manual-rule-error": "خٱتا د قانۊن ڤالٛرشتن دٱسی زڤوݩ",
        "undo-success": "نمۊئٱ ڤیرایش ناْ ٱنجومشیو بٱکؽت.\nلوتفٱن اؽ فٱرخؽ کاْ ها د هار ناْ ڤارسی بٱکؽت تا یاٛ کارؽ کاْ مؽهایت ٱنجوم باٛیؽت،ۉ اۊساْ آلشتؽا هار ناْآمادٱ بٱکؽت سی یٱ کاْ خونسا کردن ڤیرایش ناْ ٱنجوم باٛیؽت.",
        "undo-failure": "سی ری ڤ ری بیئن اؽ ڤیرایش ڤا ڤیرایشؽا مؽنجایی، نمۊئٱ اؽ ڤیرایش ناْ خونسا بٱکؽت.",
        "difference-missing-revision": "{{PLURAL:$2|یاٛ ڤیرایش|$2 ڤیرایش}} د فٱرخ مؽنجا($1) {{PLURAL:$2|پاٛدا ناٛی|پاٛدا ناٛییٱ}}.\n\nشایٱد بانی جاڤٱنٱ ڤٱ ڤا یاٛ ڤیرگار ڤٱ هٱنگوم ناٛییٱ کاْ د یاٛ بٱلگٱ پاکسا بیٱ هوم پاٛڤٱن باٛئٱ بۊئٱ.\nشایٱد جۏزییات د  [{{fullurl:{{#Special:Log}}/delete|page={{FULLPAGENAMEE}}}} deletion log]  پاٛدا مۊئٱن.",
        "searchresults": "نتيجٱیا پاٛ جۊری",
        "searchresults-title": "نٱتيجٱیا پاٛ جۊری سی \"$1\"",
-       "titlematches": "داسون بلگه یکی بیه",
-       "textmatches": "هومسازی نیسسٱ بلگٱ.",
+       "titlematches": "داسوݩ بٱلگٱ یٱکؽ بیٱ",
+       "textmatches": "Ù\87Ù\88Ù\85سازÛ\8c Ù\86Û\8cسسٱ Ø¨Ù±Ù\84Ú¯Ù±.",
        "notextmatches": "نیسسٱ بٱلگٱ هومسازی نارٱ",
        "prevn": "ڤادما {{PLURAL:$1|$1}}",
        "nextn": "نوئایی {{PLURAL:$1|$1}}",
        "next-page": "بٱلگٱ نوئایی",
        "prevn-title": "زیتر $1 {{PLURAL:$1|نٱتیجٱ|نٱتيجٱيا}}",
        "nextn-title": "دمایی $1 {{PLURAL:$1|نٱتيجٱ|نٱتيجؽا}}",
-       "shown-title": "نشوݩ داٛین $1 {{PLURAL:$1|نتیجٱ|نتیجٱیا}} سی هٱر بٱلگٱ",
+       "shown-title": "نشوݩ داٛئن $1 {{PLURAL:$1|نتیجٱ|نتیجٱیا}} سی هٱر بٱلگٱ",
        "viewprevnext": "ديئن ($1 {{int:pipe-separator}} $2) ($3)",
        "searchmenu-exists": "'''ایچاْ بٱلگاٛیؽ هؽ ڤ نوم\"[[:$1]]\" کاْ ها د اؽ ڤیکی'''",
        "searchmenu-new": "'''اؽ بٱلگٱ ناْ دۏرس كو \"[[:$1]]\" د اؽ ڤیکی!'''",
        "searchprofile-articles": "بٱلگٱيا مؽنونٱ دار",
-       "searchprofile-images": "ڤارسگرؽا خلکمٱن",
-       "searchprofile-everything": "همٱ چی",
+       "searchprofile-images": "ڤارسگرؽا Ø®Ù±Ù\84Ú©Ù\85Ù±Ù\86",
+       "searchprofile-everything": "Ù\87Ù±Ù\85Ù± Ú\86Û\8c",
        "searchprofile-advanced": "پیشکردٱ",
        "searchprofile-articles-tooltip": "بٱگٱرد مؽن $1",
-       "searchprofile-images-tooltip": "جانؽایا ناْ پاٛ جۊری کو",
-       "searchprofile-everything-tooltip": "همٱ مؽنونٱیا ناْ پاٛ جۊری كو (شامل بٱلگٱيا چٱک چنٱ)",
+       "searchprofile-images-tooltip": "جانؽایا ناْ پاٛجۊری کو",
+       "searchprofile-everything-tooltip": "همٱ مؽنونٱیا ناْ پاٛجۊری كو (شامل بٱلگٱيا چٱک چنٱ)",
        "searchprofile-advanced-tooltip": "نوم جايا نوم دؽار بٱگٱرد",
        "search-result-size": "$1 ({{PLURAL:$2|1 کلٱمٱ|$2 کلٱمٱیا}})",
        "search-result-category-size": "{{PLURAL:$1|1 ٱندوم|$1 ٱندومؽا}} ({{PLURAL:$2|1 زؽردٱسٱ|$2 زؽردٱسٱیا}}، {{PLURAL:$3|1 جانؽا|$3 جانؽایا}}",
        "search-redirect": "(ڤرگٱشتن سی $1)",
        "search-section": "(بٱئرجا $1)",
-       "search-category": "(دسه $1)",
+       "search-category": "(دٱسٱ $1)",
        "search-file-match": "(یٱکؽ کردن مؽنونٱ جانؽا)",
        "search-suggest": "مٱنزۊرت يٱ بی:$1",
-       "search-rewritten": "Ù\86ئشÙ\88Ù\99 Ø¯Ø£Ø¦Ù\86 Ù\86أتÛ\8cجÛ\95 Û\8cا Ø³Û\8c $1. Ø³Û\8c Ù\86ئÙ\85Ù\88Ù\99Ù\86Û\95 Ø¨Ø£Ú¯Ø£Ø±Ø¯Û\8cت سی $2.",
+       "search-rewritten": "Ù\86Ø´Ù\88Ý© Ø¯Ø§Ù\9bئÙ\86 Ù\86ٱتÛ\8cجٱÛ\8cا Ø³Û\8c $1. Ø³Û\8c Ù\86Ù\85Û\8aÙ\86Ù± Ø¨Ù±Ú¯Ù±Ø±Ø¯Ø½ت سی $2.",
        "search-interwiki-caption": "پروجه یا خوئر",
-       "search-interwiki-default": "$1 نتیجه یا:",
-       "search-interwiki-more": "(بیشتر)",
-       "search-relatedarticle": "مرتوط",
-       "searchrelated": "مرتوط",
-       "searchall": "همٱ",
+       "search-interwiki-default": "$1 نٱتیجٱیا:",
+       "search-interwiki-more": "(بؽشتر)",
+       "search-relatedarticle": "مورتٱبت",
+       "searchrelated": "مورتٱبت",
+       "searchall": "Ù\87Ù±Ù\85Ù±",
        "showingresults": "نمایش بؽشترونٱ {{PLURAL:$1|'''۱''' نتیجٱ|'''$1''' نتیجٱ}} د هار، شرۊ د شمارٱ'''$2'''.",
-       "showingresultsinrange": "نمایش بؽشترونٱ {{PLURAL:$1|'''۱''' نتیجٱ|'''$1''' نتیجٱ}} د هار، شرۊ د شمارٱ'''$2''' تا شمارٱ '''$3'''.",
+       "showingresultsinrange": "نمایش بؽشترونٱ {{PLURAL:$1|'''۱''' نٱتیجٱ|'''$1''' نٱتیجٱ}} د هار، شرۊ د شمارٱ'''$2''' تا شمارٱ '''$3'''.",
        "search-showingresults": "{{PLURAL:$4|نٱتیجٱیا<strong>$1</strong> د <strong>$3</strong>|نٱتیجٱیا<strong>$1 - $2</strong د <strong>$3</strong>}}",
-       "search-nonefound": "هیچ نتیجاٛیؽ ڤا پاٛجۊری تو یٱکؽ نؽ.",
-       "powersearch-legend": "پی جوری پیشکرده",
-       "powersearch-ns": "د نوم جايا نوم ديار پی جوری بک:",
-       "powersearch-togglelabel": "ڤارئسÛ\8c Ú©Ø¦Ø±Ø¯Ø¦ن:",
-       "powersearch-toggleall": "هأمە",
-       "powersearch-togglenone": "هيش كوم",
-       "powersearch-remember": "د ویر داشتن انتخاو سی پی جوریا نهایی",
-       "search-external": "پی جوری د در",
-       "searchdisabled": "Ù\85ئÙ\86 Ø¬Ù\88رÛ\8c Ø¯ {{SITENAME}} Ú©Ù\86شتگر Ù\86ئ.\nÙ\85Ù\88Ù\82تاÙ\8b Ù\85Û\8c ØªÙ\88Ù\86Û\8cت Ù\85ئÙ\86 Ø¬Ù\88رÛ\8c Google Ù\86Ù\87 Ø¨Ù\88Ù\86Û\8cت Ù\88Ù\87 Ú©Ø§Ø±.\nد Ù\88Û\8cرتÙ\88 Ø¨Ø§ Ú©Ù\87 Ù\86تÛ\8cجÙ\87 Û\8cاÛ\8cÛ\8c Ú©Ù\87 Ø¯ Ù\85ئÙ\86 Ø¬Ù\88رÛ\8c Ù\88ا Ø§Ù\88 Ø±Ù\88شت Ù\88Ù\87 Ø¯Ø³Øª Ù\85Û\8cاÙ\86 Ø´Ø§Û\8cت Ù\88Ù\87 Ø±Ù\88ز Ù\86بان.",
-       "search-error": "یه گل خطا سی اوسنی که پی جوری می کردیت اتفاق افتائه:$1",
-       "preferences": "Ø®Ù\88صÙ\88Ù\99Û\8cات Ù\87Ø£نی",
+       "search-nonefound": "هیچ نٱتیجاٛیؽ ڤا پاٛجۊری تو یٱکؽ نؽ.",
+       "powersearch-legend": "پاٛجۊری پیشکردٱ",
+       "powersearch-ns": "د نوم جايا نوم دؽار پاٛجۊری کو:",
+       "powersearch-togglelabel": "ڤارسÛ\8c Ú©Ø±Ø¯ن:",
+       "powersearch-toggleall": "هٱمٱ",
+       "powersearch-togglenone": "هیچ کوم",
+       "powersearch-remember": "د ڤیر داشتن اْنتخاب سی پاٛجۊرؽا نهایی",
+       "search-external": "پاٛجۊری د دٱر",
+       "searchdisabled": "Ù\85ؽÙ\86 Ø¬Û\8aرÛ\8c Ø¯ {{SITENAME}} Ú©Ù\86شگٱر Ù\86ؽ.\nÙ\85Ù\88Ú¤Ù±Ù\82ٱتٱÙ\86 Ù\85ؽتÙ\88Ù\86ؽت Ù\85ؽÙ\86 Ø¬Û\8aرÛ\8c Google Ù\86اÙ\92 Ø¨Ù\88Ù\86ؽت Ú¤ Ú©Ø§Ø±.\nد Ú¤Û\8cرتÙ\88 Ø¨Ø§ Ú©Ø§Ù\92 Ù\86ٱتÛ\8cجٱÛ\8cاÛ\8cؽ Ú©Ø§Ù\92 Ø¯ Ù\85ؽÙ\86 Ø¬Ù\88Ù¾Û\8aرÛ\8c Ú¤Ø§ Ø§Ù\88 Ø±Ù±Ú¤Ø´ Ú¤ Ø¯Ù±Ø³ Ù\85ؽاÙ\86 Ø´Ø§Û\8cٱد Ú¤ Ø±Û\8aز Ù\86Û\8aئٱن.",
+       "search-error": "یاٛ خٱتا سی اۊسنؽ کاْ پاٛجۊری مؽ کردؽت اْتفاق اوفتایٱ:$1",
+       "preferences": "Ø®Ù\88سÛ\8aسÛ\8cات Ù\87نی",
        "mypreferences": "چیا هنی",
-       "prefs-edits": "Ø´Ù\88Ù\85ارÛ\95 Ú¤Û\8cراÛ\8cئشتÛ\8cا:",
-       "prefsnologintext2": "لطف بکیت بیایت وامین و ترجیحات خوتونه آلشت بئیت.",
-       "prefs-skin": "پوس",
-       "skin-preview": "Ù¾Û\8cØ´ Ø³Ø¦Û\8cÙ\84",
-       "datedefault": "هیچ ترجیحات دش نئ",
-       "prefs-labs": "گزینشتیا ازماشتی",
-       "prefs-user-pages": "بألگە کاریار",
-       "prefs-personal": "جانیاگە کاریار",
-       "prefs-rc": "Ø¢Ù\84ئشتÛ\8cا Ø¦Û\8cسئنی",
-       "prefs-watchlist": "سئÛ\8cÙ\84 Ø¨Ø£رگ",
-       "prefs-editwatchlist": "Ú¤Û\8cراÛ\8cئشت Ø³Ø¦Û\8cÙ\84 Ø¨Ø£رگ",
-       "prefs-editwatchlist-label": "دادÙ\87 Û\8cا Ù\86Ù\87 Ø¯ Ø³Û\8cÙ\84 Ø¨Ø±Ú¯ Ø®Ù\88تÙ\88 Ù\88Û\8cراÛ\8cشت Ø¨Ú©Û\8cت:",
-       "prefs-editwatchlist-edit": "داسÙ\88Ù\86اÙ\86Ù\87 Ø³Û\8cÙ\84 Ø¨Ú©Û\8cت Ù\88 Ù\88Ù\86Ù\88Ù\86Ù\87 Ø¯ Ø³Û\8cÙ\84 Ø¨Ø±Ú¯ Ø®Ù\88تÙ\88 Ù\88ردارÛ\8cت",
-       "prefs-editwatchlist-raw": "Ú¤Û\8cراÛ\8cئشتکارÛ\8c Ø±Ø£Ø¯Û\8cÙ\81Û\8c Ø³Ø¦Û\8cÙ\84 Ø¨Ø£رگ",
-       "prefs-editwatchlist-clear": "سئÛ\8cÙ\84 Ø¨Ø£Ø±Ú¯Ø¦ ØªÙ\88Ù\99Ù\86Û\95 Ù¾Ø§Ú© Ø¨Ø£Ú©Û\8cت",
-       "prefs-watchlist-days": "روزیا نه د سیل برگ نشو دئه بو:",
-       "prefs-watchlist-days-max": "$1 بیشترونه {{PLURAL:$1|روز|روزیا}}",
+       "prefs-edits": "Ø´Ù\85ارٱ Ú¤Û\8cراÛ\8cشؽا:",
+       "prefsnologintext2": "لوتف بٱکؽت بؽایؽت ڤامؽن ۉ تٱرجیهؽا خوتو ناْ آلشت بٱکؽت.",
+       "prefs-skin": "پۊس",
+       "skin-preview": "Ù¾Û\8cØ´ Ø³Ø§Ù\9bÙ\84Ù\9b",
+       "datedefault": "هیچ تٱرجیهاتؽ دش نؽ",
+       "prefs-labs": "گوزینشؽا آزمایشی",
+       "prefs-user-pages": "بٱلگٱ کاریار",
+       "prefs-personal": "جانؽاگٱ کاریار",
+       "prefs-rc": "Ø¢Ù\84شتؽا Ø§Û\8cسنی",
+       "prefs-watchlist": "ساÙ\9bÙ\84Ù\9b Ø¨Ù±رگ",
+       "prefs-editwatchlist": "Ú¤Û\8cراÛ\8cØ´ Ø³Ø§Ù\9bÙ\84Ù\9b Ø¨Ù±رگ",
+       "prefs-editwatchlist-label": "دادٱÛ\8cا Ù\86اÙ\92 Ø¯ Ø³Ø§Ù\9bÙ\84Ù\9b Ø¨Ù±Ø±Ú¯ Ø®Ù\88تÙ\88 Ú¤Û\8cراÛ\8cØ´ Ø¨Ù±Ú©Ø½ت:",
+       "prefs-editwatchlist-edit": "داسÙ\88Ù\86ؽا Ù\86اÙ\92 Ø³Ø§Ù\9bÙ\84Ù\9b Ø¨Ù±Ú©Ø½Øª Û\89 Ú¤Ù\86Ù\88 Ù\86اÙ\92 Ø¯ Ø³Ø§Ù\9bÙ\84Ù\9b Ø¨Ù±Ø±Ú¯ Ø®Ù\88تÙ\88 Ú¤Ø±Ø¯Ø§Ø±Ø½ت",
+       "prefs-editwatchlist-raw": "Ú¤Û\8cراÛ\8cشتکارÛ\8c Ø±Ø¯Û\8cÙ\81Û\8c Ø³Ø§Ù\9bÙ\84Ù\9b Ø¨Ù±رگ",
+       "prefs-editwatchlist-clear": "ساÙ\9bÙ\84Ù\9b Ø¨Ù±Ø±Ú¯ ØªÙ\88 Ù\86اÙ\92 Ù¾Ø§Ú© Ø¨Ù±Ú©Ø½ت",
+       "prefs-watchlist-days": "رۊزؽا د ساٛلٛ بٱرگ نشوݩ داٛئٱ بۊئٱن:",
+       "prefs-watchlist-days-max": "$1 بؽشترونٱ {{PLURAL:$1|رۊز|رۊزؽا}}",
        "prefs-watchlist-edits": "بیشترونه انازه آلشتیایی که د سیل برگ گپ بیه نشو دئه بیه:",
-       "prefs-watchlist-edits-max": "Ø´Ù\85ارÙ\87 Ø¨Û\8cشترÙ\88Ù\86Ù\87:1000",
-       "prefs-watchlist-token": "Ù\86Ø´Ù\88Ù\86Ù\87 Ø³Û\8cÙ\84 Ø¨رگ:",
-       "prefs-misc": "شیڤئسئن",
-       "prefs-resetpass": "رازÛ\8cÙ\86Ù\87 Ú¯Ù\88اردÙ\86 Ù\86Ù\87 Ø¢Ù\84شت Ø¨Ù\83Ù\8aت",
-       "prefs-changeemail": "تÛ\8cرÙ\86ئشÙ\88Ù\99Ù\86 Ø£Ù\86جÙ\88Ù\85اÙ\86اÙ\85Û\95 ØªÙ\88Ù\99Ù\86Û\95 Ø¢Ù\84ئشت Ø¨Ø£Ú©Û\8cت",
-       "prefs-setemail": "يه گل انجومانامه بنیت",
-       "prefs-email": "Ú\86Û\8cا Ù\87Ù\86Û\8c Ø§Ù\86جÙ\88Ù\85اÙ\86اÙ\85Ù\87",
-       "prefs-rendering": "شیڤە",
-       "saveprefs": "ئÙ\85اÛ\8cÛ\95 Ú©Ø¦Ø±Ø¯Ø¦ن",
-       "restoreprefs": "د نو زنه کردن همه میزونکاریا پیش فرض(د همه جایا)",
-       "prefs-editing": "د حال و بال ڤیرایئشت",
-       "searchresultshead": "پئÛ\8c Ø¬Ù\88Ù\99ری",
+       "prefs-watchlist-edits-max": "Ø´Ù\85ارٱ Ø¨Ø½Ø´ØªØ±Ù\88Ù\86Ù±:1000",
+       "prefs-watchlist-token": "Ù\86Ø´Ù\88Ù\86Ù± Ø³Ø§Ù\9bÙ\84Ù\9b Ø¨Ù±رگ:",
+       "prefs-misc": "شؽڤسن",
+       "prefs-resetpass": "رازÛ\8cÙ\86Ù± Ú¯Ù\88ئارسÙ\86 Ù\86اÙ\92 Ø¢Ù\84شت Ø¨Ù±Ú©Ø½ت",
+       "prefs-changeemail": "تÛ\8cرÙ\86Ø´Ù\88Ý© Ù±Ù\86جÙ\88Ù\85اÙ\86اÙ\85Ù± ØªÙ\88 Ù\86اÙ\92 Ø¢Ù\84شت Ø¨Ù±Ú©Ø½ت",
+       "prefs-setemail": "یاٛ ٱنجومانامٱ بٱنؽت",
+       "prefs-email": "Ú\86Û\8cا Ù\87Ù\86Û\8c Ø§Ù\86جÙ\88Ù\85اÙ\86اÙ\85Ù±",
+       "prefs-rendering": "شؽڤٱ",
+       "saveprefs": "Ø¢Ù\85ادٱ Ú©Ø±Ø¯ن",
+       "restoreprefs": "د نۊ زنٱ کردن هٱمٱ میزونکاریا پیش فٱرز(د هٱمٱ جایا)",
+       "prefs-editing": "د هال ۉ بال ڤیرایش",
+       "searchresultshead": "پاÙ\9bجÛ\8aری",
        "stub-threshold": "آستوٙنە ڤیرایئشتکاریا د یأک دیسئسە <a href=\"#\" class=\"stub\">ناقئص</a> (بایت):",
-       "stub-threshold-sample-link": "نئموٙنە",
-       "stub-threshold-disabled": "Ù\86اکÙ\88Ù\86ئشتگأر Ø¨Û\8cÛ\8cÛ\95",
-       "recentchangesdays": "روزیا آلشتیا تازه باو نه نشو بیه:",
-       "recentchangesdays-max": "$1 بیشترونه {{PLURAL:$1|روز|روزیا}}",
+       "stub-threshold-sample-link": "نمۊنٱ",
+       "stub-threshold-disabled": "Ù\86اکÙ\86شگٱر Ø¨Û\8cÙ±",
+       "recentchangesdays": "رۊزؽا آلشتیا تازٱ بۊ ناْ نشوݩ باٛیٱ:",
+       "recentchangesdays-max": "$1 بؽشترونٱ {{PLURAL:$1|رۊز|رۊزؽا}}",
        "recentchangescount": "انازه ویرایشتیایی که دیاری می که:",
        "prefs-help-recentchangescount": "یه شامل آلشتیا تازه،ویرگاریا بلگه و پهرستنومه یا هئ.",
        "prefs-help-watchlist-token2": "یه یه گل کلیت رازینه دار سی خوارک تیارگه سیل برگه شمانه.\nهر کسی که شما مئشناسیت می تونه سیل برگ شما نه بوحونه،په ونه هومبئری نکیت.[[Special:ResetTokens|ار لازمه ونه آلشت بئیت ایچه نه بپورنیت]].",
-       "savedprefs": "ویجه گیا هنی تو اماییه بیه.",
-       "timezonelegend": "گات Ø±Ø§Ø³Ø§Ú¯Ù\87",
-       "localtime": "گات ولاتی:",
-       "timezoneuseserverdefault": "ویکی پیش فرض($1) وه کار بونیت",
+       "savedprefs": "ویژگیا هنی تو آمادٱ بیٱ.",
+       "timezonelegend": "گات Ø±Ø§Ø³Ø§Ú¯Ù±:",
+       "localtime": "گات ڤلاتی:",
+       "timezoneuseserverdefault": "ڤیکی پیش فٱرز($1) ڤ کار بونؽت",
        "timezoneuseoffset": "هنی",
-       "servertime": "گات رسینه جا:",
-       "guesstimezone": "وا جاگرد پر بوئه",
-       "timezoneregion-africa": "اÙ\81رقا",
-       "timezoneregion-america": "اÙ\85رÙ\83ا",
-       "timezoneregion-antarctica": "قطو هار ونه",
-       "timezoneregion-arctic": "قطو شمال",
-       "timezoneregion-asia": "آسيا",
-       "timezoneregion-atlantic": "جهون آو آتلانتیک",
-       "timezoneregion-australia": "استراليا",
-       "timezoneregion-europe": "اوروپا",
-       "timezoneregion-indian": "جهوناو هند",
-       "timezoneregion-pacific": "جهوناو آروم",
+       "servertime": "گات رٱسینٱجا:",
+       "guesstimezone": "ڤا جاگرد پور بۊئٱ",
+       "timezoneregion-africa": "اÙ\92Ù\81رÛ\8cقا",
+       "timezoneregion-america": "اÙ\92Ù\85رÛ\8cÚ©ا",
+       "timezoneregion-antarctica": "قوتب هارگٱ",
+       "timezoneregion-arctic": "قوتب شمال",
+       "timezoneregion-asia": "آسؽا",
+       "timezoneregion-atlantic": "جهوݩ آو آتلانتیک",
+       "timezoneregion-australia": "اوسترالٛیا",
+       "timezoneregion-europe": "اورۊپا",
+       "timezoneregion-indian": "جهوݩ آو هند",
+       "timezoneregion-pacific": "جهوݩ آو آروم",
        "allowemail": "انجومانامه نه سی کاریاریا هنی کنشتگر کو",
-       "prefs-searchoptions": "پئÛ\8c Ø¬Ù\88Ù\99ری",
+       "prefs-searchoptions": "پاÙ\9bجÛ\8aری",
        "prefs-namespaces": "نوم جایا",
-       "default": "پيش فرض",
-       "prefs-files": "جانیایا",
-       "prefs-custom-css": "سی اس اس جاافتائه",
-       "prefs-custom-js": "جاوا نیسسه جاافتائه",
+       "default": "پيش فٱرز",
+       "prefs-files": "جانؽایا",
+       "prefs-custom-css": "سی اْس اْس جا اوفتایٱ",
+       "prefs-custom-js": "جاڤا نیسسٱ جا اوفتایٱ",
        "prefs-common-config": " سی اس اس/جاوا اسکریپت بهر بیه سی همه پوسه یا:",
-       "prefs-reset-intro": "شما می تونیت ای بلگه سی د نو زنه کردن ترجیحات خوت وه شکل تیارگه پیش فرض وه کار بوونیت.\nیه ورئشت پذیر نئ.",
-       "prefs-emailconfirm-label": "پش راست کردن انجومانامه:",
-       "youremail": "أنجومانامە:",
+       "prefs-reset-intro": "شما مؽ تونؽت اؽ بٱلگٱ ناْ سی د نۊ زنٱ کردن تٱرجیهؽا خوت ڤ شکل تیارگٱ پیش فٱرز ڤ کار بونؽت.\nیٱ ورگٱشت پٱزیر نؽ.",
+       "prefs-emailconfirm-label": "تٱیید کردن ٱنجومانامٱ:",
+       "youremail": "ٱنجومانامٱ:",
        "username": "{{GENDER:$1|نوم کاریاری}}:",
-       "prefs-memberingroups": "{{GENDER:$2|أندوم}}  {{PLURAL:$1|دأسە|دأسە یا}}:",
+       "prefs-memberingroups": "{{GENDER:$2|ٱندوم}}  {{PLURAL:$1|دٱسٱ|دٱسٱیا}}:",
        "prefs-memberingroups-type": "$1",
-       "prefs-registration": "گات Ø«Ù\88ت Ù\86ام:",
+       "prefs-registration": "گات Ø³Ù±Ø¨Øª Ù\86Ù\88م:",
        "prefs-registration-date-time": "$1",
-       "yourrealname": "نوم راستكی:",
-       "yourlanguage": "زوٙن:",
+       "yourrealname": "نوم راسی:",
+       "yourlanguage": "زڤوݩ:",
        "yourvariant": "مینونٱ آلشتگٱر زڤوݩ:",
-       "prefs-help-variant": "قسٱ ڤری اْنتخاویی شما سی نمایش مینونٱ بٱلگٱیا د اؽ ڤیکی.",
-       "yournick": "اÙ\85ضا ØªØ§Ø²Ù\87:",
-       "prefs-help-signature": "ویر و باوریا نیسسه بیه د بلگه چک چنه باید وا«<nowiki>~~~~</nowiki>» امضا بان؛ ای نشون وه شکل خودانجومی وه امضا شما و مؤر ویرگار تبدیل بوئه.",
-       "badsig": "ئÙ\85ضا Ø®Ù\88Ù\85 Ø¨Û\8c Ø¦ØªØ¦Ú¤Ø§Ø±.\nسأردÛ\8cسÛ\8cا Ø¦Ú\86 ØªÛ\8c Ø¦Ù\85 Ø¦Ù\84 Ù\86Û\95 Ú¤Ø§Ø±Ø¦Ø³Û\8c Ø¨Ø£Ú©Û\8cت.",
-       "badsiglength": "اÙ\85ضا Ø´Ù\85ا Ù\81رÙ\87 Ú¯Ù¾Ù\87.\nدرازا Ø§Ù\85ضا Ø¨Ø§Û\8cد Ú©Ù\85تر  Ø¯ $1 {{PLURAL:$1|Ù\86Û\8cسÙ\87}} Ø¨Ù\88ئÙ\87.",
-       "yourgender": "شما بیشتر میهایت که چه جوری گوته بوئه؟",
-       "gender-unknown": "د گاتی کئ شوما میائیت ڤا ڤیرئموٙ، نأرم أفزار دوبیشتأر  کألمیە یا جئنس خومثا نە ڤئ کار مئیرە",
-       "gender-male": "وه(پیا) بلگه یا ویکی نه ویرایشت می که",
-       "gender-female": "وه(زئنه)بلگه یا ویکی نه ویرایشت می که",
-       "prefs-help-gender": "اÙ\86جÙ\88Ù\85 Ø¯Ø¦Ù\86 Ø§Û\8c Ù\85Û\8cزÙ\88Ù\86کارÛ\8c Ø¯Ù\84 Ø¨Ù\87 Ù\87اÛ\8cÛ\8cÙ\87.\nÙ\86رÙ\85 Ø§Ù\81زار Ø³Û\8c Ù\87شارÙ\87 Ø¯Ø±Ø³Øª Ù\88Ù\87 Ø¬Ù\86سÛ\8cت Ù\88 Ú¯Ù\88تÙ\86 Ø´Ù\85ا Ø³Û\8c Ú©Ø³Ù\88Ù\86ا ØªØ± Ø¯ Ø´Ú©Ù\84 Ø¯Ø±Ø³Øª Ù\88Ù\87 Ú©Ø§Ø± Ø¨Ø³Ù\86 Û\8cÙ\87 Ú¯Ù\84 Ø¯Ø³ØªÙ\88ر Ø²Ù\88Ù\86 Û\8cÙ\87 Ù\86Ù\87 Ù\88Ù\87 Ú©Ø§Ø± Ù\85Û\8c Ø¨Ù\86Ù\87.\nاÛ\8c Ø¯Ù\88Ù\86سÙ\85Ù\86Û\8cا Ø³Û\8c Ú©Ù\84 Ø®Ù\84Ú© Ø¯Û\8cارÙ\86.",
-       "email": "أنجومانامە",
-       "prefs-help-realname": "نوم راستکی دل به حائه.\nار شما ونه وارد بکیت د گات واگردونی وه آرنگیاتو و نسوت دئن ونو وه خوتو نوم راستکی وه کار گرته بوئه.",
-       "prefs-help-email": "تیرنشون ایمیل دل بهاییه.اما سی وایافت رازینه گواردن دش میهایتش.شما باید رازینه گواردنتونه د ویر بوریت",
-       "prefs-help-email-others": "شما می تونید سی پیوند گرتن تو وا نهایین انجومانامه مین یه هوم پیوند د بلگه کاریاری یا بلگه چک چنه تو انتخاو بکید.تیرنشون انجومانامه تونه د گاتی که کاریاریا هنی وا تو پیوند می گرن دیار نی.",
-       "prefs-help-email-required": "تیرنشون انجومانامه واس با.",
-       "prefs-info": "دÙ\88Ù\86سÙ\85Ù\86Û\8cا Ø§Ù\88Ù\84Û\8cÙ\87",
-       "prefs-i18n": "جهون ولاتمنی",
-       "prefs-signature": "ئÙ\85ضا",
-       "prefs-dateformat": "شلک گات",
-       "prefs-timeoffset": "جا وه جایی گاتی",
-       "prefs-advancedediting": "گزینه یا خلکمنی",
-       "prefs-editor": "ويرايشتگر",
-       "prefs-preview": "Ù¾Û\8cØ´ Ø³Ø¦Û\8cÙ\84",
-       "prefs-advancedrc": "گزینه یا پیشکرده",
-       "prefs-advancedrendering": "گزینه یا پیشکرده",
-       "prefs-advancedsearchoptions": "گزینه یا پیشکرده",
-       "prefs-advancedwatchlist": "گزینه یا پیشکرده",
-       "prefs-displayrc": "گزینه یا نه نشو بیه",
-       "prefs-displaywatchlist": "گزینه یا نه نشو بیه",
-       "prefs-tokenwatchlist": "نشو",
-       "prefs-diffs": "فرخیا",
-       "prefs-help-prefershttps": "کارگرایی ای ترجیح نها وامین اومائن نهایی شما وه کار گرته بوئه.",
-       "prefswarning-warning": "آلشتیا شما ری ترجیحاتتو هنی اماییه نبیه.\nار ای بلگه نه بی یه که ری \"$1\" بپورنیت ول بکیت ترجیحیا شما اماییه نبوئن.",
-       "prefs-tabs-navigation-hint": "نکته: شما می تونید د کلیتیا لادیار کن چپ و راست نه سی رئتن مین تبیا که هان د نوم گه تبیا وه کار بونیت.",
+       "prefs-help-variant": "قسٱ ڤری اْنتخابی شما سی نمایش مؽنونٱ بٱلگٱیا د اؽ ڤیکی.",
+       "yournick": "اÙ\92Ù\85زا ØªØ§Ø²Ù±:",
+       "prefs-help-signature": "ڤویر ۉ باڤٱرؽا نیسسٱ بیٱ د بٱلگٱ چٱک چنٱ بایٱد ڤا«<nowiki>~~~~</nowiki>» اْمزا بۊئٱن؛ اؽ نشوݩ ڤ شکل خودٱنجومؽ ڤ اْمزا شما ۉ مۉئر ڤیرگار تٱبدیلٛ مۊئٱ.",
+       "badsig": "اÙ\92Ù\85زا Ø®Ù\88Ù\85 Ø¨Ø½ Ø§Ù\92تبار.\nسٱردÛ\8cسؽا Ø§Ù\92Ú\86 ØªÛ\8c Ø§Ù\92Ù\85 Ø§Ù\92Ù\84 Ù\86اÙ\92 Ú¤Ø§Ø±Ø³Û\8c Ø¨Ù±Ú©Ø½ت.",
+       "badsiglength": "اÙ\92Ù\85زا Ø´Ù\85ا Ù\81رٱ Ú¯Ù±Ù¾Ù±.\nدرازا Ø§Ù\92Ù\85زا Ø¨Ø§Û\8cٱد Ú©Ù±Ù\85تر Ø¯ $1 {{PLURAL:$1|Ù\86Û\8cسسٱ}} Ø¨Û\8aئٱ.",
+       "yourgender": "شما بؽشتر مؽهایت کاْ چاْ جۊری گوتٱ بۊئٱ؟",
+       "gender-unknown": "د گاتؽ کاْ شما مؽهایؽت ڤ ڤیرمو، نٱرم ٱفزار دۏبؽشتر کلٱمٱیا جنس خونسا ناْ ڤ کار میرٱ",
+       "gender-male": "ڤٱ (پؽا) بٱلگٱیا ڤیکی ناْ ڤیرایش مؽکٱ",
+       "gender-female": "ڤٱ (زٱن) بٱلگٱیا ڤیکی ناْ ڤیرایش مؽکٱ",
+       "prefs-help-gender": "Ù±Ù\86جÙ\88Ù\85 Ø¯Ø§Ù\9bئÙ\86 Ø½ Ù\85Û\8cزÙ\88Ù\86کارÛ\8c Ø¯Ù\84 Ù\87اÛ\8cÛ\8c Ø¦Ù±.\nÙ\86ٱرÙ\85 Ù±Ù\81زار Ø³Û\8c Ù\87شارٱ Ø¯Û\8fرس Ú¤ Ø¬Ù\86سÛ\8cٱت Û\89 Ú¯Ù\88تÙ\86 Ø´Ù\85ا Ø³Û\8c Ú©Ø³Ù\88Ù\86ؽا ØªØ± Ú¤ Ø´Ú©Ù\84 Ø¯Û\8fرس Ú¤ Ú©Ø§Ø±Ú¯Ø±ØªÙ\86 Û\8cاÙ\9b Ø¯Ù±Ø³ØªÛ\8aر Ø²Ú¤Ù\88Ý© Û\8cÙ± Ù\86اÙ\92 Ú¤ Ú©Ø§Ø± Ù\85اÙ\9bÛ\8cرٱ.\nاؽ Ø¯Ù\88Ù\86سÙ\85Ù±Ù\86Û\8cا Ø³Û\8c Ú©Ù\88Ù\84Ù\9b Ù\85ٱردÙ\85 Ø¯Ø½Ø§Ø±Ù±Ù\86.",
+       "email": "ٱنجومانامٱ",
+       "prefs-help-realname": "نوم راسی دل هایی ئٱ.\nٱر شما ڤٱ ناْ ڤارد بٱکؽت د گات ڤاگٱردونی ڤ آرنڳؽا تو نسبٱت داٛئن ڤنو ڤ خوتو نوم راسی ڤ کار گرتٱ بۊئٱ.",
+       "prefs-help-email": "تیرنشوݩ ایماٛیلٛ دل هایی ئٱ.ڤلی سی ڤایافت رازینٱ گوئارسن مؽهایتش ٱر شما رازینٱ گوئارسن تو ناْ دڤیر بۉرؽت.",
+       "prefs-help-email-others": "شما مؽ تونؽت سی پاٛڤٱن گرتن تو ڤا نؽاین ٱنجومانامٱ مؽن یاٛ هوم پاٛڤٱن د بٱلگٱ کاریاری یا بٱلگٱ چٱک چنٱ تو اْنتخاب بٱکؽت. تیرنشوݩ ٱنجومانامٱ تو ناْ د گاتؽ کاْ کاریارؽا هنی ڤا تو پاٛڤٱن ماٛیرٱن دؽار نؽ.",
+       "prefs-help-email-required": "تیرنشوݩ ٱنجومانامٱ اْجباری ئٱ.",
+       "prefs-info": "دÙ\88Ù\86سÙ\85Ù±Ù\86Û\8cا Ù±Ú¤Ù\84Û\8cÙ±",
+       "prefs-i18n": "مؽن زایاراٛیی کردن",
+       "prefs-signature": "اÙ\92Ù\85زا",
+       "prefs-dateformat": "قالب ڤیرگار",
+       "prefs-timeoffset": "جا ڤ جایی زمونی",
+       "prefs-advancedediting": "گوزینٱیا عومۊمی",
+       "prefs-editor": "ڤیرایشگٱر",
+       "prefs-preview": "Ù¾Û\8cØ´ Ø³Ø§Ù\9bÙ\84Ù\9b",
+       "prefs-advancedrc": "گوزینٱیا پیشکردٱ",
+       "prefs-advancedrendering": "گوزینٱیا پیشکردٱ",
+       "prefs-advancedsearchoptions": "گوزینٱیا پیشکردٱ",
+       "prefs-advancedwatchlist": "گوزینٱیا پیشکردٱ",
+       "prefs-displayrc": "گوزینٱیا نمایش",
+       "prefs-displaywatchlist": "گوزینٱیا نمایش",
+       "prefs-tokenwatchlist": "نشوݩ",
+       "prefs-diffs": "فٱرخؽا",
+       "prefs-help-prefershttps": "کارگرایی اؽ تٱرجی نها ڤامؽن اوماین نهایی شما ڤ کار گرتٱ مۊئٱ.",
+       "prefswarning-warning": "آلشتؽا شما ری تٱرجیهات تو هنی آمادٱ ناٛییٱ.\nٱر اؽ بٱلگٱ ناْ بؽ یٱ کاْ ری \"$1\" بٱپۊرنؽت ڤلٛ بٱکؽت تٱرجیهؽا شما آمادٱ نمۊئٱن.",
+       "prefs-tabs-navigation-hint": "نۏکتٱ: شما مؽ تونؽت د کلٛیلٛؽا لٛادؽار کن چٱپ ۉ راس ناْ سی جیلٛ باٛن تبؽا کاْ هان د نومگٱ تبؽا ڤ کار بڤٱنؽت.",
        "userrights": "حوقوٙق دیڤوٙنداری کاریار",
        "userrights-lookup-user": "جأرغە یا کاریاری نە دیڤوٙنداری بأکیت",
-       "userrights-user-editname": "Û\8cئ Ú¯Ø¦Ù\84 Ù\86Ù\88Ù\85 Ú©Ø§Ø±Û\8cارÛ\8c Ú¤Ø§Ø±Ø¦Ø¯ Ø¨Ø£Ú©Û\8cت:",
+       "userrights-user-editname": "Û\8cاÙ\9b Ù\86Ù\88Ù\85 Ú©Ø§Ø±Û\8cارÛ\8c Ú¤Ø§Ø±Ø¯ Ø¨Ù±Ú©Ø½ت:",
        "editusergroup": "ڤیرایئشت جأرغە یا کاریاری",
        "editinguser": "آلئشت دأئن حوقوٙق کاریاری کاریار '''[[کاریار:$1|$1]]''' $2",
        "userrights-editusergroup": "ڤیرایئشت جأرغە یا کاریاری",
        "saveusergroups": "ئمایە کئردئن جأرغە یا کاریاری",
-       "userrights-groupsmember": "أندوم:",
-       "userrights-groupsmember-auto": "أندوم نادیار:",
+       "userrights-groupsmember": "ٱندوم:",
+       "userrights-groupsmember-auto": "ٱندوم نادؽار:",
        "userrights-groupsmember-type": "$1",
        "userrights-groups-help": "شما می تونیت دسه یای که ای کاریار ها دشو آلشت بئیتو:\n* جعوه نشودار وه ای مئنیه که کاریار ها د او دسه.\n* جعوه بی نشون وه ای مئنیه که کاریار د او دسه نئ.\n* نشون* د ای مئنیه که ار شما او دسه نه اضاف بکیتو د نهاتر نموئه ؤردارینش یا برعسگش.",
-       "userrights-reason": "دألیل:",
-       "userrights-no-interwiki": "شما سی ویرایشت حقوق کاریار د ویکی یا هنی دسرسی ناریت.",
-       "userrights-nodatabase": "پاگا دونسمنی $1 یا نیئش یا د ولاتنشین نئ.",
-       "userrights-changeable-col": "گرویایی که شما تونیت ویرایشت بکید",
-       "userrights-unchangeable-col": "گرویایی که شما نتونیت ویرایشت بکید",
+       "userrights-reason": "دلٛیلٛ:",
+       "userrights-no-interwiki": "شما سی ڤیرایش هقۊق کاریار ڤ ڤیکی یا هنی دٱسرسی نارؽت.",
+       "userrights-nodatabase": "پایگا دونسمٱنی $1 یا نؽسش یا د ڤلاتنشیݩ نؽ.",
+       "userrights-changeable-col": "گرۊیایؽ کاْ شما مؽ تونؽت ڤیرایش بٱکؽت",
+       "userrights-unchangeable-col": "گرۊیایؽ کاْ شما نمؽ تونؽت ڤیرایش بٱکؽت",
        "userrights-irreversible-marker": "$1*",
-       "userrights-conflict": "تعارض دسرسیا کاریاری! لطف بکیت یه گل وارسی انجوم بئیت و آلشتانه پشت راس بکیت.",
-       "group": "جأرغە",
-       "group-user": "کاریاریا",
-       "group-autoconfirmed": "کاریاریا خود پوشت راس بییە",
+       "userrights-conflict": "تعاروز دٱسرسیا کاریاری! لوتف بٱکؽت یاٛ ڤارسی ٱنجوم باٛیؽتۉ آلشتؽا ناْ تٱیید بٱکؽت.",
+       "group": "جٱرغٱ:",
+       "group-user": "کاریارؽا",
+       "group-autoconfirmed": "کاریارؽا تٱیید بیٱ خودکار",
        "group-bot": "بوتؽا",
        "group-sysop": "سٱردیڤونکارؽا",
        "group-bureaucrat": "بوروکراتیا",
        "group-suppress": "تیە پایا",
-       "group-all": "(هأمە)",
+       "group-all": "(هٱمٱ)",
        "group-user-member": "{{GENDER:$1|کاریار}}",
-       "group-autoconfirmed-member": "{{GENDER:$1|کاریار خودأنجومکار}}",
+       "group-autoconfirmed-member": "{{GENDER:$1|کاریار خودٱنجومکار}}",
        "group-bot-member": "{{GENDER:$1|بوت}}",
-       "group-sysop-member": "{{GENDER:$1|دÛ\8cÚ¤Ù\88Ù\99Ù\86دار}}",
+       "group-sysop-member": "{{GENDER:$1|دیڤوندار}}",
        "group-bureaucrat-member": "{{GENDER:$1|بوروکرات}}",
-       "group-suppress-member": "{{GENDER:$1|تیە پا}}",
-       "grouppage-user": "{{ns:project}}:کاریاریا",
-       "grouppage-autoconfirmed": "{{ns:project}}:کاریار خودأنجومکار",
-       "grouppage-bot": "{{ns:project}}:ربات‌ها",
+       "group-suppress-member": "{{GENDER:$1|تیٱ پا}}",
+       "grouppage-user": "{{ns:project}}:کاریارؽا",
+       "grouppage-autoconfirmed": "{{ns:project}}:کاریار خودٱنجومکار",
+       "grouppage-bot": "{{ns:project}}:روباتؽا",
        "grouppage-sysop": "{{ns:project}}:دیڤوندارؽا",
-       "grouppage-bureaucrat": "{{ns:project}}:دÛ\8cÚ¤Ù\88Ù\99Ù\86دارÛ\8cا",
-       "grouppage-suppress": "{{ns:project}}:تیە پا",
-       "right-read": "حأنئن بألگە یا",
+       "grouppage-bureaucrat": "{{ns:project}}:دÛ\8cÚ¤Ù\88Ù\86دارؽا",
+       "grouppage-suppress": "{{ns:project}}:تیٱ پا",
+       "right-read": "ڤٱنن بٱلگٱیا",
        "right-edit": "ڤیرایئشت بألگە یا",
-       "right-createpage": "بÙ\84Ú¯Ù\87 Û\8cا Ù\86Ù\87 Ø±Ø§Ø³ Ø¨Ú©Û\8cت(Ù\88Ù\86Ù\88 Ú©Ù\87 Ø¯Ø´Ù\88 Ø¨Ù\84Ú¯Ù\87 Û\8cا Ú\86Ú© Ú\86Ù\86Ù\87 Ù\86ئ)",
-       "right-createtalk": "بÙ\84Ú¯Ù\87 Û\8cا Ú\86Ú© Ú\86Ù\86Ù\87 Ù\86Ù\87 Ø±Ø§Ø³ Ø¨Ú©Û\8cد",
-       "right-createaccount": "یه گل حساو کاروری تازه راس بکیت",
-       "right-minoredit": "نشودار کردن همه ویرایشتیا چی حیرده",
+       "right-createpage": "بٱÙ\84Ú¯Ù±Û\8cا Ù\86اÙ\92 Ø¯Û\8fرس Ø¨Ù±Ú©Û\8cت(Ú¤Ù\86Ù\88 Ú©Ø§Ù\92 Ø¯Ø´Ù\88 Ø¨Ù±Ù\84Ú¯Ù±Û\8cا Ú\86Ù±Ú© Ú\86Ù\86Ù± Ù\86ؽ)",
+       "right-createtalk": "بٱÙ\84Ú¯Ù±Û\8cا Ú\86Ù±Ú© Ú\86Ù\86Ù± Ù\86اÙ\92 Ø¯Û\8fرس Ø¨Ù±Ú©Ø½Øª",
+       "right-createaccount": "یاٛ هساو کاریاری تازٱ دۏرس بٱکؽت",
+       "right-minoredit": "نشوݩ دار کردن هٱمٱ ڤیرایشؽا ڤ عنڤان هیردٱ",
        "right-move": "بٱلگٱیا ناْ جا ڤ جا کو",
        "right-move-subpages": "بٱلگٱیا ۉ زؽر بٱلگٱیا شوناْ جا ڤ جا کو",
        "right-move-rootuserpages": "بٱلگٱیا ریشاٛیی کاریار ناْ جا ڤ جا کو",
-       "right-move-categorypages": "دسه بلگه یا نه جا وه جا بکیت",
+       "right-move-categorypages": "دٱسٱ بٱلگٱیا ناْ جا ڤ جا بٱکؽت",
        "right-movefile": "جانؽایا ناْ جا ڤ جا کو",
-       "right-suppressredirect": "اوسه که بلگه یا د بین رئتنه هیچ واگردونی سی بلگه یا سرچشمه دروس نبیه",
-       "right-upload": "سوار کردن جانیایا",
-       "right-reupload": "سوارکرد هنی جانیایی که دماتر بئیشه",
-       "right-reupload-own": "سوارکرد هنی جانیایی که د دماتر وه دس همو کاریار سوارکرد بیه.",
-       "right-reupload-shared": "باطÙ\84â\80\8cکردÙ\86 Ù\88Ù\84ات Ù\86Ø´Û\8cÙ\86Û\8c Ø¬Ø§Ù\86Û\8cاÛ\8cا Ù\87Ù\88Ù\85بئر Ø¨Û\8cÙ\87",
-       "right-upload_by_url": "سوار کرد جانیایا د یو آر ال",
+       "right-suppressredirect": "اۊساْ کاْ بٱلگٱیا د باٛن رٱتنٱ هیچ ڤاگٱردونی سی بٱلگٱیا سرچشمٱ دۏرس ناٛییٱ",
+       "right-upload": "سڤار کردن جانیاؽا",
+       "right-reupload": "سڤارکرد هنی جانؽایؽ کاْ نوئاتر بیشونٱ",
+       "right-reupload-own": "سڤارکرد هنی جانؽایؽ کاْ نوئاتر ڤ دٱس هاٛ اؽ کاریار سڤارکرد بیٱ.",
+       "right-reupload-shared": "باتÙ\84â\80\8cکردÙ\86 Ú¤Ù\84ات Ù\86Ø´Û\8cÙ\86Û\8c Ø¬Ø§Ù\86ؽاÛ\8cا Ù\87Ù\88Ù\85بٱر Ø¨Û\8cÙ±",
+       "right-upload_by_url": "سڤارکرد جانؽایا د یۊ آر اْل",
        "right-purge": "پاک کردن مینجاگر بلگه بی یه که بلگه پشت راس کردن دیاری بکه",
-       "right-autoconfirmed": "د محدودیتیا سرعت آی پی-پایه کارگرایی ناره",
-       "right-bot": "باور بیه چی یه گل پردازشت خودانجوم",
-       "right-nominornewtalk": "حیرده ویرایشت بلگه یا چک چنه وه شکلی که باعث گوتن پیغوم تازه نبوئه.",
-       "right-apihighlimits": "سخÙ\85 Ø¨Ø§Ù\84اتر Ø¯ Ù\88Ù\87 Ú©Ø§Ø± Ø¨Ø³ن API",
+       "right-autoconfirmed": "د مٱئدۊدیٱتؽا سۏرعٱت آی پی-پایٱ کارگرایی نارٱ",
+       "right-bot": "باڤٱر بیٱ چی یاٛ پٱردازش خودٱنجوم",
+       "right-nominornewtalk": "هیردٱ ڤیرایش بٱلگٱیا چٱک چنٱ ڤ شکلؽ کاْ باعس گوتن پاٛغوم تازٱ نمۊئٱ.",
+       "right-apihighlimits": "سخÙ\85 Ø±Û\8fتر Ø¯ Ú¤ Ú©Ø§Ø± Ø¨Ù±Ø³Øªن API",
        "right-writeapi": "د نیسٱنن اٛی پی آی ڤ کار باٛیرؽت",
-       "right-delete": "بÙ\84Ú¯Û\8cا Ù\86Ù\87 پاکسا کو",
-       "right-bigdelete": "بÙ\84Ú¯Ù\87 Û\8cاÛ\8cÛ\8c Ú©Ù\87 Ù\88Û\8cرگار Ú¯Ù¾Û\8c Ø¯Ø§Ø±Ù\86 Ù¾Ø§Ú©Ø³Ø§ Ø¨Ú©Û\8cت",
-       "right-deletelogentry": "پاکسا کردن و ناپاکسا کردن داده واریایی ویجه ای د پهرستنومه",
-       "right-deleterevision": "پاکساکردن و ناپاکساکردن وانئریا ویجه ای د بلگه یا",
-       "right-deletedhistory": "دÛ\8cئÙ\86 Ú\86Û\8cا Ù¾Ø§Ú©Ø³Ø§ Ø¨Û\8cÙ\87 Ø¯ Ù\88Û\8cرگارØ\8c Ø¨Û\8c Û\8cÙ\87 Ú©Ù\87 Ù\86Û\8cسسÙ\87 Ù\88Ù\86Ù\88Ù\86Ù\87 Ø¨Ø¤Ù\86Û\8cت.",
-       "right-deletedtext": "دÛ\8cئÙ\86 Ù\86Û\8cسسÙ\87 Ù¾Ø§Ú©Ø³Ø§ Ø¨Û\8cÙ\87 Ù\88 Ø¢Ù\84شتÛ\8cاÛ\8cÛ\8c Ú©Ù\87 Ù\87اÙ\86 Ù\85Û\8cÙ\86جا Ù\88اÙ\86ئرÛ\8cا Ù¾Ø§Ú©Ø³Ø§ Ø¨Û\8cÙ\87",
-       "right-browsearchive": "بÙ\84Ú¯Ù\87 Û\8cا Ù¾Ø§Ú© Ø¨Û\8cÙ\87 Ù\86Ù\87 Ù¾Û\8c Ø¬Ù\88ری کو",
-       "right-undelete": "ای بلگه نه پاکسا نكيد",
-       "right-suppressrevision": "دیئن و زنه کردن وانئریایی بلگه یایی که د دس کاریاریا نهو بینه",
-       "right-viewsuppressed": "دیئن وانئریایی که د تیه هر کاریاری قام بیه",
-       "right-suppressionlog": "دÛ\8cئÙ\86 Ù¾Ù\87رستÙ\86Ù\88Ù\85Ù\87 Û\8cا Ø®ØµÙ\88صی",
-       "right-block": "کاریاریا هنی د ویرایشت منع بوئن",
-       "right-blockemail": "کاریار نه د کل کردن انجومانامه منع کو",
-       "right-hideuser": "قلف کردن یه گل نوم کاریاری،قام کردن وه د ور تیه کل خلک",
-       "right-ipblock-exempt": "کارگرایی نگرتن د قطع دسرسیا آی پی، خودانجوم یا فاصله دار",
-       "right-unblockself": "خوشه قلف نکید",
-       "right-protect": "Ø¢Ù\84شت Ø¯Ø¦Ù\86 Ø§Ù\86ازÙ\87 Ù¾Ø± Ù\88 Ù¾Û\8cÙ\85 Ú©Ø±Ø¯Ù\86 Ø¨Ù\84Ú¯Ù\87 Û\8cا Ù\88 Ù\88Û\8cراÛ\8cشت Ø¨Ù\84Ú¯Ù\87 Û\8cا Ù¾Ø± Ù\88 Ù¾Û\8cÙ\85 Ø¨Û\8cÙ\87 تافی",
-       "right-editprotected": "ویرایشت بلگه یا پر و پیم بیه چی «{{int:protect-level-sysop}}»",
-       "right-editsemiprotected": "ویرایشت بلگه یا پر و پیم بیه چی «{{int:protect-level-autoconfirmed}}»",
-       "right-editcontentmodel": "ویرایشت مدل مینونه یه گل بلگه",
-       "right-editinterface": "راوط کاریار نه ویرایشت کو",
-       "right-editusercss": "جانیایا سی اس اس کاریاریا هنی نه ویرایشت کو",
-       "right-edituserjs": "جانیایا جاوا اسکریپت کاریاریا هنی نه ویرایشت کو",
-       "right-editmyusercss": "جانیایا سی اس اس کاریار خوتو نه ویرایشت کو",
-       "right-editmyuserjs": "جانیایا جاوا اسکریپت کاریار خوتو نه ویرایشت کو",
-       "right-viewmywatchlist": "سیل برگ خوتونه بوینیت",
-       "right-editmywatchlist": "سیل برگ خوتونه ویرایشت بکیت. د ویرتو با که پاره ای د انجومکاریا بی دسرسی ئم می تونن ای بلگه یا نه اضافه بکن.",
-       "right-viewmyprivateinfo": "دÙ\88Ù\86سÙ\85Ù\86Û\8cا Ø´ØµÙ\82Û\8c Ø®Ù\88تÙ\88Ù\86Ù\87 Ø¨Ù\88Û\8cÙ\86Û\8cت(Ú\86Û\8c ØªÛ\8cرÙ\86Ø´Ù\88Ù\86 Ø§Ù\86جÙ\88Ù\85اÙ\86اÙ\85Ù\87Ø\8cÙ\86Ù\88Ù\85 Ø±Ø§Ø³ØªÚ©ی)",
-       "right-editmyprivateinfo": "دÙ\88Ù\86سÙ\85Ù\86Û\8cا Ø´ØµÙ\82Û\8c Ø®Ù\88تÙ\88Ù\86Ù\87 Ù\88Û\8cراÛ\8cشت Ø¨Ú©Û\8cد(Ú\86Û\8c ØªÛ\8cرÙ\86Ø´Ù\88Ù\86 Ø§Ù\86جÙ\88Ù\85اÙ\86اÙ\85Ù\87Ø\8cÙ\86Ù\88Ù\85 Ø±Ø§Ø³ØªÚ©ی)",
-       "right-editmyoptions": "اÙ\88Ù\84Ù\88Û\8cتÛ\8cا ØªÙ\88Ù\86Ù\87 Ù\88Û\8cراÛ\8cشت Ø¨Ú©Û\8cت",
-       "right-rollback": "Ú\86Ù\88اشÙ\87 Ú©Ø±Ø¯Ù\86 Ø³Ø±Û\8cع Ù\88Û\8cراÛ\8cشتÛ\8cا Ø¢Ø®Ø±Û\8c Ú©Ø§Ø±Û\8cارÛ\8c Ú©Ù\87 Û\8cÙ\87 Ø¨Ù\84Ú¯Ù\87 Ù\88Û\8cجÙ\87 Ù\86Ù\87 Ù\88Û\8cراÛ\8cشت Ø¯Ø¦Ù\87",
-       "right-markbotedits": "نشودار کردن ویرایشتیا چواشه بیه چی ویرایشتیا یه گل بات",
-       "right-noratelimit": "کارگرا Ù\86بÛ\8cئÙ\86 Ø¯ Ù\85حدÙ\88دÛ\8cت Ø³Ø±Ø¹ت",
-       "right-import": "بÙ\84Ú¯Ù\87 Û\8cا Ù\86Ù\87 Ø¯ Ù\88Û\8cÚ©Û\8c Ù\87Ù\86Û\8c Ù\88ارد Ø¨Ú©Û\8cد",
-       "right-importupload": "دئÙ\86 Ø¨Ù\84Ú¯Ù\87 Û\8cا Ø¯ Û\8cÙ\87 Ú¯Ù\84 Ø¬Ø§Ù\86Û\8cا Ø³Ù\88ار Ø¨Û\8cÙ\87",
-       "right-patrol": "سردیاری کردن د ویرایشتیا کسونا تر",
-       "right-autopatrol": "سردیاری کردن خودانجوم د ویرایشتیا خوش",
-       "right-patrolmarks": "دیئن سردیس سردیاری کردن د آلشتیا ایسنی",
-       "right-unwatchedpages": "دیئن نوم گه بلگه یا دیئه نبیه",
-       "right-mergehistory": "وه یک شیوسن ویرگار ای بلگه",
-       "right-userrights": "حقوق همه کاریاریانه ویرایشت بکید",
-       "right-userrights-interwiki": "حقوق همه کاریاریانه د ویکی یا هنی ویرایشت بکید",
-       "right-siteadmin": "پاگا Ø¯Ù\88Ù\86سÙ\85Ù\86Û\8c Ù\86Ù\87 Ù\82Ù\84Ù\81 Ø¨Ú©Û\8cد Û\8cا Ù\86Ú©Û\8cد",
-       "right-override-export-depth": "وه در دئن بلگه یایی که بلگه یا هوم پیوند بیه تا پی یا 5 ها دشو",
-       "right-sendemail": "سی کاریاریا هنی انجومانامه کل بکید",
+       "right-delete": "بٱÙ\84Ú¯Ù±Û\8cا Ù\86اÙ\92 پاکسا کو",
+       "right-bigdelete": "بٱÙ\84Ú¯Ù±Û\8cاÛ\8cÛ\8c Ú©Ø§Ù\92 Ú¤Û\8cرگار Ú¯Ù±Ù¾Ø½ Ø¯Ø§Ø±Ù±Ù\86 Ù¾Ø§Ú©Ø³Ø§ Ø¨Ù±Ú©Ø½ت",
+       "right-deletelogentry": "پاکسا کردن ۉ ناپاکسا کردن دادٱ ڤارؽا ڤیژاٛیؽ د پهرستنومٱ",
+       "right-deleterevision": "پاکساکردن ۉ ناپاکساکردن ڤانریا ڤیژاٛیؽ د بٱلگٱیا",
+       "right-deletedhistory": "دÛ\8cئÙ\86 Ú\86Û\8cا Ù¾Ø§Ú©Ø³Ø§ Ø¨Û\8cÙ± Ø¯ Ú¤Û\8cرگارØ\8c Ø¨Ø½ Û\8cÙ± Ú©Ø§Ù\92 Ù\86Û\8cسسٱ Ú¤Ù\86Ù\88 Ù\86اÙ\92 Ø¨Ù\88Ù\86ؽت",
+       "right-deletedtext": "دÛ\8cئÙ\86 Ù\86Û\8cسسٱ Ù¾Ø§Ú©Ø³Ø§ Ø¨Û\8cÙ± Û\89 Ø¢Ù\84شتؽاÛ\8cؽ Ú©Ø§Ù\92 Ù\87اÙ\86 Ù\85ؽÙ\86جا Ú¤Ø§Ù\86رÛ\8cا Ù¾Ø§Ú©Ø³Ø§ Ø¨Û\8cÙ±",
+       "right-browsearchive": "بٱÙ\84Ú¯Ù±Û\8cا Ù¾Ø§Ú© Ø¨Û\8cÙ± Ù\86اÙ\92 Ù¾Ø§Ù\9bجÛ\8aری کو",
+       "right-undelete": "اؽ بٱلگٱ ناْ پاکسا نٱکؽت",
+       "right-suppressrevision": "دیئن ۉ زنٱ کردن ڤانریا بٱلگٱیایؽ کاْ د دٱس کاریارؽا نهوݩ بینٱ",
+       "right-viewsuppressed": "دیئن ڤانریایؽ کاْ د تیٱ هٱر کاریارؽ قایم بیٱ",
+       "right-suppressionlog": "دÛ\8cئÙ\86 Ù¾Ù\87رستÙ\86Ù\88Ù\85Ù±Û\8cا Ø®Ù\88سÛ\8aسی",
+       "right-block": "کاریاریا هنی د ڤیرایش مٱنع بۊئٱن",
+       "right-blockemail": "کاریار ناْ د کولٛ کردن ٱنجومانامٱ مٱنع کو",
+       "right-hideuser": "قلف کردن یاٛ نوم کاریاری، قایم کردن ڤٱ د ڤٱر تیٱ کولٛ خٱلک",
+       "right-ipblock-exempt": "کارگرایی نٱگرتن د قٱت دٱسرسیا آی پی، خودٱنجوم یا فاسلٱ دار",
+       "right-unblockself": "ڤاز کردن دسرسی خود",
+       "right-protect": "Ø¢Ù\84شت Ø¯Ø§Ù\9bئÙ\86 Ù±Ù\86دازٱ Ù¾Ø± Û\89 Ù¾Û\8cÙ\85 Ú©Ø±Ø¯Ù\86 Ø¨Ù±Ù\84Ú¯Ù±Û\8cا Û\89 Ú¤Û\8cراÛ\8cØ´ Ø¨Ù±Ù\84Ú¯Ù±Û\8cا Ù¾Ø± Û\89 Ù¾Û\8cÙ\85 Ø¨Û\8cÙ± تافی",
+       "right-editprotected": "ڤیرایش بٱلگٱیا پر ۉ پیم بیٱ چی «{{int:protect-level-sysop}}",
+       "right-editsemiprotected": "ڤیرایش بٱلگٱیا پر ۉ پیم بیٱ چی «{{int:protect-level-autoconfirmed}}»",
+       "right-editcontentmodel": "ڤیرایش مودل مؽنونٱ یاٛ بٱلگٱ",
+       "right-editinterface": "رابت کاریار ناْ ڤیرایش کو",
+       "right-editusercss": "جانیایا سی اْس اْس کاریارؽا هنی ناْ ڤیرایش کو",
+       "right-edituserjs": "جانؽایا جاڤا اْسکریپت کاریارؽا هنی ناْ ڤیرایش کو",
+       "right-editmyusercss": "جانؽایا سی اْس اْس کاریار خوتو ناْ ڤیرایش کو",
+       "right-editmyuserjs": "جانؽایا جاڤا اْسکریپت کاریار خوتو ناْ ڤیرایش کو",
+       "right-viewmywatchlist": "ساٛلٛ بٱرگ خوتو ناْ باٛینؽت",
+       "right-editmywatchlist": "ساٛلٛ بٱرگ خوت ناْ ڤیرایش بٱکؽت. د ڤیرتو با کاْ د ٱنجومکاریا بی دٱسرسی هٱم مؽ تونٱن ای بٱلگٱیا ناْ اْزافٱ کو.",
+       "right-viewmyprivateinfo": "دÙ\88Ù\86سÙ\85Ù±Ù\86Û\8cا Ø´Ù±Ø®Ø³Û\8c Ø®Ù\88تÙ\88 Ù\86اÙ\92 Ø¨Ø§Ù\9bÛ\8cÙ\86ؽت (Ú\86Û\8c ØªÛ\8cرÙ\86Ø´Ý© Ù±Ù\86جÙ\88Ù\85اÙ\86اÙ\85Ù±Ø\8c Ù\86Ù\88Ù\85 Ø±Ø§Ø³ی)",
+       "right-editmyprivateinfo": "دÙ\88Ù\86سÙ\85Ù±Ù\86Û\8cا Ø´Ù±Ø®Ø³Û\8c Ø®Ù\88تÙ\88 Ù\86اÙ\92 Ú¤Û\8cراÛ\8cØ´ Ø¨Ù±Ú©Ø½Øª (Ú\86Û\8c ØªÛ\8cرÙ\86Ø´Ù\88Ý© Ù±Ù\86جÙ\88Ù\85اÙ\86اÙ\85Ù±Ø\8c Ù\86Ù\88Ù\85 Ø±Ø§Ø³ی)",
+       "right-editmyoptions": "اÙ\88Ù\84Ù±Ú¤Û\8cٱتؽا ØªÙ\88 Ù\86اÙ\92 Ú¤Û\8cراÛ\8cØ´ Ø¨Ù±Ú©Ø½ت",
+       "right-rollback": "Ú\86Ù\88ئارشٱ Ú©Ø±Ø¯Ù\86 Ø³Ø±Û\8cع Ú¤Û\8cراÛ\8cشؽا Ø¢Ø®Ø±Û\8c Ú©Ø§Ø±Û\8cارؽ Ú©Ø§Ù\92 Û\8cاÙ\9b Ø¨Ù±Ù\84Ú¯Ù± Ú¤Û\8cÚ\98Ù± Ù\86اÙ\92 Ú¤Û\8cراÛ\8cØ´ Ú©Ø±Ø¯Ù±",
+       "right-markbotedits": "نشوݩ دار کردن ڤیرایشؽا چوئاشٱ بیٱ چی ڤیرایشؽا یاٛ روبات",
+       "right-noratelimit": "کارگرا Ù\86اÙ\9bÛ\8cئÙ\86 Ø¯ Ù\85ٱئدÛ\8aدÛ\8cٱت Ø³Û\8fرعٱت",
+       "right-import": "بٱÙ\84Ú¯Ù±Û\8cا Ù\86اÙ\92 Ø¯ Ú¤Û\8cÚ©Û\8c Ù\87Ù\86Û\8c Ú¤Ø§Ø±Ø¯ Ø¨Ù±Ú©Ø½Øª",
+       "right-importupload": "داÙ\9bئÙ\86 Ø¨Ù±Ù\84Ú¯Ù±Û\8cا Ø¯ Û\8cاÙ\9b Ø¬Ø§Ù\86Û\8cا Ø³Ú¤Ø§Ø± Ø¨Û\8cÙ±",
+       "right-patrol": "سٱردؽاری کردن د ڤیرایشؽا کٱسونؽاْ تر",
+       "right-autopatrol": "سٱردؽاری کردن خودٱنجوم د ڤیرایشؽا خوش",
+       "right-patrolmarks": "دیئن سٱردیس سٱردؽاری کردن د آلشتؽا ایسنی",
+       "right-unwatchedpages": "دیئن نومگٱ بٱلگٱیا دییٱ ناٛیٱ",
+       "right-mergehistory": "ڤ یٱک شؽڤسن ڤیرگار اؽ بٱلگٱ",
+       "right-userrights": "هقوق هٱمٱ کاریارؽا ناْ ڤیرایش بٱکؽت",
+       "right-userrights-interwiki": "هقوق هٱمٱ کاریارؽا ناْ د ڤیکی یا هنی ڤیرایش بٱکؽت",
+       "right-siteadmin": "Ù\82Ù\84Ù\81 Ú©Ø±Ø¯Ù\86 Û\89 Ú¤Ø§Ø² Ú©Ø±Ø¯Ù\86 Ù¾Ø§Û\8cگا Ø¯Ù\88Ù\86سÙ\85Ù±Ù\86Û\8c",
+       "right-override-export-depth": "ڤ دٱر داٛئن بٱگٱیایؽ کاْ بٱلگٱیا هوم پاٛڤٱن بیٱ تا قیلٛی 5 ها دشو",
+       "right-sendemail": "سی کاریارؽا هنی ٱنجومانامٱ کلٛ بٱکؽت",
        "right-managechangetags": "راس کئردئن [[Special:سأردیسیا|سأردیسیا]] پاکسا کئردئن د رئسینە جا",
        "right-applychangetags": "ڤئ کار گئرئتئنئ [[Special:سأردیسیا|سأردیسیا]] ڤاگئرد آلئشتیا ھأرکومئشوٙ.",
        "right-changetags": "Add and remove arbitrary [[Special:Tags|tags]] on individual revisions and log entries",
        "newuserlogpage": "دۏرس بیٱ ڤا کاریار",
-       "newuserlogpagetext": "Û\8cÙ\87 Ù¾Ù\87رستÙ\86Ù\88Ù\85Ù\87 Ø±Ø§Ø³ Ø¨Û\8cئÙ\86 Ú©Ø§Ø±Û\8cارÙ\87",
-       "rightslog": "Ù¾Ù\87رستÙ\86Ù\88Ù\85Ù± Ù\87Ù\88Ù\82Û\8aÙ\82 Ú©Ø§Ø±Û\8cار",
-       "rightslogtext": "Û\8cÙ\87 Ù¾Ù\87رستÙ\86Ù\88Ù\85Ù\87 Ø¢Ù\84شتÛ\8cا Ø­Ù\82Ù\88Ù\82 Ú©Ø§Ø±Û\8cارÙ\87.",
-       "action-read": "ای بلگه نه بحو",
-       "action-edit": "اؽ بٱلگٱ ناْ ڤيرايش بٱكیت",
+       "newuserlogpagetext": "Û\8cÙ± Ù¾Ù\87رستÙ\86Ù\88Ù\85Ù± Ø¯Û\8fرس Ø¨Û\8cئÙ\86 Ú©Ø§Ø±Û\8cار Ø¦Ù±.",
+       "rightslog": "پهرستنومٱ هقۊق کاریار",
+       "rightslogtext": "Û\8cÙ± Ù¾Ù\87رستÙ\86Ù\88Ù\85Ù± Ø¢Ù\84شتؽا Ù\87Ù\82Ù\88Ù\82 Ú©Ø§Ø±Û\8cار Ø¦Ù±.",
+       "action-read": "ڤٱنن اؽ بٱلگٱ",
+       "action-edit": "اؽ بٱلگٱ ناْ ڤيرايش بٱكؽت",
        "action-createpage": "راس کردن بلگیا",
        "action-createtalk": "بلگه یا چک چنه نه راس بکید",
        "action-createaccount": "هساو اؽ کاریار ناْ دۏرس بٱکؽت",
-       "action-history": "ویرگار ای بلگه نه بوینیت",
-       "action-minoredit": "ای ویرایشت نه چی یه حیرده ویرایشت نشو بیئت",
+       "action-history": "ڤیرگار اؽ بٱلگٱ ناْ باٛینؽت",
+       "action-minoredit": "اؽ ڤیرایش ناْ چی یاٛ هیردٱ ڤیرایش نشوݩ باٛیؽت",
        "action-move": "اؽ بٱلگٱ ناْ جا ڤ جا کو",
-       "action-move-subpages": "اÛ\8c Ø¨Ù\84Ú¯Ù\87 Ù\88 Ø²Û\8cر Ø¨Ù\84Ú¯Ù\87 Û\8cاشÙ\87 Ø¬Ø§ Ù\88Ù\87 Ø¬Ø§ Ø¨Ú©Û\8cد",
-       "action-move-rootuserpages": "بÙ\84Ú¯Ù\87 Û\8cا Ø±Û\8cØ´Ù\87 Ø§Û\8c Ú©Ø§Ø±Û\8cار Ù\86Ù\87 Ø¬Ø§ Ù\88Ù\87 Ø¬Ø§ Ø¨Ú©Û\8cد",
-       "action-move-categorypages": "جا وه جا کردن دسه بلگه یا",
-       "action-movefile": "ای جانیا نه جا وه جا بکید",
-       "action-upload": "ای جانیا نه سوار بکید",
-       "action-reupload": "نیسئین ری جانیا ایسنی",
-       "action-reupload-shared": "باطÙ\84 Ú©Ø±Ø¯Ù\86 Ø§Û\8c Ø¬Ø§Ù\86Û\8cا Ø±Û\8c Û\8cÙ\87 Ú¯Ù\84 Ú¯Ù\86جÛ\8cÙ\86Ù\87 Ù\87Ù\88Ù\85بئر",
-       "action-upload_by_url": "ای جانیا نه د یو آر ال سوار بکید",
-       "action-writeapi": "د Ù\86Û\8cسÙ\86Ù\86 Ø§Û\8c Ù¾Û\8c Ø¢Û\8c Ø§Ø³ØªÙ\81ادÙ\87 Ø¨Ú©Û\8cد",
-       "action-delete": "ای بلگه نه پاکسا کو",
+       "action-move-subpages": "ؽ Ø¨Ù±Ù\84Ú¯Ù± Û\89 Ø²Ø½Ø± Ø¨Ù±Ù\84Ú¯Ù±Û\8cا Ø´Ø§Ù\92 Ø¬Ø§ Ú¤ Ø¬Ø§ Ø¨Ù±Ú©Ø½Øª",
+       "action-move-rootuserpages": "بٱÙ\84Ú¯Ù±Û\8cا Ø±Û\8cشاÙ\9bÛ\8c Ú©Ø§Ø±Û\8cار Ù\86اÙ\92 Ø¬Ø§ Ú¤ Ø¬Ø§ Ø¨Ù±Ú©Ø½Øª",
+       "action-move-categorypages": "جا ڤ جا کردن دٱسٱ بٱلگٱیا",
+       "action-movefile": "اؽ جانؽا ناْ جا ڤ جا بٱکؽت",
+       "action-upload": "اؽ جانؽا ناْ سڤار بٱکؽت",
+       "action-reupload": "نیسٱنن ری جانؽا ایسنی",
+       "action-reupload-shared": "باتÙ\84 Ú©Ø±Ø¯Ù\86 Ø§Ø½ Ø¬Ø§Ù\86ؽا Ø±Û\8c Û\8cاÙ\9b Ú¯Ù±Ù\86جÛ\8cÙ\86Ù± Ù\87Ù\88Ù\85بٱر",
+       "action-upload_by_url": "اؽ جانؽا ناْ د یۊ آر اْل سڤار بٱکؽت",
+       "action-writeapi": "د Ù\86Û\8cسٱÙ\86Ù\86 Ø§Û\8c Ù¾Û\8c Ø¢Û\8c Ø§Ù\92ستÙ\81ادٱ Ø¨Ù±Ú©Ø½Øª",
+       "action-delete": "اؽ بٱلگٱ ناْ پاکسا کو",
        "action-deleterevision": "ای وانئری نه پاک کو",
        "action-deletedhistory": "ویرگار پاکسا بیه ای بلگه نه بوینیت",
-       "action-browsearchive": "بÙ\84Ú¯Ù\87 Û\8cا Ù¾Ø§Ú© Ø¨Û\8cÙ\87 Ù\86Ù\87 Ù¾Û\8c Ø¬Ù\88رÛ\8c Ø¨Ú©Û\8cد",
+       "action-browsearchive": "بٱÙ\84Ú¯Ù±Û\8cا Ù¾Ø§Ú© Ø¨Û\8cÙ± Ù\86اÙ\92 Ù¾Ø§Ù\9bجÛ\8aرÛ\8c Ø¨Ù±Ú©Ø½Øª",
        "action-undelete": "ای بلگه نه پاک نکو",
        "action-suppressrevision": "وانئری و زنه کردن وانئریا پاک بیه",
-       "action-suppressionlog": "ای پهرستنومه خصوصی نه بوینیت",
-       "action-block": "ای کاریار نه د ویرایشت کردن نهاگری کو",
-       "action-protect": "ریترازیا پر و پیم کاری د ای بلگه نه آلشت بکید",
-       "action-rollback": "Ú\86Ù\88اشÙ\87 Ú©Ø±Ø¯Ù\86 Ø³Ø±Û\8cع Ù\88Û\8cراÛ\8cشتÛ\8cا Ø¢Ø®Ø±Û\8c Ú©Ø§Ø±Û\8cارÛ\8c Ú©Ù\87 Û\8cÙ\87 Ø¨Ù\84Ú¯Ù\87 Ù\88Û\8cجÙ\87 Ù\86Ù\87 Ù\88Û\8cراÛ\8cشت Ø¯Ø¦Ù\87",
-       "action-import": "بÙ\84Ú¯Ù\87 Û\8cا Ù\86Ù\87 Ø¯ Ù\88Û\8cÚ©Û\8c Ù\87Ù\86Û\8c Ù\88ارد Ø¨Ú©Û\8cد",
-       "action-importupload": "بÙ\84Ú¯Ù\87 Û\8cا Ù\86Ù\87 Ø¯ Ø¬Ø§Ù\86Û\8cا Ø³Ù\88ار Ø¨Û\8cÙ\87 Ù\88ارد Ø¨Ú©Û\8cد",
-       "action-patrol": "سردیاری کردن د ویرایشتیا کسونا تر",
-       "action-autopatrol": "سردیاری کردن د ویرایشتیا خوتو",
-       "action-unwatchedpages": "دیئن نوم گه بلگه یا دیئه نبیه",
-       "action-mergehistory": "وه یک شیوسن ویرگار ای بلگه",
-       "action-userrights": "حقوق همه کاریاریا نه ویرایشت بکید",
-       "action-userrights-interwiki": "حقوق همه کاریاریانه د ویکی یا هنی ویرایشت بکید",
-       "action-siteadmin": "رسینه جا نه قلف بکید یا نکید",
-       "action-sendemail": "انجومانامه یا نه کل کو",
-       "action-editmywatchlist": "سیل برگ خوتونه ویرایشت بکید",
-       "action-viewmywatchlist": "سیل برگ خوتونه بوینیت",
-       "action-viewmyprivateinfo": "دÙ\88Ù\86سÙ\85Ù\86Û\8cا Ø®Ù\88تÙ\88Ù\86Ù\87 Ø¨Ù\88Û\8cÙ\86Û\8cت",
-       "action-editmyprivateinfo": "دÙ\88Ù\86سÙ\85Ù\86Û\8cا Ø´ØµÙ\82Û\8c Ø®Ù\88تÙ\88Ù\86Ù\87 Ù\88Û\8cراÛ\8cشت Ø¨Ú©Û\8cد",
-       "action-editcontentmodel": "ویرایشت مدل مینونه یه گل بلگه",
+       "action-suppressionlog": "اؽ پهرستنومٱ خسوسی ناْ باٛینؽت",
+       "action-block": "اؽ کاریار ناْ د ڤیرایش کردن نهاگیری کو",
+       "action-protect": "ریترازؽا پر ۉ پیم کاری د اؽ بٱلگٱ ناْ آلشت بٱکؽت",
+       "action-rollback": "Ú\86Ù\88ئارشٱ Ú©Ø±Ø¯Ù\86 Ø³Ø±Û\8cع Ú¤Û\8cراÛ\8cشؽا Ø¢Ø®Ø±Û\8c Ú©Ø§Ø±Û\8cارؽ Ú©Ø§Ù\92 Û\8cاÙ\9b Ø¨Ù±Ù\84Ú¯Ù± Ú¤Û\8cÚ\98Ù± Ù\86اÙ\92 Ú¤Û\8cراÛ\8cØ´ Ø¯Ø§Ù\9bئٱ",
+       "action-import": "بٱÙ\84Ú¯Ù±Û\8cا Ù\86اÙ\92 Ø¯ Ú¤Û\8cÚ©Û\8c Ù\87Ù\86Û\8c Ú¤Ø§Ø±Ø¯ Ø¨Ù±Ú©Ø½Øª",
+       "action-importupload": "بٱÙ\84Ú¯Ù±Û\8cا Ù\86اÙ\92 Ø¯ Ø¬Ø§Ù\86ؽا Ø³Ú¤Ø§Ø± Ø¨Û\8cÙ± Ú¤Ø§Ø±Ø¯ Ø¨Ù±Ú©Ø½Øª",
+       "action-patrol": "سٱردؽاری کردن د ڤیرایشؽا کٱسونؽا تر",
+       "action-autopatrol": "سٱردؽاری کردن د ڤیرایشؽا خوتو",
+       "action-unwatchedpages": "دیئن نومگٱ بٱلگٱیا دیئٱ ناٛیٱ",
+       "action-mergehistory": "ڤ یٱک شؽڤسن ڤیرگار اؽ بٱلگٱ",
+       "action-userrights": "هقوق هٱمٱ کاریارؽا ناْ ڤیرایش بٱکؽت",
+       "action-userrights-interwiki": "هقوق هٱمٱ کاریارؽا ناْ د ڤیکی یا هنی ڤیرایش بٱکؽت",
+       "action-siteadmin": "قلف کردن ۉ ڤاز کردن پایگا دونسمٱنی",
+       "action-sendemail": "ٱنجومانامٱیا ناْ کلٛ- کو",
+       "action-editmywatchlist": "ساٛل بٱرگ خوتو ناْ ڤیرایش بٱکؽت",
+       "action-viewmywatchlist": "ساٛل بٱرگ خوتو ناْ باٛینؽت",
+       "action-viewmyprivateinfo": "دÙ\88Ù\86سÙ\85Ù±Ù\86Û\8cا Ø®Ù\88تÙ\88 Ù\86اÙ\92 Ø¨Ø§Ù\9bÛ\8cÙ\86ؽت",
+       "action-editmyprivateinfo": "دÙ\88Ù\86سÙ\85Ù±Ù\86Û\8cا Ø´Ù±Ø®Ø³Û\8c Ø®Ù\88تÙ\88 Ù\86اÙ\92Ú¤Û\8cراÛ\8cØ´ Ø¨Ù±Ú©Ø½Øª",
+       "action-editcontentmodel": "ڤیرایش مودل میپؽنونٱ یاٛ بٱلگٱ",
        "action-managechangetags": "راس کردن و پاکسا کردن سردیسیا د رسینه جا",
-       "action-applychangetags": "سردیسیا نه واگرد آلشتیایی که خوتو دئیته وه کار بیئریت",
-       "action-changetags": "اضاف کردن یا جا وه جاکاری سردیسیا دل وه حایی د وانئریا و پهرستنومه یا شخصی",
-       "nchanges": "$1 {{PLURAL:$1|آلشت|آلشتیا}}",
+       "action-applychangetags": "سٱردیسیا ناْ ڤاگرد آلشتؽایؽ کاْ خوتو داٛئؽتٱ ڤ کار باٛیرؽت",
+       "action-changetags": "اْزاف کردن یا جا ڤ جاکاری سٱردیسؽا دل بهایی د ڤانریا ۉ پهرستنومٱیا شٱخسی",
+       "nchanges": "$1 {{PLURAL:$1|آلشت|آلشتؽا}}",
        "enhancedrc-since-last-visit": "$1 {{PLURAL:$1|د آخری دیئن}}",
        "enhancedrc-history": "ڤیرگار",
        "recentchanges": "آلشتؽا ایسنی",
        "recentchanges-noresult": "هیچ آلشتؽ د درازا دۉرٱ دؽار بیٱ ڤا اؽ ماٛعیارؽا یٱکی ناٛی.",
        "recentchanges-feed-description": "دۏ بؽشتر آلشتؽا تازباو ناْ د ڤیکی کا ها د هڤال هون پاٛگیری کو.",
        "recentchanges-label-newpage": "اؽ ڤیرایش یاٛ بٱلگٱ تازٱ دۏرس کردٱ.",
-       "recentchanges-label-minor": "یٱ یاٛ ڤیرایش کوچکٱ",
+       "recentchanges-label-minor": "یٱ یاٛ ڤیرایش کوچک ئٱ",
        "recentchanges-label-bot": "اؽ ڤيرايش ناْ ياٛ بوت ٱنجوم داٛیٱ",
-       "recentchanges-label-unpatrolled": "اؽ ڤيرايش هنی تيٱ ڤاداشت ناٛیئٱ",
-       "recentchanges-label-plusminus": "ٱندازٱ بٱلگٱ ڤ شماراٛ اؽ بایتؽا آلشت کردٱ.",
+       "recentchanges-label-unpatrolled": "اؽ ڤيرايش هنی تيٱ ڤاداشت ناٛییٱ",
+       "recentchanges-label-plusminus": "ٱندازٱ بٱلگٱ ڤ شمارٱ اؽ بایتؽا آلشت کردٱ.",
        "recentchanges-legend-heading": "<strong>میرات:</strong>",
        "recentchanges-legend-newpage": "{{int:recentchanges-label-newpage}} (همچنو باٛینؽت [[ڤیژٱ:بٱلگٱیا تازٱ|نوم گٱ بٱلگٱیا تازٱ]])",
        "recentchanges-legend-plusminus": "(<em>±123</em>)",
        "rcshowhidebots": "$1 روباتؽا یا بوتؽا",
        "rcshowhidebots-show": "نشوݩ داٛئن",
        "rcshowhidebots-hide": "قایم کردن",
-       "rcshowhideliu": "$1 کاریاریا سٱبت نوم کردٱ",
+       "rcshowhideliu": "$1 کاریارؽا سٱبت نوم کردٱ",
        "rcshowhideliu-show": "نشوݩ داٛئن",
        "rcshowhideliu-hide": "قایم کردن",
        "rcshowhideanons": "کاریار نادؽار $1",
        "rcshowhideanons-show": "نشوݩ داٛئن",
        "rcshowhideanons-hide": "قایم کردن",
        "rcshowhidepatr": "$1 ڤیرایشیا تیٱ پرس بیٱ",
-       "rcshowhidepatr-show": "Ù\86ئشÙ\88Ù\99 Ø¯Ø£ئن",
-       "rcshowhidepatr-hide": "قام کئردئن",
+       "rcshowhidepatr-show": "Ù\86Ø´Ù\88Ý© Ø¯Ø§Ù\9bئن",
+       "rcshowhidepatr-hide": "قایم کردن",
        "rcshowhidemine": "ڤیرایشؽا ماْ $1",
        "rcshowhidemine-show": "نشوݩ داٛئن",
        "rcshowhidemine-hide": "قایم کردن",
-       "rcshowhidecategorization": "جأرغە کاری بألگە $1",
+       "rcshowhidecategorization": "جٱرغٱ کاری بٱلگٱ $1",
        "rcshowhidecategorization-show": "نئشوٙ دأئن",
-       "rcshowhidecategorization-hide": "قام کئردئن",
-       "rclinks": "آخرین آلشتؽا $1 د آخرین رۊزؽا دؽاری بٱک $2",
+       "rcshowhidecategorization-hide": "قایم کردن",
+       "rclinks": "آخرین آلشتؽا $1 د آخری رۊزؽا دؽاری بٱک $2",
        "diff": "فٱرق",
        "hist": "ڤیرگار",
        "hide": "قایم کردن",
        "unpatrolledletter": "!",
        "rc-change-size": "$1",
        "rc-change-size-new": "$1 {{PLURAL:$1|بایت|بایتؽا}} دما آلشتکاری",
-       "newsectionsummary": "/* $1 */ Ø¨Ù\87رجا ØªØ§Ø²Ù\87",
-       "rc-enhanced-expand": "جزيات نشون بيئه",
-       "rc-enhanced-hide": "جزياته قام كو",
+       "newsectionsummary": "/* $1 */ Ø¨Ù±Ø¦Ø±Ø¬Ø§ ØªØ§Ø²Ù±",
+       "rc-enhanced-expand": "نشوݩ داٛئن جۏزئیات",
+       "rc-enhanced-hide": "قایم کردن جۏزئیات",
        "rc-old-title": "کاملٱن چی \"$1\"دۏرس بیٱ",
        "recentchangeslinked": "آلشتؽا تاٛ یٱکؽ",
        "recentchangeslinked-feed": "آلشتؽا تاٛ یٱک",
        "recentchangeslinked-toolbox": "آلشتؽا تاٛ یٱک",
        "recentchangeslinked-title": "آلشتؽا تاٛ یٱکؽ د $1",
-       "recentchangeslinked-summary": "اؽ نوم بٱلگٱ تازٱ د بٱلگٱیایی کاْ ڤا بٱلگٱیا ڤیژٱ هوم پاٛڤٱن بینٱ آلشت بیٱ(یا سی ٱندومؽا دٱسٱ بٱنی بیٱ)\nبٱلگٱیایی کاْ هان د [[Special:Watchlist|your watchlist]]ۉ گٱپ بینٱ",
+       "recentchangeslinked-summary": "اؽ نوم بٱلگٱ تازٱ د بٱلگٱیایؽ کاْ ڤا بٱلگٱیا ڤیژٱ هوم پاٛڤٱن بینٱ آلشت بیٱ(یا سی ٱندومؽا دٱسٱ بٱنی بیٱ)\nبٱلگٱیایی کاْ هان د [[Special:Watchlist|your watchlist]]ۉ گٱپ بینٱ",
        "recentchangeslinked-page": "نوم بٱلگٱ:",
        "recentchangeslinked-to": "آلشتؽایؽ کاْ د بٱلگٱیا هوم پاٛڤٱن بینٱ ڤ جا بٱلگٱ داٛیٱ بیٱ نشوݩ باٛیٱ",
-       "recentchanges-page-added-to-category": "[[:$1]]د دأسە ئضاف بی",
+       "recentchanges-page-added-to-category": "[[:$1]]ڤ دٱسٱ اْزاف بی",
        "recentchanges-page-added-to-category-bundled": "[[:$1]] و {{PLURAL:$2|بألگە تأکی|$2 بألگە یا}} د دأسە ئضاف بییئن",
        "recentchanges-page-removed-from-category": "[[:$1]] د دٱسٱ جگا بی",
        "recentchanges-page-removed-from-category-bundled": "[[:$1]] و {{PLURAL:$2|بألگە تأکی|$2 بألگە یا}} د دأسە ئضاف بییئن",
-       "autochange-username": "Ø¢Ù\84ئشتکارÛ\8c Ø®Ù\88دأÙ\86جÙ\88Ù\85 Ù\85ئدیاڤیکی",
+       "autochange-username": "Ø¢Ù\84شتکارÛ\8c Ø®Ù\88دٱÙ\86جÙ\88Ù\85 Ù\85دیاڤیکی",
        "upload": "سڤار کردن جانؽا",
-       "uploadbtn": "سوڤار کئردئن جانیا",
-       "reuploaddesc": "سوار کردن نه انجوم شیو بکیت و د ورئردیت جابلگ سوارکرد",
-       "upload-tryagain": "کل کردن توضیحیا آلشت دئیه بیه جانیا",
+       "uploadbtn": "سڤار کردن جانؽا",
+       "reuploaddesc": "سڤار کردن ناْ ٱنجوم شؽڤ بٱکؽت ۉ ڤرگٱردؽد  جا بٱلگٱ سڤارکرد",
+       "upload-tryagain": "کلٛ کردن تۉزیهؽا آلشت داٛئٱ بیٱ جانؽا",
        "uploadnologin": "هنی نۏمایتٱ ڤامیٛن",
        "uploadnologintext": "لطفن $1 سی سوارکرد جانیایا.",
        "upload_directory_missing": "نشونگه سوارکرد ($1) وجود ناره و نبوئه دروسش بکی.",
index 2e5a6ba..7614e38 100644 (file)
        "protectedpages-timestamp": "Laiko žyma",
        "protectedpages-page": "Puslapis",
        "protectedpages-expiry": "Galioja iki",
-       "protectedpages-performer": "Užrakinantis naudotojas",
+       "protectedpages-performer": "Užrakinęs naudotojas",
        "protectedpages-params": "Užrakinimo nuostatos",
        "protectedpages-reason": "Priežastis",
        "protectedpages-submit": "Rodyti puslapius",
index 4763645..70c07c3 100644 (file)
        "rcfilters-clear-all-filters": "Noņemt visus filtrus",
        "rcfilters-show-new-changes": "Skatīt jaunās izmaiņas kopš $1",
        "rcfilters-search-placeholder": "Filtrēt pēdējās izmaiņas (pārlūko vai sāc rakstīt)",
+       "rcfilters-search-placeholder-mobile": "Filtri",
        "rcfilters-invalid-filter": "Nederīgs filtrs",
        "rcfilters-empty-filter": "Nav aktīvu filtru. Tiek rādītas visas izmaiņas.",
        "rcfilters-filterlist-title": "Filtri",
index 1ecabd7..a6b7a5b 100644 (file)
        "search-interwiki-more": "(уште)",
        "search-interwiki-more-results": "повеќе ставки",
        "search-relatedarticle": "Поврзано",
+       "search-invalid-sort-order": "Не го препознавам редоследот на подредување во $1; ќе применам основно подредување. Важечки редоследи се: $2",
+       "search-unknown-profile": "Не го препознавам пребарувачкиот профил на $1; ќе го применам основниот.",
        "searchrelated": "поврзано",
        "searchall": "сè",
        "showingresults": "Подолу {{PLURAL:$1|е прикажана <strong>1</strong> ставка|се прикажани <strong>$1</strong> ставки}} почнувајќи од бр. <strong>$2</strong>.",
        "rcfilters-clear-all-filters": "Тргни ги сите филтри",
        "rcfilters-show-new-changes": "Погл. нови промени од $1 наваму",
        "rcfilters-search-placeholder": "Филтрирање на промени (користете го менито или пребарајте назив на филтер)",
+       "rcfilters-search-placeholder-mobile": "Филтри",
        "rcfilters-invalid-filter": "Неважечки филтер",
        "rcfilters-empty-filter": "Нема активни филтри. Прикажани се сите придонеси.",
        "rcfilters-filterlist-title": "Филтри",
        "changecontentmodel": "Промена на содржинскиот модел на страница",
        "changecontentmodel-legend": "Промена на содржински модел",
        "changecontentmodel-title-label": "Наслов на страницата",
+       "changecontentmodel-current-label": "Тековен содржински модел:",
        "changecontentmodel-model-label": "Нов содржински модел",
        "changecontentmodel-reason-label": "Причина:",
        "changecontentmodel-submit": "Смени",
        "block-log-flags-angry-autoblock": "овозможено проширено автоблокирање",
        "block-log-flags-hiddenname": "сокриено корисничко име",
        "range_block_disabled": "Администраторската можност да блокираат IP групи е исклучена.",
+       "ipb-prevent-user-talk-edit": "Уредувањето на сопствената разговорна страница мора да биде дозволено при делумен блок, освен ако не вклучува ограничување за разговорната страница.",
        "ipb_expiry_invalid": "Погрешен рок на истекување.",
        "ipb_expiry_old": "Времето на истекување е постаро од тековното време.",
        "ipb_expiry_temp": "Скриените блокирања на корисникот мора да бидат перманентни.",
        "move-page-legend": "Премести страница",
        "movepagetext": "Со користењето на овој образец можете да преименувате страница, преместувајќи ја целата нејзина историја под ново име.\nСтариот наслов ќе стане пренасочувачка страница кон новиот наслов.\nАвтоматски можете да ги подновите пренасочувањата кои покажуваат кон првобитниот наслов.\nАко не изберете автоматско подновување, проверете на [[Special:DoubleRedirects|двојни]] или [[Special:BrokenRedirects|прекинати пренасочувања]].\nНа вас е одговорноста да се осигурате дека врските ќе продолжат да насочуваат таму за каде се предвидени.\n\nИмајте предвид дека страницата '''нема''' да биде преместена ако веќе постои страница со новиот наслов, освен ако е не е пренасочување и нема историја на минати уредувања. Тоа значи дека можете да ја преименувате страницата како што била претходно доколку сте направиле грешка без да ја прекриете постоечката страница.\n\n'''Напомена:'''\nОва може да биде драстична и неочекувана промена за популарна страница;\nосигурајте се дека сте ги разбрале последиците од ова пред да продолжите.",
        "movepagetext-noredirectfixer": "Со користењето на овој образец можете да преименувате страница, преместувајќи ја целата нејзина историја под ново име.\nСтариот наслов ќе стане пренасочувачка страница кон новиот наслов.\nАвтоматски можете да ги подновите пренасочувањата кои покажуваат кон првобитниот наслов.\nНе заборавајте да проверите [[Special:DoubleRedirects|двојни]] и [[Special:BrokenRedirects|прекинати пренасочувања]].\nНа вас е одговорноста да се осигурате дека врските ќе продолжат да насочуваат таму за каде се предвидени.\n\nИмајте предвид дека страницата '''НЕМА''' да биде преместена ако веќе постои страница со новиот наслов, освен ако е празна или ако е пренасочување и нема историја на минати уредувања. Тоа значи дека можете да ја преименувате страницата како што била претходно доколку сте направиле грешка без да ја прекриете постоечката страница.\n\n'''Напомена:'''\nОва може да биде драстична и неочекувана промена за популарна страница;\nосигурајте се дека сте ги разбрале последиците од ова пред да продолжите.",
+       "movepagetext-noredirectsupport": "Со користењето на овој образец можете да преименувате страница, преместувајќи ја целата нејзина историја под ново име.\nНа вас е одговорноста да се осигурате дека врските ќе продолжат да насочуваат таму за каде се предвидени.\n\nИмајте предвид дека страницата <strong>нема</strong> да биде преместена ако веќе постои страница со новиот наслов.\nТоа значи дека можете да ја преименувате страницата како што била претходно доколку сте направиле грешка без да ја прекриете постоечката страница.\n\n<strong>Напомена:</strong>\nОва може да биде драстична и неочекувана промена за популарна страница;\nосигурајте се дека сте ги разбрале последиците од ова пред да продолжите.",
        "movepagetalktext": "Ако го штиклирате кутивчево, соодветната разговорна страница ќе биде автоматски преместена на нов наслов, освен ако таму веќе постои разговорна страница  што не е празна.\n\nВо тој случај, ќе треба да ја преместите или споите страницата рачно, доколку сакате.",
        "moveuserpage-warning": "'''Предупредување:''' На пат сте да преместите корисничка страница. Имајте предвид дека само страницата ќе биде преместена, а самиот корисник ''нема'' да биде преименуван.",
        "movecategorypage-warning": "<strong>Предупредување:</strong> Преместувате категориска страница. Имајте предвид дека ќе се премести само страницата, а страниците во старата категорија <em>нема</em> да се прекатегоризираат во новата.",
        "linkaccounts": "Поврзи сметки",
        "linkaccounts-success-text": "Сметката е поврзана.",
        "linkaccounts-submit": "Поврзи сметки",
+       "cannotunlink-no-provider-title": "Нема поврзани сметки за одврзување",
+       "cannotunlink-no-provider": "Немате поврзани сметки за одврзување.",
        "unlinkaccounts": "Одврзи сметки",
        "unlinkaccounts-success": "Сметката е одврзана.",
        "authenticationdatachange-ignored": "Промената на податоците во заверката не е обработена. Можеби не е поставен услужник?",
        "gotointerwiki": "Го напуштате {{SITENAME}}",
        "gotointerwiki-invalid": "Укажаниот наслов е неважечки.",
        "gotointerwiki-external": "Го напуштате {{SITENAME}} упатени кон [[$2]], кое е посебно мрежно место.\n\n'''[$1 Продолжете кон $1]'''",
-       "undelete-cantedit": "Не можете да ја вратите оваа страница бидејќи уредувањето на страницава не ви е дозволено.",
+       "undelete-cantedit": "Не можете да ја вратите оваа страница бидејќи не ви е дозволено да ја уредувате.",
        "undelete-cantcreate": "Не можете да ја вратите страницава бидејќи не постои страница со таков назив и не ви е дозволено да ја создадете.",
        "pagedata-title": "Податоци за страницата",
        "pagedata-text": "Страницава дава посредник за податоци за страниците. Укажете го насловот на страницата во URL-то, користејќи ја синтаксата за потстраници.\n* Префрлањето на содржината се заснова на заглавието Прифати на вашиот клиент. Ова значи дека податоците за страницата ќе бидат ставени во форматот кој го претпочита вашиот клиент.",
index d2c5154..157b87c 100644 (file)
@@ -1,20 +1,21 @@
 {
        "@metadata": {
                "authors": [
-                       "Awangba Mangang"
+                       "Awangba Mangang",
+                       "Amire80"
                ]
        },
        "tog-underline": "ꯃꯔꯤꯂꯩꯅꯥꯍꯜꯂꯨ:",
        "tog-hideminor": "ꯂꯣꯠꯂꯨ ꯈꯖꯤꯛꯇꯪ ꯁꯣꯏꯕꯗꯒꯤ",
        "tog-hidepatrolled": "ꯂꯣꯠꯂꯨ ꯈꯖꯤꯛꯇꯪ ꯁꯣꯏꯕꯗꯒꯤ ꯍꯧꯖꯤꯛꯀꯤ ꯑꯍꯣꯡꯕꯗꯒꯤ",
        "tog-newpageshidepatrolled": "Hide patrolled pages from new page list",
-       "tog-hidecategorization": "ê¯\82ꯥê¯\83ꯥê¯\8fê¯\92ꯤ ê¯\83ê¯\8aꯪ ê¯\83ê¯\85ꯥê¯\8e ê¯\85ꯥê¯\8fê¯\95ꯥ ê¯\82ꯣꯠê¯\84ꯥ",
+       "tog-hidecategorization": "ꯂꯃꯥꯏꯒꯤ ꯃꯊꯪ ꯃꯅꯥꯎ ꯅꯥꯏꯕꯥ ꯂꯣꯠꯄꯥ",
        "tog-extendwatchlist": "ꯌꯦꯡꯅꯕꯥ ꯄꯥꯔꯦꯡꯗꯨ ꯄꯥꯛꯊꯣꯛꯍꯟꯂꯨ ꯑꯍꯣꯡꯕꯥ ꯂꯣꯏꯅꯥ ꯎꯅꯕꯥ, ꯍꯧꯖꯤꯛꯀꯤ ꯈꯋꯥꯏꯗꯒꯤ ꯑꯅꯛꯄ ꯑꯍꯣꯡꯕꯗꯨ ꯅꯠꯇꯕꯥ",
        "tog-usenewrc": "Group changes by page in recent changes and watchlist",
        "tog-numberheadings": "ꯃꯁꯥ ꯃꯇꯣꯝꯇꯥ-ꯃꯁꯤꯡ-ꯃꯀꯣꯛꯁꯤꯡ",
        "tog-editondblclick": "꯲ ꯔꯛ ꯅꯝꯃꯒꯥ ꯂꯥꯃꯥꯏꯗꯨ ꯁꯦꯝꯒꯠꯂꯨ",
        "tog-editsectiononrightclick": "section titles ꯗ ꯌꯦꯠꯅ ꯅꯝꯗꯨꯅꯥ ꯁꯦꯝꯒꯠꯂꯛꯅꯕꯒꯤ ꯃꯐꯝꯗꯨ ꯌꯥꯍꯟꯂꯨ",
-       "tog-watchcreations": "ê¯\8dꯥê¯\9eê¯\86ꯤê¯\9fê¯\82ꯨ ê¯\91ꯩê¯\85ꯥ ê¯\81ꯦê¯\9dê¯\82ê¯\9bê¯\84ꯥ ê¯\82ꯥê¯\83ꯥê¯\8fê¯\97ꯨ ê¯\91ê¯\83ê¯\97ꯤ ê¯\91ꯩê¯\85ꯥ ê¯\8aꯥê¯\92ꯠê¯\82ê¯\9bê¯\84ꯥ ê¯\90ꯥê¯\8fê¯\9c ê¯\97ꯨ ê¯\91ꯩê¯\92ꯤ ê¯\8cꯦꯡê¯\85ê¯\95ꯥ ê¯\84ê¯\94ꯦꯡê¯\97ꯥ",
+       "tog-watchcreations": "ꯍꯥꯞꯆꯤꯟꯂꯨ ꯑꯩꯅꯥ ꯁꯦꯝꯂꯛꯄꯥ ꯂꯃꯥꯏꯗꯨ ꯑꯃꯗꯤ ꯑꯩꯅꯥ ꯊꯥꯒꯠꯂꯛꯄꯥ ꯐꯥꯏꯜ ꯗꯨ ꯑꯩꯒꯤ ꯌꯦꯡꯅꯕꯥ ꯄꯔꯦꯡꯗꯥ",
        "tog-watchdefault": "ꯍꯥꯞꯆꯤꯟꯂꯨ ꯂꯥꯃꯥꯏ ꯑꯃꯗꯤ ꯑꯩꯅꯥ ꯁꯦꯝꯒꯠꯂꯛꯄꯥ ꯐꯥꯏꯜ ꯗꯨ ꯑꯩꯒꯤ ꯌꯦꯡꯅꯕꯥ ꯄꯔꯦꯡꯗ",
        "tog-watchmoves": "I move to my watchlist ꯍꯥꯞꯆꯤꯟꯂꯨ ꯂꯥꯃꯥꯏ ꯑꯃꯗꯤ ꯐꯥꯏꯜꯁꯤꯡ",
        "tog-watchdeletion": "ꯍꯥꯞꯆꯤꯟꯂꯨ ꯂꯥꯃꯥꯏ ꯑꯃꯗꯤ ꯐꯥꯏꯜ ꯑꯩꯅꯥ ꯀꯛꯊꯠꯈꯤꯕꯥ ꯗꯨ ꯑꯩꯒꯤ ꯌꯦꯡꯅꯕꯥ ꯄꯔꯦꯡꯗꯥ",
        "tog-previewontop": "Show preview before edit box",
        "tog-previewonfirst": "Show preview on first edit",
        "tog-enotifwatchlistpages": "Email me when a page or a file on my watchlist is changed",
-       "tog-enotifusertalkpages": "Email me when my user talk page is changed",
+       "tog-enotifusertalkpages": "ꯏꯃꯦꯜ ꯊꯥꯔꯛꯎ ꯑꯩꯒꯤ ꯁꯤꯖꯤꯟꯅꯔꯤꯕ ꯋꯥ ꯍꯥꯏꯐꯝ ꯂꯃꯥꯏꯗꯨ ꯍꯣꯡꯗꯣꯛꯄꯥ ꯃꯇꯝꯗ",
        "tog-enotifminoredits": "Email me also for minor edits of pages and files",
        "tog-enotifrevealaddr": "Reveal my email address in notification emails",
        "tog-shownumberswatching": " watching users ꯀꯤ ꯃꯁꯤꯡꯗꯨ ꯎꯨꯠꯂꯨ",
-       "tog-oldsig": "ꯅꯪꯒꯤ gi hanna leijariba khutyek:",
+       "tog-oldsig": "ꯅꯪꯒꯤ ꯍꯥꯟꯅ ꯂꯩꯖꯔꯤꯕ ꯈꯨꯠꯌꯦꯛ:",
        "tog-fancysig": "Khutyek tu wikitext oina khanlu(Mashamatomta Samnafam yaodaba)",
        "tog-uselivepreview": "Anouba ꯂꯥꯃꯥꯏ chingthadabida hannagido ootlu",
        "tog-forceeditsummary": "Ahangba semgatlakpa hapchinlak pa matamda hairak o eingonda",
        "nov": "ꯅꯣꯚ",
        "dec": "ꯗꯦꯈ",
        "january-date": "$1 ꯖꯥꯅꯨꯋꯥꯔꯤ",
-       "february-date": "$1 ê¯\84ꯦꯕꯔꯨꯋꯥꯔꯤ",
+       "february-date": "$1 ê¯\90ꯦꯕꯔꯨꯋꯥꯔꯤ",
        "march-date": "$1ꯃꯥꯔꯆ",
        "april-date": "$1 ꯑꯦꯄꯔꯤꯜ",
        "may-date": "$1 ꯃꯦ",
        "category-file-count-limited": "The following {{PLURAL:$1|file is|$1 files are}} in the current category.",
        "listingcontinuesabbrev": "ꯆꯠꯊꯕꯥ",
        "index-category": "Indexed ꯂꯥꯃꯥꯏꯁꯤꯡ",
-       "noindex-category": "Noindexed ꯂꯃꯥꯏꯁꯤꯡ",
-       "broken-file-category": " ꯀꯥꯏꯔꯕꯥ file links ꯒꯥ ꯂꯣꯏꯅꯕꯥ ꯂꯥꯃꯥꯏꯁꯤꯡ",
+       "noindex-category": "ꯏꯟꯗꯦꯛꯁ ꯌꯥꯎꯗꯕ ꯂꯃꯥꯏꯁꯤꯡ",
+       "broken-file-category": " ꯀꯥꯏꯔꯕꯥ ꯐꯥꯏꯜꯒꯥ ꯂꯣꯏꯅꯕꯥ ꯂꯃꯥꯏꯁꯤꯡ",
        "about": "ꯄꯣꯠꯇꯨꯗꯤ ꯃꯔꯝꯗꯥ",
-       "article": "ê¯\82ꯥê¯\83ꯥê¯\8fê¯\81ꯤê¯\97ꯥ ê¯\8cꯥê¯\8eê¯\95ꯥ ê¯\84ꯨê¯\9dê¯\85ê¯\83ê¯\9b",
+       "article": "ꯂꯃꯥꯏꯁꯤꯗꯥ ꯌꯥꯎꯕꯥ ꯄꯨꯝꯅꯃꯛ",
        "newwindow": "(ꯑꯅꯧꯕꯥ ꯊꯣꯡꯅꯥꯎꯗꯥ ꯍꯥꯡꯗꯣꯛ ꯎ)",
        "cancel": "ꯇꯣꯛꯄ",
        "moredotdotdot": "ꯋꯥꯠꯂꯤ",
        "mytalk": "ꯉꯥꯡꯐꯝ",
        "anontalk": "ꯉꯥꯡꯐꯝ",
        "navigation": "ꯆꯠꯄꯥ",
-       "and": "ꯑꯃꯁꯨꯡ #꯳꯲; ꯑꯃꯁꯨꯪ",
-       "faq": "FAQ",
-       "actions": "Actions",
+       "and": "&#32;ꯑꯃꯁꯨꯡ",
+       "faq": "ꯇꯍꯋ",
+       "actions": "ꯊꯕꯛ ꯄꯥꯡꯊꯣꯛꯄ",
        "namespaces": "ꯃꯃꯤꯡꯒꯤ ꯃꯐꯝ",
        "variants": "ꯈꯦꯠꯅꯕꯥ",
        "navigation-heading": "ꯆꯠꯅꯕ ꯌꯦꯡꯐꯝ",
        "view": "ꯃꯤꯠꯌꯦꯡ",
        "view-foreign": "$1 ꯗꯥ ꯌꯦꯡꯉꯨ",
        "edit": "ꯁꯦꯝꯒꯠꯄꯥ",
-       "edit-local": "Edit local description",
+       "edit-local": "ꯁꯦꯝꯒꯠꯄꯒꯤ ꯑꯀꯨꯞꯄ ꯋꯥꯔꯣꯜ",
        "create": "ꯁꯥꯕꯥ",
        "create-local": "ꯁꯨꯋꯥꯏꯒꯤ ꯁꯟꯗꯣꯛꯅꯥ ꯇꯥꯛꯄꯗꯨ ꯍꯥꯞꯆꯤꯟꯂꯨ",
        "delete": "ꯀꯛꯊꯠꯄꯥ",
        "undelete_short": "ꯀꯛꯊꯠꯀꯅꯨ {{PLURAL:$1|ꯁꯦꯝꯒꯠꯄ ꯑꯃꯥ|ꯁꯦꯝꯒꯠꯄꯁꯤꯡ $1}}",
-       "viewdeleted_short": "ꯌꯦꯡꯕ {{PLURAL:$1|one deleted edit|$1 deleted edits}}",
+       "viewdeleted_short": "ꯌꯦꯡꯕ {{PLURAL:$1|ꯀꯛꯊꯠꯈꯔꯕ ꯁꯦꯝꯒꯠꯄ ꯱|$1 ꯀꯛꯊꯠꯈꯔꯕ ꯁꯦꯝꯒꯠꯄꯁꯤꯡ}}",
        "protect": "ꯉꯥꯛꯊꯣꯛꯂꯕꯥ",
        "protect_change": "ꯑꯍꯣꯡꯕꯥ",
        "unprotect": "ꯍꯥꯛꯊꯣꯛꯂꯕꯥ ꯗꯨ ꯍꯣꯡꯕꯥ",
        "tool-link-userrights-readonly": "ꯌꯦꯡꯕ {{GENDER:$1|ꯁꯤꯖꯤꯟꯅꯔꯤꯕꯁꯤꯡ}} ꯀꯥꯡꯕꯨꯁꯤꯡ",
        "tool-link-emailuser": "ꯏꯃꯦꯜ ꯊꯥꯈꯣ {{GENDER:$1|ꯁꯤꯖꯤꯟꯅꯔꯤꯕ}}",
        "imagepage": "ꯐꯥꯏꯜ ꯂꯃꯥꯏꯗꯨ ꯎꯠꯂꯨ",
-       "mediawikipage": "ê¯\84ꯥê¯\8eê¯\96ꯦê¯\9cê¯\92ꯤ ê¯\82ꯥê¯\83ꯥê¯\8fê¯\97ꯨ ê¯\8eꯨꯠê¯\82ꯨ",
+       "mediawikipage": "ꯄꯥꯎꯖꯦꯜꯒꯤ ꯂꯃꯥꯏꯗꯨ ꯎꯨꯠꯂꯨ",
        "templatepage": "ꯇꯦꯝꯄꯂꯦꯠꯀꯤ ꯂꯃꯥꯏꯗꯨ ꯎꯨꯠꯂꯨ",
        "viewhelppage": "ꯃꯇꯦꯡ ꯄꯥꯡꯅꯕꯒꯤ ꯂꯥꯃꯥꯏꯗꯨ ꯎꯨꯠꯂꯨ",
        "categorypage": "ꯃꯆꯥꯈꯥꯏꯕ ꯂꯃꯥꯏꯗꯨ ꯎꯨꯠꯂꯨ",
        "redirectto": "Redirect to:",
        "lastmodifiedat": "$1 ꯗꯥ ꯃꯁꯤꯒꯤ ꯂꯃꯥꯏꯁꯤ ꯑꯔꯣꯏꯕꯥ ꯁꯦꯝꯒꯠꯈꯤꯕꯥ, $2 ꯗꯥ",
        "viewcount": "This page has been accessed {{PLURAL:$1|once|$1 times}}?",
-       "protectedpage": "ê¯\8dꯥê¯\9bê¯\8aꯣê¯\9bê¯\82ê¯\95ꯥ ê¯\82ꯥꯃꯥꯏ",
-       "jumpto": "ꯑꯗꯨꯗꯥ  ꯆꯣꯡꯈꯠꯄꯥ",
+       "protectedpage": "ê¯\89ꯥê¯\9bê¯\8aꯣê¯\9bê¯\82ê¯\95ꯥ ê¯\82ꯃꯥꯏ",
+       "jumpto": "ꯑꯗꯨꯗꯥ  ꯆꯣꯡꯈꯠꯄꯥ:",
        "jumptonavigation": "ꯆꯠꯄꯥ",
        "jumptosearch": "ꯊꯤꯕꯥ",
        "view-pool-error": "Sorry, the servers are overloaded at the moment.\nToo many users are trying to view this page.\nPlease wait a while before you try to access this page again.\n\n$1",
        "pool-timeout": "ꯃꯇꯝ ꯍꯦꯟꯂꯦ ꯂꯣꯟꯅꯕꯥꯒꯤ ꯉꯥꯏꯕꯥ",
        "pool-queuefull": "ꯄꯨꯜꯒꯤ ꯄꯔꯦꯡ ꯊꯟꯂꯦ",
        "pool-errorunknown": "ꯁꯛꯈꯪꯗꯕꯥ ꯑꯔꯥꯟꯕꯥ",
-       "pool-servererror": "$1 ꯄꯨꯜ ꯀꯥꯎꯟꯇꯔ ꯇꯧꯅꯕꯥ ꯂꯩꯇꯔꯦ",
+       "pool-servererror": "$1 ꯄꯨꯜ ꯀꯥꯎꯟꯇꯔ ꯇꯧꯅꯕꯥ ꯂꯩꯇꯔꯦ ꯫",
        "poolcounter-usage-error": "$1:ꯁꯤꯖꯤꯟꯅꯔꯤꯕꯥ ꯑꯔꯥꯟꯕꯥ",
        "aboutsite": "{{SITENAME}} ꯃꯔꯝꯗꯥ",
        "aboutpage": "Project:ꯃꯔꯝꯗꯥ",
        "versionrequired": "ꯃꯦꯗꯤꯌꯥ ꯋꯤꯀꯤꯅ ꯋꯥꯠꯂꯤꯕꯥ $1ꯕꯔꯖꯟ",
        "versionrequiredtext": "ꯃꯦꯗꯤꯌꯥ ꯋꯤꯀꯤꯅ ꯋꯥꯠꯂꯤꯕꯥ $1ꯕꯔꯖꯟ ꯃꯁꯤꯒꯤ ꯂꯥꯃꯥꯏꯁꯤꯗꯥ ꯁꯤꯖꯤꯟꯅꯕꯥ [[Special:Version|version page]].",
        "ok": "ꯌꯥꯔꯦ",
-       "retrievedfrom": "$1 ꯃꯐꯝꯗꯨꯗꯒꯤ ꯑꯣꯏꯔꯛꯄꯥ",
-       "youhavenewmessages": "{{PLURAL:$3|ꯅꯪꯉꯣꯟꯗ ꯂꯩ}} $1 ($2).",
+       "retrievedfrom": "\"$1\" ꯃꯐꯝꯗꯨꯗꯒꯤ ꯑꯣꯏꯔꯛꯄꯥ",
+       "youhavenewmessages": "{{PLURAL:$3|ꯅꯪꯉꯣꯟꯗ ꯂꯩ}} $1 ($2) ꯫",
        "youhavenewmessagesfromusers": "{{PLURAL:$4|You have}} $1 from {{PLURAL:$3|another user|$3 users}} ($2).",
-       "youhavenewmessagesmanyusers": "ꯅꯪ $1 ꯂꯩꯔꯦ $2 ꯁꯤꯖꯤꯟꯅꯔꯤꯕꯥ ꯃꯌꯥꯝꯗꯒꯤ",
+       "youhavenewmessagesmanyusers": "ꯅꯪ $1 ꯂꯩꯔꯦ $2 ꯁꯤꯖꯤꯟꯅꯔꯤꯕꯥ ꯃꯌꯥꯝꯗꯒꯤ ꯫",
        "newmessageslinkplural": "{{PLURAL:$1|ꯑꯅꯧꯕ ꯄꯥꯎꯖꯦꯜ ꯱|꯹꯹꯹=ꯑꯅꯧꯕ ꯄꯥꯎꯖꯦꯜꯁꯤꯡ}}",
        "newmessagesdifflinkplural": "ꯑꯔꯣꯏꯕꯥ {{PLURAL:$1|ꯑꯍꯣꯡꯕ|꯹꯹꯹=ꯑꯍꯣꯡꯕꯁꯤꯡ}}",
-       "youhavenewmessagesmulti": "$1 ê¯\85ꯪê¯\92ꯤ ê¯\91ê¯\85ꯧê¯\95ꯥ ê¯\83ꯦê¯\81ꯦê¯\81",
+       "youhavenewmessagesmulti": "$1 ê¯\97 ê¯\85ꯪê¯\92ꯤ ê¯\91ê¯\85ꯧê¯\95ꯥ ê¯\84ꯥê¯\8eê¯\96ꯦê¯\9c ê¯\82ꯩê¯\94ꯦ",
        "editsection": "ꯁꯦꯝꯒꯠꯄ",
        "editold": "ꯁꯦꯝꯒꯠꯄ",
        "viewsourceold": "ꯍꯧꯔꯛꯐꯝ ꯎꯨꯠꯄ",
        "editlink": "ꯁꯦꯝꯒꯠꯄꯥ",
        "viewsourcelink": "ꯍꯧꯔꯛꯐꯝ ꯎꯨꯠꯄ",
        "editsectionhint": "ꯁꯦꯝꯒꯠꯄꯒꯤ ꯁꯔꯨꯛ: $1",
-       "toc": "ê¯\91ê¯\8cꯥê¯\8eê¯\95ꯥ",
+       "toc": "ê¯\91ê¯\8cꯥê¯\8eê¯\95ê¯\81ꯤꯡ",
        "showtoc": "ꯎꯨꯠꯂꯨ",
        "hidetoc": "ꯂꯣꯇꯄ",
        "collapsible-collapse": "ꯁꯨꯞꯆꯤꯟꯕꯥ",
        "thisisdeleted": "View or restore $1?",
        "viewdeleted": "$1 ꯌꯦꯡꯍꯟꯂꯨ?",
        "restorelink": "{{PLURAL:$1|one deleted edit|$1 deleted edits}}",
-       "feedlinks": "ꯐꯤꯗ",
+       "feedlinks": "ꯐꯤꯗ:",
        "feed-invalid": "ꯌꯥꯎꯗꯕꯥ subscription feed type.",
-       "feed-unavailable": "Syndication feeds are not available",
+       "feed-unavailable": "ꯐꯤꯗꯁ ꯐꯪꯗꯦ",
        "site-rss-feed": "$1 RSS feed",
        "site-atom-feed": "$1 ꯑꯦꯇꯣꯝ ꯇꯥꯛꯄꯥ",
        "page-rss-feed": "\"$1\" RSS feed",
        "viewsource": "ꯍꯧꯔꯛꯐꯝ ꯎꯨꯠꯂꯨ",
        "viewsource-title": "$1 ꯒꯤ ꯍꯧꯔꯛꯐꯝ ꯎꯨꯠꯂꯨ",
        "viewsourcetext": "ꯅꯪꯅꯥ ꯌꯦꯡꯕꯥ ꯌꯥꯒꯅꯤ ꯑꯃꯗꯤ ꯃꯁꯤꯒꯤ ꯂꯥꯃꯥꯏꯁꯤꯒꯤ ꯍꯧꯔꯛꯐꯝ ꯁꯤꯟꯇꯣꯛ ꯎ",
-       "mycustomjsonprotected": "ê¯\85ꯪê¯\85ꯥ ê¯\83ê¯\81ꯤ json ê¯\82ꯥê¯\83ꯥê¯\8fê¯\81ꯤ ê¯\81ꯦê¯\9dê¯\92ꯠê¯\85ê¯\95ꯥ ê¯\91ê¯\8cꯥê¯\95ꯥ ê¯\84ꯤê¯\97ꯦ",
-       "mycustomjsprotected": "JavaScript ê¯\82ꯥê¯\83ꯥê¯\8fê¯\81ꯤ ê¯\85ꯪê¯\85ꯥ ê¯\81ꯦê¯\9dê¯\92ꯠê¯\85ê¯\95ê¯\92ꯤ ê¯\91ê¯\8cꯥê¯\95ꯥ ê¯\84ꯤê¯\97ꯦ",
+       "mycustomjsonprotected": "ê¯\85ꯪê¯\85ꯥ ê¯\83ê¯\81ꯤ json ê¯\82ê¯\83ꯥê¯\8fê¯\81ꯤ ê¯\81ꯦê¯\9dê¯\92ꯠê¯\85ê¯\95ꯥ ê¯\91ê¯\8cꯥê¯\95ꯥ ê¯\84ꯤê¯\97ꯦ ê¯«",
+       "mycustomjsprotected": "JavaScript ê¯\82ê¯\83ꯥê¯\8fê¯\81ꯤ ê¯\85ꯪê¯\89ꯣê¯\9fê¯\97 ê¯\81ꯦê¯\9dê¯\92ꯠê¯\85ê¯\95ê¯\92ꯤ ê¯\91ê¯\8cꯥê¯\95ꯥ ê¯\84ꯤê¯\97ꯦ ê¯«",
        "myprivateinfoprotected": "ꯅꯪꯅꯥ ꯅꯪꯒꯤ ꯑꯔꯣꯟꯕꯥ ꯑꯀꯨꯞꯄ ꯋꯥꯔꯣꯜ ꯂꯧꯐꯝ ꯁꯤ ꯁꯦꯝꯒꯠꯅꯕꯒꯤ ꯑꯌꯥꯕꯥ ꯄꯤꯗꯦ",
        "mypreferencesprotected": "ꯅꯪꯅꯥ ꯅꯪꯒꯤ ꯄꯔꯤꯐꯔꯦꯟꯁ ꯁꯤ ꯁꯦꯝꯒꯠꯅꯕꯒꯤ ꯑꯌꯥꯕꯥ ꯄꯤꯗꯦ",
-       "ns-specialprotected": "ê¯\91ê¯\88ê¯\9fê¯\85ê¯\95ꯥ ê¯\82ꯥê¯\83ꯥê¯\8fê¯\85ꯤ ê¯\81ꯦê¯\9dê¯\97ꯣê¯\9bê¯\84ꯥ ê¯\8cꯥê¯\97ꯦ",
+       "ns-specialprotected": "ê¯\91ê¯\88ê¯\9fê¯\85ê¯\95ꯥ ê¯\82ê¯\83ꯥê¯\8fê¯\85ꯤ ê¯\81ꯦê¯\9dê¯\97ꯣê¯\9bê¯\84ꯥ ê¯\8cꯥê¯\97ꯦ ê¯«",
        "titleprotected": "ꯃꯁꯤꯒꯤ ꯃꯃꯤꯡ ꯁꯤ ꯉꯥꯛꯊꯣꯛꯂꯦ ꯁꯥꯒꯠꯂꯛꯄꯗꯥ [[User:$1|$1]].\nThe reason given is <em>$2</em>.",
        "filereadonlyerror": "Unable to modify the file \"$1\" because the file repository \"$2\" is in read-only mode.\n\nThe system administrator who locked it offered this explanation: \"$3\".",
        "invalidtitle": "ꯃꯃꯤꯡꯁꯤ ꯂꯩꯇꯦ",
        "logouttext": "<strong>You are now logged out.</strong>\n\nNote that some pages may continue to be displayed as if you were still logged in, until you clear your browser cache.",
        "cannotlogoutnow-title": "ꯍꯧꯖꯤꯛ ꯂꯣꯒ out ꯇꯧꯕꯥ ꯌꯥꯗꯦ",
        "cannotlogoutnow-text": "$1 ꯁꯤꯖꯤꯟꯅꯔꯤꯉꯩꯗꯥ ꯂꯣꯒꯒꯤꯡ out ꯁꯤ ꯑꯣꯏꯊꯣꯛꯇꯦ",
-       "welcomeuser": "$1ꯂꯦꯡꯁꯤꯟꯕꯤꯔꯛꯁꯤ",
+       "welcomeuser": "ꯂꯦꯡꯁꯤꯟꯕꯤꯔꯛꯁꯤ,$1!",
        "welcomecreation-msg": "ꯅꯪꯒꯤ ꯑꯦꯀꯥꯎꯟꯇ ꯁꯤ ꯁꯥꯈꯔꯦ\nꯅꯪꯒꯤ ꯑꯄꯥꯝꯕꯒꯤ ꯃꯇꯨꯡ ꯏꯟꯅꯥ ꯍꯣꯡꯗꯣꯛꯄꯥ ꯌꯥꯔꯦ ꯅꯪꯅꯥ {{SITENAME}} [[Special:Preferences|preferences]] ꯅꯪꯒꯤ ꯑꯄꯥꯝꯕꯒꯤ ꯃꯇꯨꯡꯏꯟꯅꯥ.",
-       "yourname": "ꯁꯤꯖꯤꯟꯅꯔꯤꯕꯥ ꯃꯃꯤꯡ",
+       "yourname": "ꯁꯤꯖꯤꯟꯅꯔꯤꯕꯥ ꯃꯃꯤꯡ:",
        "userlogin-yourname": "ꯁꯤꯖꯤꯟꯅꯔꯤꯕ ꯃꯃꯤꯡ",
        "userlogin-yourname-ph": "ꯅꯪꯒꯤ ꯁꯤꯖꯤꯟꯅꯔꯤꯕ ꯃꯃꯤꯡ ꯏꯌꯨ",
        "createacct-another-username-ph": "ꯁꯤꯖꯤꯟꯅꯔꯤꯕꯥ ꯃꯃꯤꯡ ꯗꯨ ꯏꯁꯤꯟꯂꯣ",
-       "yourpassword": "ꯆꯪꯁꯤꯟꯅꯕꯥ ꯋꯥꯍꯩ",
+       "yourpassword": "ꯆꯪꯁꯤꯟꯅꯕꯥ ꯋꯥꯍꯩ:",
        "userlogin-yourpassword": "ꯆꯪꯁꯤꯟꯅꯕꯥ ꯋꯥꯍꯩ",
        "userlogin-yourpassword-ph": "ꯄꯥꯁꯋ꯭ꯔꯗ ꯏꯔꯛ ꯎ",
        "createacct-yourpassword-ph": "ꯄꯥꯁꯋ꯭ꯔꯗ ꯏꯔꯛ ꯎ",
-       "yourpasswordagain": "ꯑꯃꯨꯛꯍꯟꯅꯥ ꯄꯥꯁꯋ꯭ꯔꯗ ꯅꯝꯃꯨ",
+       "yourpasswordagain": "ꯑꯃꯨꯛꯍꯟꯅꯥ ꯄꯥꯁꯋ꯭ꯔꯗ ꯅꯝꯃꯨ:",
        "createacct-yourpasswordagain": "Confirm password",
        "createacct-yourpasswordagain-ph": "ꯑꯃꯨꯛ ꯍꯟꯅꯥ ꯄꯥꯁꯋ꯭ꯇ ꯏꯌꯨ",
        "userlogin-remembermypassword": "ꯑꯩꯕꯨ ꯑꯗꯨꯝ ꯂꯣꯒ ꯏꯟ ꯇꯧꯍꯟꯂꯨ",
        "yourdomainname": "ꯅꯪꯒꯤ ꯗꯣꯃꯦꯟ:",
        "password-change-forbidden": "ꯅꯪꯅꯥ ꯃꯁꯤꯒꯤ ꯋꯤꯀꯤ ꯁꯤꯗꯥ ꯄꯥꯁꯋ꯭ꯔꯗ ꯍꯣꯡꯕꯥ ꯌꯥꯔꯣꯏ ꯫",
        "externaldberror": "There was either an authentication database error or you are not allowed to update your external account.",
-       "login": "Chang Sinba",
+       "login": "ꯆꯪꯁꯤꯟꯕ ꯃꯅꯨꯪꯗ",
        "login-security": "ꯅꯪꯒꯤ ꯁꯛꯑꯣꯡ ꯈꯟꯗꯣꯛꯄꯥ",
        "nav-login-createaccount": "ꯂꯣꯒ ꯏꯟ/ꯑꯦꯀꯥꯎꯟ ꯁꯥꯕꯥ",
        "logout": "ꯊꯣꯛꯂꯛꯄꯥ",
        "userlogin-noaccount": "ꯑꯦꯀꯥꯎꯟ ꯂꯩꯇꯕꯔꯥ?",
        "userlogin-joinproject": "{{SITENAME}} ꯌꯥꯎꯕ",
        "createaccount": "ꯑꯩꯒꯤ ꯑꯣꯏꯕꯥ ꯑꯃꯥ ꯁꯦꯝꯕꯥ",
-       "userlogin-resetpassword-link": "ê¯\84ꯥê¯\81ê¯\8bê¯\94ê¯\87 ê¯\80ꯥê¯\8eê¯\88ê¯\94ê¯\95ꯥ",
+       "userlogin-resetpassword-link": "ê¯\84ꯥê¯\81ê¯\8b꯭ê¯\94ꯠ ê¯\80ꯥê¯\8eê¯\88ê¯\94ê¯\95ꯥ?",
        "userlogin-helplink2": "ꯂꯣꯒꯒꯤꯡ ꯇꯧꯗꯨꯅꯥ ꯃꯇꯦꯡ ꯄꯥꯡ ꯎ",
        "userlogin-loggedin": "You are already logged in as {{GENDER:$1|$1}}.\nUse the form below to log in as another user.",
        "userlogin-reauth": "You must log in again to verify that you are {{GENDER:$1|$1}}.",
        "pt-login-continue-button": "ꯂꯣꯘ ꯏꯟ ꯃꯈꯥ ꯆꯠꯊꯧ",
        "pt-createaccount": "ꯑꯩꯒꯤ ꯑꯣꯏꯕꯥ ꯑꯃꯥ ꯁꯦꯝꯕꯥ",
        "pt-userlogout": "ꯊꯣꯛꯂꯛꯄꯥ",
-       "changepassword": "ê¯\84ꯥê¯\81ê¯\8b꯭ê¯\94ê¯\87 ꯍꯣꯡꯗꯣꯛꯄꯥ",
-       "oldpassword": "ê¯\91ê¯\94ꯤê¯\95ꯥ ê¯\84ꯥê¯\81ê¯\8b꯭ê¯\94ê¯\97:",
-       "newpassword": "ê¯\91ê¯\85ꯧê¯\95ꯥ ê¯\84ꯥê¯\81ê¯\8b꯭ê¯\94ê¯\97",
-       "retypenew": "ê¯\91ê¯\83ꯨê¯\9bê¯\8dê¯\9fê¯\85ꯥ ê¯\91ê¯\85ꯧê¯\95ꯥ  ê¯\84ꯥê¯\81ê¯\8b꯭ê¯\94ê¯\97ꯅꯝꯃꯨ:",
-       "resetpass_submit": "ê¯\84ꯥê¯\81ê¯\8bê¯\94ê¯\87 ê¯\81ꯦê¯\9dê¯\83ꯨ ê¯±ê¯\97ꯤ ê¯\82ꯣê¯\92 ê¯\8fê¯\9f",
-       "changepassword-success": "ê¯\85ꯪê¯\92ꯤ ê¯\84ꯥê¯\81ê¯\8bê¯\94ê¯\87 ê¯\81ꯤ ê¯\8dꯣꯡê¯\97ꯣê¯\9bê¯\88ê¯\94ꯦ",
+       "changepassword": "ê¯\84ꯥê¯\81ê¯\8b꯭ê¯\94ꯠ ꯍꯣꯡꯗꯣꯛꯄꯥ",
+       "oldpassword": "ê¯\91ê¯\94ꯤê¯\95ꯥ ê¯\84ꯥê¯\81ê¯\8b꯭ê¯\94ꯠ:",
+       "newpassword": "ê¯\91ê¯\85ꯧê¯\95ꯥ ê¯\84ꯥê¯\81ê¯\8b꯭ê¯\94ꯠ:",
+       "retypenew": "ê¯\91ê¯\83ꯨê¯\9bê¯\8dê¯\9fê¯\85ꯥ ê¯\91ê¯\85ꯧê¯\95ꯥ  ê¯\84ꯥê¯\81ê¯\8b꯭ê¯\94ꯠꯅꯝꯃꯨ:",
+       "resetpass_submit": "ê¯\84ꯥê¯\81ê¯\8b꯭ê¯\94ꯠ ê¯\81ꯦê¯\9dê¯\83ꯨ ê¯±ê¯\81ꯨꯡ ê¯\86ꯪê¯\81ꯤê¯\9fê¯\82ꯨ",
+       "changepassword-success": "ê¯\85ꯪê¯\92ꯤ ê¯\84ꯥê¯\81ê¯\8b꯭ê¯\94ꯠ ê¯\81ꯤ ê¯\8dꯣꯡê¯\97ꯣê¯\9bê¯\88ê¯\94ꯦ!",
        "changepassword-throttled": "You have made too many recent login attempts.\nPlease wait $1 before trying again.",
-       "botpasswords": "ê¯\95ꯣê¯\87 ê¯\84ꯥê¯\81ê¯\8bê¯\94ê¯\87",
+       "botpasswords": "ê¯\95ꯣꯠ ê¯\84ꯥê¯\81ê¯\8b꯭ê¯\94ꯠê¯\81ꯤꯡ",
        "botpasswords-summary": "<em>Bot passwords</em> allow access to a user account via the API without using the account's main login credentials. The user rights available when logged in with a bot password may be restricted.\n\nIf you don't know why you might want to do this, you should probably not do it. No one should ever ask you to generate one of these and give it to them.",
-       "botpasswords-disabled": "ê¯\95ꯣê¯\87 ê¯\84ꯥê¯\81ê¯\8bê¯\94ê¯\87 ê¯\8cꯥê¯\89ꯟꯗꯕꯥ",
+       "botpasswords-disabled": "ê¯\95ꯣꯠ ê¯\84ꯥê¯\81ê¯\8b꯭ê¯\94ꯠê¯\81ꯤꯡ ê¯\8cꯥê¯\8dꯟꯗꯕꯥ",
        "botpasswords-label-appid": "ꯕꯣꯠ ꯃꯃꯤꯡ:",
        "botpasswords-label-create": "ꯁꯥꯕꯥ",
        "botpasswords-label-update": "ꯅꯧꯊꯣꯛꯍꯟꯕꯥ",
        "botpasswords-label-cancel": "ꯀꯛꯊꯠꯄꯥ",
        "botpasswords-label-delete": "ꯃꯥꯡꯍꯟꯕꯥ",
-       "botpasswords-label-resetpassword": "ê¯\84ꯥê¯\81ê¯\8b꯭ê¯\87 ꯑꯃꯨꯛ ꯍꯟꯅꯥ ꯁꯦꯝꯗꯣꯛꯄꯥ",
-       "botpasswords-label-grants": "ꯆꯥꯟꯅꯕꯥ ꯌꯥꯕꯥ ꯑꯌꯥꯕ",
+       "botpasswords-label-resetpassword": "ê¯\84ꯥê¯\81ê¯\8b꯭ê¯\94ꯠ ꯑꯃꯨꯛ ꯍꯟꯅꯥ ꯁꯦꯝꯗꯣꯛꯄꯥ",
+       "botpasswords-label-grants": "ꯆꯥꯟꯅꯕꯥ ꯌꯥꯕꯥ ꯑꯌꯥꯕꯁꯤꯡ:",
        "botpasswords-help-grants": "Grants allow access to rights already held by your user account. Enabling a grant here does not provide access to any rights that your user account would not otherwise have. See the [[Special:ListGrants|table of grants]] for more information.",
-       "botpasswords-created-title": "ê¯\95ꯣꯠ ê¯\84ꯥê¯\81ê¯\8b꯭ê¯\94ê¯\87 ꯁꯥꯈꯔꯦ",
-       "botpasswords-deleted-title": "ê¯\95ꯣꯠ ê¯\84ꯥê¯\81ê¯\8b꯭ê¯\94ê¯\87 ꯀꯛꯊꯠꯈꯔꯦ",
-       "resetpass_forbidden": "ê¯\84ꯥê¯\81ê¯\8b꯭ê¯\94ê¯\87 ê¯\8dꯣꯡê¯\97ꯣꯧꯄꯥ ꯌꯥꯔꯣꯏ",
-       "resetpass_forbidden-reason": "$1 ꯄꯥꯁꯋ꯭ꯔꯇ ꯍꯣꯡꯗꯣꯧꯄꯥ ꯌꯥꯔꯣꯏ",
-       "resetpass-no-info": "ê¯\83ê¯\81ꯤê¯\92ꯤ ê¯\82ꯥê¯\83ꯥê¯\8fê¯\81ꯤ ê¯\8dꯦꯧê¯\87ꯥ ê¯\8cꯧê¯\85ê¯\95ꯥ ê¯\85ꯪ ê¯\81ꯣê¯\8fê¯\97ê¯\85ꯥ ê¯\82ꯣê¯\92 ê¯\8fê¯\9f ê¯\87ꯧê¯\92ê¯\97ê¯\95ê¯\85ꯤ",
-       "resetpass-submit-loggedin": "ê¯\84ꯥê¯\81ê¯\8b꯭ê¯\94ê¯\87 ꯍꯣꯡꯗꯣꯛꯄꯥ",
+       "botpasswords-created-title": "ê¯\95ꯣꯠ ê¯\84ꯥê¯\81ê¯\8b꯭ê¯\94ꯠ ꯁꯥꯈꯔꯦ",
+       "botpasswords-deleted-title": "ê¯\95ꯣꯠ ê¯\84ꯥê¯\81ê¯\8b꯭ê¯\94ꯠ ꯀꯛꯊꯠꯈꯔꯦ",
+       "resetpass_forbidden": "ê¯\84ꯥê¯\81ê¯\8b꯭ê¯\94ꯠ ê¯\8dꯣꯡê¯\97ꯣê¯\9bꯄꯥ ꯌꯥꯔꯣꯏ",
+       "resetpass_forbidden-reason": "$1:ꯄꯥꯁꯋ꯭ꯔꯠꯁꯤꯡ ꯍꯣꯡꯗꯣꯛꯄꯥ ꯌꯥꯔꯣꯏ",
+       "resetpass-no-info": "ê¯\83ê¯\81ꯤê¯\92ꯤ ê¯\82ê¯\83ꯥê¯\8fê¯\81ꯤ ê¯\8dꯦê¯\9bê¯\87ꯥ ê¯\8cꯧê¯\85ê¯\95ꯥ ê¯\85ꯪ ê¯\81ꯣê¯\8fê¯\97ê¯\85ꯥ ê¯\82ꯣê¯\92 ê¯\8fê¯\9f ê¯\87ꯧê¯\92ê¯\97ê¯\95ê¯\85ꯤ ê¯«",
+       "resetpass-submit-loggedin": "ê¯\84ꯥê¯\81ê¯\8b꯭ê¯\94ꯠ ꯍꯣꯡꯗꯣꯛꯄꯥ",
        "resetpass-submit-cancel": "ꯀꯛꯊꯠꯄꯥ",
-       "resetpass-wrong-oldpass": "ê¯\83ê¯\87ê¯\9d ê¯\88ê¯\94ꯥê¯\92ꯤ ê¯\91ꯣê¯\8fê¯\85ꯥ ê¯\80ê¯\94ꯤê¯\9dê¯\87ꯥ ê¯\8cꯥê¯\8eê¯\97ꯦ  ê¯\85ꯠê¯\87ê¯\94ê¯\92ꯥ ê¯\8dꯧê¯\96ꯤꯧê¯\80ꯤ ê¯\84ꯥê¯\81ê¯\8b꯭ê¯\94ê¯\87꯫\nê¯\85ꯪê¯\85ꯥ ê¯\84ꯥê¯\81ê¯\8bê¯\91ê¯\94ê¯\87 ê¯\8dꯥê¯\9fê¯\85ê¯\97ê¯\92ꯤ ê¯\8dꯣꯡê¯\82ê¯\9dê¯\82ê¯\85ꯤ ê¯\85ꯠê¯\87ê¯\94ê¯\92ꯥ ê¯\8dꯪê¯\92ꯠê¯\82ꯨ ê¯\89ꯥê¯\8fê¯\8dꯥê¯\9bê¯\80ꯤ ê¯\91ꯣê¯\8fê¯\95ꯥ ê¯\84ꯥê¯\81ê¯\8b꯭ê¯\94ê¯\87",
-       "resetpass-temp-password": "ê¯\89ꯩê¯\8dꯥê¯\9bê¯\80ꯤ ê¯\91ꯣê¯\8fê¯\95ꯥ ê¯\84ꯥê¯\81ê¯\8b꯭ê¯\94ê¯\87",
+       "resetpass-wrong-oldpass": "ê¯\83ê¯\87ê¯\9d ê¯\88ê¯\94ꯥê¯\92ꯤ ê¯\91ꯣê¯\8fê¯\85ꯥ ê¯\80ê¯\94ꯤê¯\9dê¯\87ꯥ ê¯\8cꯥê¯\8eê¯\97ꯦ  ê¯\85ꯠê¯\87ê¯\94ê¯\92ꯥ ê¯\8dꯧê¯\96ꯤê¯\9bê¯\80ꯤ ê¯\84ꯥê¯\81ê¯\8b꯭ê¯\94ꯠ ê¯« ê¯\85ꯪê¯\85ꯥ ê¯\84ꯥê¯\81ê¯\8b꯭ê¯\94ꯠ ê¯\8dꯥê¯\9fê¯\85ê¯\97ê¯\92ꯤ ê¯\8dꯣꯡê¯\82ê¯\9dê¯\82ê¯\85ꯤ ê¯\85ꯠê¯\87ê¯\94ê¯\92ꯥ ê¯\8dꯪê¯\92ꯠê¯\82ꯨ ê¯\89ꯥê¯\8fê¯\8dꯥê¯\9bê¯\80ꯤ ê¯\91ꯣê¯\8fê¯\95ꯥ ê¯\84ꯥê¯\81ê¯\8b꯭ê¯\94ꯠ ê¯«",
+       "resetpass-temp-password": "ê¯\89ꯩê¯\8dꯥê¯\9bê¯\80ꯤ ê¯\91ꯣê¯\8fê¯\95ꯥ ê¯\84ꯥê¯\81ê¯\8b꯭ê¯\94ꯠ:",
        "resetpass-expired": "ꯅꯪꯒꯤ ꯄꯥꯁꯋ꯭ꯔꯇ ꯁꯤ ꯌꯥꯗꯔꯦ ꯫ ꯆꯥꯟꯕꯤꯗꯨꯅ ꯑꯅꯧꯕ ꯱ ꯁꯦꯝꯃꯣ ꯂꯣꯒ ꯏꯟ ꯇꯧꯅꯕ ꯫",
-       "passwordreset": "ê¯\84ꯥê¯\81ê¯\8b꯭ê¯\87 ꯁꯦꯝꯗꯣꯛꯄꯥ",
-       "passwordreset-username": "ꯁꯤꯖꯤꯟꯅꯔꯤꯕꯥ ꯃꯃꯤꯡ",
-       "passwordreset-domain": "ꯗꯣꯃꯦꯟ",
-       "passwordreset-email": "ê¯\8fê¯\83ꯦê¯\9c ê¯\82ꯥê¯\90ê¯\9d",
-       "passwordreset-emailtitle": "{{SITENAME}} ꯑꯀꯨꯞꯄꯥ ꯃꯔꯣꯜ",
+       "passwordreset": "ê¯\84ꯥê¯\81ê¯\8b꯭ê¯\94ꯠ ꯁꯦꯝꯗꯣꯛꯄꯥ",
+       "passwordreset-username": "ꯁꯤꯖꯤꯟꯅꯔꯤꯕꯥ ꯃꯃꯤꯡ:",
+       "passwordreset-domain": "ꯗꯣꯃꯦꯟ:",
+       "passwordreset-email": "ê¯\8fê¯\83ꯦê¯\9c ê¯\82ꯩê¯\90ê¯\9d:",
+       "passwordreset-emailtitle": "{{SITENAME}} ꯑꯀꯨꯞꯄꯥ ꯃꯔꯣꯜ",
        "passwordreset-emailtext-ip": "Someone (probably you, from IP address $1) requested a reset of your\npassword for {{SITENAME}} ($4). The following user {{PLURAL:$3|account is|accounts are}}\nassociated with this email address:\n\n$2\n\n{{PLURAL:$3|This temporary password|These temporary passwords}} will expire in {{PLURAL:$5|one day|$5 days}}.\nYou should log in and choose a new password now. If someone else made this\nrequest, or if you have remembered your original password, and you no longer\nwish to change it, you may ignore this message and continue using your old\npassword.",
        "passwordreset-emailtext-user": "User $1 on {{SITENAME}} requested a reset of your password for {{SITENAME}}\n($4). The following user {{PLURAL:$3|account is|accounts are}} associated with this email address:\n\n$2\n\n{{PLURAL:$3|This temporary password|These temporary passwords}} will expire in {{PLURAL:$5|one day|$5 days}}.\nYou should log in and choose a new password now. If someone else made this\nrequest, or if you have remembered your original password, and you no longer\nwish to change it, you may ignore this message and continue using your old\npassword.",
-       "passwordreset-emailelement": "$1 ê¯\81ꯤê¯\96ꯤê¯\9fê¯\85ê¯\94ꯤê¯\95ꯥ\n$2 ê¯\89ꯩê¯\8dꯥê¯\9bê¯\80ꯤ ê¯\91ꯣê¯\8fê¯\95ꯥ ê¯\84ꯥê¯\81ê¯\8bê¯\94ê¯\87",
+       "passwordreset-emailelement": "$1 ê¯\81ꯤê¯\96ꯤê¯\9fê¯\85ê¯\94ꯤê¯\95ꯥ\n$2 ê¯\89ꯩê¯\8dꯥê¯\9bê¯\80ꯤ ê¯\91ꯣê¯\8fê¯\95ꯥ ê¯\84ꯥê¯\81ê¯\8bê¯\94ꯠ",
        "changeemail-oldemail": "ꯍꯧꯖꯤꯛꯀꯤ ꯏꯃꯦꯜ ꯑꯦꯗ꯭ꯔꯦꯁ:",
        "changeemail-newemail": "ꯑꯅꯧꯕ ꯏꯃꯦꯜ ꯑꯦꯗ꯭ꯔꯦꯁ:",
-       "changeemail-none": "ꯑꯃꯥꯇꯥ ꯅꯠꯇꯦ",
-       "changeemail-password": "ê¯\85ꯪê¯\92ꯤ {{SITENAME}} ê¯\84ꯥê¯\81ê¯\8b꯭ê¯\94ê¯\97:",
+       "changeemail-none": "(ꯑꯃꯥꯇꯥ ꯅꯠꯇꯦ)",
+       "changeemail-password": "ê¯\85ꯪê¯\92ꯤ {{SITENAME}} ê¯\84ꯥê¯\81ê¯\8b꯭ê¯\94ꯠ:",
        "changeemail-submit": "ꯏ-ꯃꯦꯜ ꯍꯣꯡꯕꯥ",
        "bold_sample": "ꯆꯥꯎꯅꯥ ꯏꯕꯥ",
        "bold_tip": "ꯆꯥꯎꯅꯥ ꯏꯕꯥ",
        "link_sample": "ꯑꯄꯤꯕ ꯃꯃꯤꯡ ꯁꯝꯅꯐꯝ",
        "link_tip": "ꯃꯅꯨꯡꯒꯥ ꯁꯝꯅꯐꯝ",
        "extlink_sample": "http://www.example.com link title",
-       "extlink_tip": "ꯑꯇꯣꯞꯄꯥꯒꯥ ꯁꯝꯅꯐꯝ",
+       "extlink_tip": "ꯑꯇꯣꯞꯄꯥꯒꯥ ꯁꯝꯅꯐꯝ (ꯀꯥꯎꯒꯅꯨ http:// prefix)",
        "headline_sample": "ꯃꯀꯣꯛꯀꯤ ꯋꯥꯔꯦꯡ ꯄꯥꯔꯦꯡ",
        "headline_tip": "꯲ ꯁꯨꯕꯥ ꯃꯀꯣꯛꯀꯤ ꯄꯔꯦꯡ",
        "nowiki_sample": "ꯍꯥꯞꯆꯤꯟꯂꯨ non formating ꯋꯥꯔꯦꯡꯗꯨ ꯁꯤꯗꯥ",
        "anoneditwarning": "<strong>Warning:</strong> ꯅꯪ ꯃꯅꯨꯡ ꯆꯪꯗꯔꯤ꯬꯬ ꯫ Your IP address will be publicly visible if you make any edits. If you <strong>[$1 log in]</strong> or <strong>[$2 create an account]</strong>, your edits will be attributed to your username, along with other benefits.",
        "loginreqlink": "Chang Sinba",
        "accmailtitle": "ꯄꯥꯁꯋ꯭ꯔꯇ ꯊꯥꯕ",
-       "newarticle": "ꯑꯅꯧꯕꯥ",
+       "newarticle": "(ꯑꯅꯧꯕꯥ)",
        "newarticletext": "You have followed a link to a page that does not exist yet.\nTo create the page, start typing in the box below (see the [$1 help page] for more info).\nIf you are here by mistake, click your browser's <strong>ꯍꯟꯕ</strong> button.",
        "anontalkpagetext": "----\n<em>This is the discussion page for an anonymous user who has not created an account yet, or who does not use it.</em>\nWe therefore have to use the numerical IP address to identify him/her.\nSuch an IP address can be shared by several users.\nIf you are an anonymous user and feel that irrelevant comments have been directed at you, please [[Special:CreateAccount|create an account]] or [[Special:UserLogin|log in]] to avoid future confusion with other anonymous users.",
        "noarticletext": "There is currently no text in this page.\nYou can [[Special:Search/{{PAGENAME}}|search for this page title]] in other pages,\n<span class=\"plainlinks\">[{{fullurl:{{#Special:Log}}|page={{FULLPAGENAMEE}}}} search the related logs],\nor [{{fullurl:{{FULLPAGENAME}}|action=edit}} create this page]</span>.",
        "missing-revision": "The revision #$1 of the page named \"{{FULLPAGENAME}}\" does not exist.\n\nThis is usually caused by following an outdated history link to a page that has been deleted.\nDetails can be found in the [{{fullurl:{{#Special:Log}}/delete|page={{FULLPAGENAMEE}}}} deletion log].",
        "userpage-userdoesnotexist-view": "$1 ꯁꯤꯖꯤꯟꯅꯔꯤꯕ ꯑꯦꯀꯥꯎꯅ ꯁꯤ ꯃꯤꯡ ꯆꯟꯗꯔꯤ ꯫",
        "updated": "(ꯅꯧꯊꯣꯛꯍꯟꯂꯦ)",
-       "note": "<ꯑꯀꯟꯕ>ꯏꯁꯤꯟꯒꯗꯕ:</ꯑꯀꯟꯕ>",
+       "note": "<strong>ꯏꯁꯤꯟꯒꯗꯕ:</strong>",
        "continue-editing": "ꯁꯦꯝꯒꯠꯄꯒꯤ ꯃꯐꯝꯗꯨꯗꯥ ꯆꯠꯂꯨ",
        "editing": "$1 ꯁꯦꯝꯒꯠꯂꯤ",
        "creating": "ꯁꯥꯔꯤ $1",
        "editingsection": "ꯏꯔꯤꯕ $1 (ꯁꯥꯔꯨꯛ)",
-       "yourtext": "ê¯\85ꯪê¯\92ꯤ ê¯\87ꯦê¯\80ê¯\81",
-       "yourdiff": "ꯈꯦꯠꯅꯕꯥ ꯁꯤꯡ",
+       "yourtext": "ê¯\85ꯪê¯\92ꯤ ê¯\8bꯥê¯\8dꯩ ê¯\8bꯥê¯\87ꯥ",
+       "yourdiff": "ꯈꯦꯠꯅꯕꯥꯁꯤꯡ",
        "copyrightwarning": "Please note that all contributions to {{SITENAME}} are considered to be released under the $2 (see $1 for details).\nIf you do not want your writing to be edited mercilessly and redistributed at will, then do not submit it here.<br />\nYou are also promising us that you wrote this yourself, or copied it from a public domain or similar free resource.\n<strong>Do not submit copyrighted work without permission!</strong>",
        "templatesused": "ꯃꯁꯤꯒꯤ ꯂꯃꯥꯏꯁꯤꯗ ꯁꯤꯖꯤꯟꯅꯕ {{PLURAL:$1|ꯇꯦꯝꯄꯂꯦꯠ|ꯇꯦꯝꯄꯂꯦꯠꯁꯤꯡ}}:",
        "templatesusedpreview": "{{PLURAL:$1|ꯇꯦꯝꯄꯂꯦꯠ|ꯇꯦꯝꯄꯂꯦꯠꯁꯤꯡ}} ꯄꯔꯤꯕꯤꯌꯨ ꯗ ꯁꯤꯖꯤꯟꯅꯕ:",
        "hiddencategories": "This page is a member of {{PLURAL:$1|1 hidden category|$1 hidden categories}}:",
        "permissionserrors": "ꯑꯌꯥꯕꯥꯗꯨ ꯁꯣꯏꯔꯦ",
        "permissionserrorstext-withaction": "$2 ꯗ ꯅꯪꯒꯤ ꯑꯌꯥꯕ ꯂꯩꯇꯦ, ꯃꯈꯥꯒꯤ {{PLURAL:$1|ꯃꯔꯝ|ꯃꯔꯝꯁꯤꯡ}} ꯃꯇꯨꯡ ꯏꯟꯅꯥ:",
-       "moveddeleted-notice": "ê¯\83ê¯\81ꯤê¯\92ꯤ ê¯\82ꯥê¯\83ꯥê¯\8fê¯\81ꯤ ê¯\80ê¯\9bê¯\8aꯠê¯\88ê¯\94ꯦ. \nê¯\80ê¯\9bê¯\8aꯠê¯\84ꯥ, ê¯\89ꯥê¯\9bê¯\8aꯣê¯\9bê¯\84ꯥ ê¯\91ê¯\83ê¯\97ꯤ ê¯\82ꯣê¯\92 ê¯\82ꯦꯡê¯\8dê¯\9fê¯\95ꯥ ê¯\82ꯥê¯\83ꯥê¯\8fê¯\92ꯤê¯\97ê¯\83ê¯\9b ê¯\87ꯨ ê¯\83ê¯\88ꯥê¯\92ꯤ ê¯\81ꯤê¯\97ꯥ ê¯\94ꯤê¯\90ê¯\94ꯦê¯\9fê¯\81 ê¯\8eꯨꯠê¯\82ꯦ",
-       "edit-conflict": "ꯁ‍ꯦꯝꯒꯠꯐꯝꯒꯤ ꯈꯠꯅ ꯆꯩꯅꯕꯥ",
+       "moveddeleted-notice": "ê¯\83ê¯\81ꯤê¯\92ꯤ ê¯\82ê¯\83ꯥê¯\8fê¯\81ꯤ ê¯\80ê¯\9bê¯\8aꯠê¯\88ê¯\94ꯦ ê¯«\nê¯\80ê¯\9bê¯\8aꯠê¯\84ꯥ, ê¯\89ꯥê¯\9bê¯\8aꯣê¯\9bê¯\84ꯥ ê¯\91ê¯\83ê¯\81ꯨꯡ ê¯\82ꯣê¯\92 ê¯\82ꯦꯡê¯\8dê¯\9fê¯\95ꯥ ê¯\82ê¯\83ꯥê¯\8fê¯\92ꯤê¯\97ê¯\83ê¯\9b ê¯\87ꯨ ê¯\83ê¯\88ꯥê¯\92ꯤ ê¯\81ꯤê¯\97ꯥ  ê¯\8eꯨꯠê¯\82ꯦ ê¯«",
+       "edit-conflict": "ꯁ‍ꯦꯝꯒꯠꯐꯝꯒꯤ ꯈꯠꯅ ꯆꯩꯅꯕꯥ ꯫",
        "postedit-confirmation-created": "ꯂꯃꯥꯏ ꯑꯁꯤ ꯁꯥꯕ ꯂꯣꯏꯈꯔꯦ ꯫",
        "postedit-confirmation-restored": "ꯂꯃꯥꯏ ꯑꯁꯤ ꯍꯥꯟꯅꯒꯤ ꯑꯖꯎꯃꯥꯏꯅꯥ ꯆꯞ ꯆꯥꯅꯥ ꯂꯩꯔꯦ ꯫",
        "postedit-confirmation-saved": "ꯅꯪꯒꯤ ꯁꯦꯝꯒꯠꯄꯗꯨ ꯇꯨꯡꯁꯤꯟꯈꯔꯦ ꯫",
        "revision-info": " $1 ꯒꯤ ꯑꯃꯨꯛꯌꯦꯡꯕ {{GENDER:$6|$2}}$7 ꯅꯥ",
        "previousrevision": "← ꯑꯔꯤꯕꯥ ꯑꯃꯨꯛ ꯍꯟꯅꯥ ꯌꯦꯡꯕꯥ",
        "nextrevision": "ꯑꯅꯧꯕꯥ ꯑꯃꯨꯛꯍꯟꯅꯥ ꯌꯦꯡꯕꯥ →",
-       "currentrevisionlink": "ꯈꯋꯥꯏꯗꯒꯤ ꯅꯧꯅꯥ ꯑꯃꯨꯛ ꯌꯦꯡꯕꯥ",
+       "currentrevisionlink": "ê¯\88ꯨê¯\8bꯥê¯\8fê¯\97ê¯\92ꯤ ê¯\85ꯧê¯\85ꯥ ê¯\91ê¯\83ꯨê¯\9b ê¯\8cꯦꯡê¯\95ꯥ",
        "cur": "ꯍꯧ",
        "next": "ꯃꯊꯪ",
        "last": "ꯃꯃꯥꯡꯒꯤ",
        "page_first": "ꯑꯍꯥꯟꯕ",
        "page_last": "ꯑꯔꯣꯏꯕ",
        "histlegend": "Diff selection: Mark the radio boxes of the revisions to compare and hit enter or the button at the bottom.<br />\nLegend: <strong>({{int:cur}})</strong> = difference with latest revision, <strong>({{int:last}})</strong> = difference with preceding revision, <strong>{{int:minoreditletter}}</strong> = ꯑꯄꯤꯛꯄ ꯁꯦꯝꯒꯠꯄ",
-       "history-fieldset-title": "ê¯\8aꯤê¯\8bꯨ ê¯\91ê¯\83ꯨê¯\9b ê¯\8dê¯\9dê¯\81ê¯\9fê¯\85ꯥ ê¯\8cꯦꯡê¯\85ê¯\95ꯥ",
-       "histfirst": "ꯈꯋꯥꯏꯗꯒꯤ ꯑꯔꯤꯕꯥ",
+       "history-fieldset-title": "ê¯\94ꯤê¯\9aꯤê¯\96ê¯\9fê¯\81ꯤꯡ ê¯\87ꯦꯡê¯\8aꯣê¯\9bê¯\84ꯥ",
+       "histfirst": "ê¯\88ꯨê¯\8bꯥê¯\8fê¯\97ê¯\92ꯤ ê¯\91ê¯\94ꯤê¯\95ꯥ",
        "histlast": "ꯑꯅꯧꯕꯥ",
        "historyempty": "(ꯑꯍꯥꯡꯕ)",
        "history-feed-title": "ꯄꯨꯋꯥꯔꯤ ꯑꯃꯨꯛ ꯍꯟꯅ ꯌꯦꯡꯕ",
        "mergehistory-fail-invalid-dest": "ꯂꯝꯊꯨꯡꯐꯝ ꯂꯃꯥꯏꯁꯤ ꯂꯩꯇꯦ ꯫",
        "mergehistory-reason": "ꯃꯔꯝ:",
        "mergelog": "ꯂꯣꯒ ꯄꯨꯟꯁꯤꯟꯕ",
-       "history-title": "Revision history of \"$1\"",
+       "history-title": "\"$1\" ꯒꯤ ꯔꯤꯚꯤꯖꯟ ꯄꯨꯋꯥꯔꯤ",
        "difference-title": "$1 ꯒꯤ ꯑꯃꯨꯛꯍꯟꯕꯥ ꯈꯦꯠꯅꯕꯥꯒꯤ ꯃꯔꯛ",
-       "lineno": "ꯂꯩ ꯏ $1",
+       "lineno": "ꯂꯩ ꯏ $1:",
        "compareselectedversions": "ꯈꯟꯒꯠꯂꯕ ꯁꯤꯡ ꯑꯃꯨꯛ ꯍꯟꯅ ꯌꯦꯡꯕꯗꯨ ꯆꯥꯡꯗꯝꯅꯧ",
        "editundo": "ꯇꯧꯒꯅꯨ",
        "diff-empty": "(ꯈꯩꯠꯅꯕ ꯂꯩꯇꯦ)",
        "searchrelated": "ꯃꯔꯤꯂꯩꯅꯔꯦ",
        "searchall": "ꯄꯨꯂꯞ",
        "search-showingresults": "{{PLURAL:$4|Result <strong>$1</strong> of <strong>$3</strong>|Results <strong>$1 – $2</strong> of <strong>$3</strong>}}",
-       "search-nonefound": "ꯃꯁꯤꯒꯤ ꯐꯣꯜꯁꯤꯒꯥ ꯆꯥꯟꯅꯕꯥ ꯂꯩꯇꯦ",
+       "search-nonefound": "ꯃꯁꯤꯒꯤ ꯐꯣꯜꯁꯤꯒꯥ ꯆꯥꯟꯅꯕꯥ ꯂꯩꯇꯦ ꯫",
        "powersearch-legend": "ꯈꯨꯃꯥꯡ ꯆꯥꯎꯁꯤꯟꯅ ꯊꯤꯕꯥ",
        "powersearch-togglelabel": "ꯑꯁꯣꯏ ꯑꯔꯥꯟ ꯌꯥꯎꯕꯔ ꯌꯦꯡꯕ:",
        "powersearch-toggleall": "ꯄꯨꯂꯞ",
        "tooltip-t-contributions": " {{GENDER:$1|ꯃꯁꯤꯒꯤ ꯁꯤꯖꯤꯟꯅꯔꯤꯕ}} ꯑꯁꯤ ꯅꯥ ꯈꯣꯝꯖꯤꯟꯂꯛꯂꯤꯕꯥ ꯄꯥꯔꯦꯡ ꯱",
        "tooltip-t-upload": "ꯐꯥꯏꯜꯁꯤꯡ ꯊꯥꯒꯠꯂꯨ",
        "tooltip-t-specialpages": "ꯑꯈꯟꯅꯕ ꯂꯥꯃꯥꯏꯁꯤꯡꯒꯤ ꯄꯥꯔꯦꯡ ꯱",
-       "tooltip-t-print": "Namba Yaba ma ong  gi Lamai",
+       "tooltip-t-print": "ꯅꯝꯕ ꯌꯥꯕ ꯃꯑꯣꯡꯒꯤ ꯂꯃꯥꯏ",
        "tooltip-t-permalink": "Amuk han na yengba lamaisigi Lengdaba Samnafam",
        "tooltip-ca-nstab-main": "ꯂꯃꯥꯏꯁꯤꯒꯤ ꯑꯌꯥꯎꯕꯁꯤꯡꯗꯨ ꯎꯨꯇꯂꯨ",
        "tooltip-ca-nstab-user": "ꯁꯤꯖꯤꯟꯅꯔꯤꯕꯥ ꯂꯥꯃꯥꯏꯁꯤ ꯌꯦꯡꯕꯥ",
        "namespacesall": "ꯄꯨꯂꯞ",
        "monthsall": "ꯄꯨꯂꯞ",
        "imgmultipagenext": "ꯃꯊꯪ ꯂꯃꯥꯏ →",
-       "imgmultigo": "ꯆꯠꯂꯨ",
+       "imgmultigo": "ꯆꯠꯂꯨ!",
        "imgmultigoto": "$1 ꯂꯃꯥꯏ ꯗ ꯆꯠꯂꯨ",
        "watchlisttools-clear": "ꯌꯦꯡꯂꯤꯕ ꯄꯥꯔꯦꯡꯗꯨ ꯁꯦꯡꯗꯣꯛ ꯨꯎ",
        "watchlisttools-view": "ꯃꯁꯤꯒ ꯆꯥꯟꯅꯕ ꯑꯍꯣꯡꯕꯗꯨ ꯎꯨꯠꯂꯨ",
index 8b4341b..065918a 100644 (file)
        "rcfilters-clear-all-filters": "Kosongkan semua penapis",
        "rcfilters-show-new-changes": "Lihat perubahan baru sejak $1",
        "rcfilters-search-placeholder": "Penapis perubahan (guna menu atau carian untuk menapis nama)",
+       "rcfilters-search-placeholder-mobile": "Penapis",
        "rcfilters-invalid-filter": "Penapis tidak sah",
        "rcfilters-empty-filter": "Tiada penapis aktif. Semua sumbangan ditunjukkan.",
        "rcfilters-filterlist-title": "Penapis",
        "wlshowhideliu": "pengguna berdaftar",
        "wlshowhideanons": "pengguna awanama",
        "wlshowhidemine": "suntingan saya",
+       "wlshowhidecategorization": "pengkategorian laman",
        "watchlist-options": "Pilihan senarai pantau",
        "watching": "Memantau...",
        "unwatching": "Menyahpantau...",
        "deleteprotected": "Anda tidak boleh menghapuskan laman ini kerana ia telah dilindungi.",
        "deleting-backlinks-warning": "'''Amaran:''' [[Special:WhatLinksHere/{{FULLPAGENAME}}|Terdapat laman lain]] yang berpaut atau bertransklusi dengan laman yang hendak anda hapus ini.",
        "rollback": "Undurkan suntingan.",
+       "rollback-confirmation-no": "Batal",
        "rollbacklink": "undur",
        "rollbacklinkcount": "mengundurkan $1 {{PLURAL:$1|suntingan}}",
        "rollbacklinkcount-morethan": "mengundurkan lebih daripada $1 {{PLURAL:$1|suntingan}}",
        "ipb-disableusertalk": "Halang pengguna ini daripada menyunting laman perbincangan sendiri apabila disekat",
        "ipb-change-block": "Sekat semula pengguna tersebut dengan tetapan ini",
        "ipb-confirm": "Sahkan sekatan",
+       "ipb-namespaces-label": "Ruang nama",
        "badipaddress": "Alamat IP tidak sah",
        "blockipsuccesssub": "Sekatan berjaya",
        "blockipsuccesstext": "[[Special:Contributions/$1|$1]] telah disekat.\n<br />Sila lihat [[Special:BlockList|senarai sekatan]] untuk menyemak sekatan.",
        "ipb-unblock": "Nyahsekat nama pengguna atau alamat IP",
        "ipb-blocklist": "Lihat sekatan sedia ada",
        "ipb-blocklist-contribs": "Sumbangan oleh {{GENDER:$1|$1}}",
+       "ipb-blocklist-duration-left": "$1 tinggal",
        "block-expiry": "Tamat:",
+       "block-target": "Nama pengguna atau alamat IP:",
        "unblockip": "Nyahsekat pengguna",
        "unblockiptext": "Gunakan borang di bawah untuk membuang sekatan bagialamat IP atau nama pengguna yang telah disekat.",
        "ipusubmit": "Tarik balik sekatan ini",
        "blocklist-userblocks": "Sorokkan sekatan akaun",
        "blocklist-tempblocks": "Sorokkan sekatan sementara",
        "blocklist-addressblocks": "Sorokkan sekatan IP tunggal",
+       "blocklist-type": "Jenis:",
+       "blocklist-type-opt-all": "Semua",
        "blocklist-rangeblocks": "Sorokkan sekatan julat",
        "blocklist-timestamp": "Cop masa",
        "blocklist-target": "Sasaran",
        "createaccountblock": "pembukaan akaun baru disekat",
        "emailblock": "e-mail disekat",
        "blocklist-nousertalk": "tidak boleh menyunting laman perbincangan sendiri",
+       "blocklist-editing": "menyunting",
+       "blocklist-editing-page": "laman",
+       "blocklist-editing-ns": "ruang nama",
        "ipblocklist-empty": "Senarai sekatan adalah kosong.",
        "ipblocklist-no-results": "Alamat IP atau nama pengguna tersebut tidak disekat.",
        "blocklink": "sekat",
        "cant-move-to-user-page": "Anda tidak mempunyai keizinan untuk memindahkan sesebuah laman ke mana-mana laman pengguna (kecuali sebagai sublamannya sahaja).",
        "cant-move-category-page": "Anda tidak mempunyai kebenaran untuk memindah laman-laman kategori.",
        "cant-move-to-category-page": "Anda tidak mempunyai kebenaran untuk memindah sebuah laman ke sebuah laman kategori.",
+       "namespace-nosubpages": "Ruang nama \"$1\" tidak membenarkan sublaman.",
        "newtitle": "Tajuk baru:",
        "move-watch": "Pantau laman ini",
        "movepagebtn": "Pindahkan laman",
        "pagemovedsub": "Pemindahan berjaya",
+       "cannotmove": "Laman tidak dapat dipindahkan, atas {{PLURAL:$1|sebab|sebab-sebab}} berikut:",
        "movepage-moved": "'''\"$1\" telah dipindahkan ke \"$2\"'''",
        "movepage-moved-redirect": "Satu lencongan telah diwujudkan.",
        "movepage-moved-noredirect": "Penciptaan lencongan telah dihalang.",
        "export-download": "Simpan sebagai fail",
        "export-templates": "Sertakan templat",
        "export-pagelinks": "Sertakan laman-laman yang dipaut sedalam:",
+       "export-manual": "Tambah laman secara manual:",
        "allmessages": "Pesanan sistem",
        "allmessagesname": "Nama",
        "allmessagesdefault": "Teks mesej asal",
        "import-mapping-namespace": "Import ke ruang nama:",
        "import-mapping-subpage": "Import sebagai sublaman bagi laman berikut:",
        "import-upload-filename": "Nama fail:",
+       "import-upload-username-prefix": "Awalan interwiki:",
        "import-comment": "Komen:",
        "importtext": "Sila eksport fail daripada sumber wiki dengan menggunakan [[Special:Export|utiliti eksport]].\nSimpan dalam komputer anda dan muat naiknya di sini.",
        "importstart": "Mengimport laman...",
        "pageinfo-display-title": "Tajuk paparan",
        "pageinfo-default-sort": "Kunci isih azali",
        "pageinfo-length": "Kepanjangan halaman (bait)",
+       "pageinfo-namespace": "Ruang nama",
        "pageinfo-article-id": "ID halaman",
        "pageinfo-language": "Bahasa isi kandungan halaman",
        "pageinfo-content-model": "Model kandungan halaman",
        "pageinfo-category-pages": "Bilangan halaman",
        "pageinfo-category-subcats": "Bilangan subkategori",
        "pageinfo-category-files": "Bilangan fail",
+       "pageinfo-user-id": "ID pengguna",
+       "pageinfo-file-hash": "Nilai cincangan",
        "markaspatrolleddiff": "Tanda ronda",
        "markaspatrolledtext": "Tanda ronda laman ini",
        "markedaspatrolled": "Tanda ronda",
        "newimages-summary": "Laman khas ini memaparkan senarai fail muat naik terakhir.",
        "newimages-legend": "Penapis",
        "newimages-label": "Nama fail (atau sebahagian daripadanya):",
+       "newimages-user": "Alamat IP atau nama pengguna",
        "newimages-showbots": "Paparkan muat naik oleh bot",
+       "newimages-mediatype": "Jenis media:",
        "noimages": "Tiada imej.",
        "ilsubmit": "Cari",
        "bydate": "mengikut tarikh",
        "version-specialpages": "Laman khas",
        "version-parserhooks": "Penyangkuk penghurai",
        "version-variables": "Pemboleh ubah",
+       "version-editors": "Penyunting",
        "version-antispam": "Pencegahan spam",
        "version-other": "Lain-lain",
        "version-mediahandlers": "Pengelola media",
        "htmlform-cloner-create": "Tambah lebih",
        "htmlform-cloner-delete": "Buang",
        "htmlform-cloner-required": "Sekurang-kurangnya satu nilai diperlukan.",
+       "htmlform-date-placeholder": "TTTT-BB-HH",
+       "htmlform-time-placeholder": "JJ:MM:SS",
+       "htmlform-datetime-placeholder": "TTTT-BB-HH JJ:MM:SS",
+       "htmlform-title-not-exists": "$1 tidak wujud.",
+       "htmlform-user-not-exists": "<strong>$1</strong> tidak wujud.",
+       "htmlform-user-not-valid": "<strong>$1</strong> bukan nama pengguna yang sah.",
        "logentry-delete-delete": "$1 telah {{GENDER:$2|menghapuskan}} laman $3",
        "logentry-delete-restore": "$1 telah {{GENDER:$2|memulihkan}} laman $3 ($4)",
        "logentry-delete-event": "$1 telah {{GENDER:$2|mengubah}} keterlihatan $5 peristiwa log di $3: $4",
        "pagelang-language": "Bahasa",
        "pagelang-use-default": "Gunakan bahasa asli",
        "pagelang-select-lang": "Pilih bahasa",
+       "pagelang-reason": "Sebab",
        "right-pagelang": "Mengubah bahasa laman",
        "action-pagelang": "mengubah bahasa laman",
        "log-name-pagelang": "Log perubahan bahasa",
        "mediastatistics-header-text": "Tekstual",
        "mediastatistics-header-executable": "Fail boleh laksana",
        "mediastatistics-header-archive": "Format mampat",
+       "mediastatistics-header-total": "Semua fail",
        "json-warn-trailing-comma": "$1 koma pengekor telah digugurkan dari JSON",
        "json-error-unknown": "Terdapat masalah dengan JSON. Ralat: $1",
        "json-error-depth": "Kedalaman tindakan maksimum telah dicecah",
        "mw-widgets-dateinput-no-date": "Tarik belum dipilih",
        "mw-widgets-titleinput-description-new-page": "laman belum wujud",
        "mw-widgets-titleinput-description-redirect": "melencong ke $1",
-       "randomrootpage": "Laman akar rawak"
+       "date-range-from": "Dari tarikh:",
+       "randomrootpage": "Laman akar rawak",
+       "log-action-filter-import": "Jenis import:",
+       "log-action-filter-all": "Semua",
+       "log-action-filter-protect-protect": "Perlindungan",
+       "log-action-filter-upload-upload": "Muat naik baru",
+       "authmanager-email-label": "E-mel",
+       "authmanager-email-help": "Alamat e-mel",
+       "authmanager-realname-label": "Nama sebenar",
+       "authmanager-realname-help": "Nama sebenar pengguna",
+       "credentialsform-account": "Nama akaun:",
+       "edit-error-short": "Ralat: $1",
+       "edit-error-long": "Ralat:\n\n$1",
+       "pagedata-title": "Data laman",
+       "pagedata-bad-title": "Tajuk tidak sah: $1.",
+       "passwordpolicies-group": "Kumpulan",
+       "passwordpolicies-policies": "Dasar-dasar"
 }
index 496bb0c..4baf25a 100644 (file)
        "systemblockedtext": "Ditt brukernavn eller IP-adresse har blitt blokkert automatisk av MediaWiki.\n\nBlokkeringen grunnes:\n\n:<em>$2</em>\n\n* Blokkeringen startet: $8\n* Blokkeringen gjelder til: $6\n* Blokkeringen er ment for: $7\n\nDin nåværende IP-adresse er $3.\nVennligst inkluder informasjonen over i alle spørsmål du spør angående dette.",
        "blockednoreason": "ingen grunn gitt",
        "blockedtext-composite": "<strong>Brukernavnet ditt eller IP-adressa di har blitt blokkert.</strong>\n\nBlokkeringen grunnes:\n\n:<em>$2</em>\n\n* Blokkeringen startet: $8\n* Blokkeringen løper ut: $6\n\n* $5\n\nIP-adressa di er $3.\nVennligst inkluder alle detaljene ovenfor i spørsmål du måtte ha angående dette.",
+       "blockedtext-composite-ids": "Relevante blokk-id-er: $1 (din IP-adresse kan også bli svartelistet)",
+       "blockedtext-composite-no-ids": "Din IP-adresse finnes i flere svartelister",
        "blockedtext-composite-reason": "Det foreligger flere blokkeringer på din konto og/eller IP-adresse",
        "whitelistedittext": "Du må $1 for å redigere artikler.",
        "confirmedittext": "Du må bekrefte e-postadressen din før du kan redigere sider. Vennligst oppgi og bekreft e-postadressen din via [[Special:Preferences|innstillingene dine]].",
        "search-interwiki-more": "(mer)",
        "search-interwiki-more-results": "flere resultater",
        "search-relatedarticle": "Relatert",
+       "search-invalid-sort-order": "Sorteringsrekkefølge $1 er ukjent, så standard sortering blir brukt. Lovlige sorteringsrekkefølger er: $2",
        "searchrelated": "relatert",
        "searchall": "alle",
        "showingresults": "Nedenfor vises opptil {{PLURAL:$1|'''ett''' resultat|'''$1''' resultater}} fra og med nummer <b>$2</b>.",
index 5ba3fbf..d47ff4c 100644 (file)
        "hidetoc": "लुकाउनुहोस्",
        "collapsible-collapse": "खुम्च्याउने",
        "collapsible-expand": "फैलाउ",
-       "confirmable-confirm": "à¤\95à¥\87 {{GENDER:$1|तपाà¤\88à¤\82}} सुनिश्चित हुनुहुन्छ ?",
+       "confirmable-confirm": "à¤\95à¥\87 {{GENDER:$1|तपाà¤\88à¤\81}} सुनिश्चित हुनुहुन्छ ?",
        "confirmable-yes": "हो",
        "confirmable-no": "होइन",
        "thisisdeleted": "$1 हेर्ने या पूर्वरुपमा फर्काउने हो?",
        "actionthrottled": "कार्य रोकियो",
        "actionthrottledtext": "स्पाम रोकथामको लागि , तपाईंलाई यो कार्य थोरै समयमा धेरै पटक गर्नबाट सिमित गरिएको छ, र तपाईंले आफ्नो सिमा पार गरिसक्नु भयो ।\nकृपया केही मिनेट पछि पुन: प्रयास गर्नुहोस्  ।",
        "protectedpagetext": "यो पृष्ठ सम्पादन हुनबाट बचाउन सम्पादनमा तथा अन्यकार्यमा रोक लगाइएको छ।",
-       "viewsourcetext": "तपाà¤\88à¤\82 यस पृष्ठको स्रोत हेर्न र प्रतिलिपी गर्न सक्नुहुन्छ ।",
+       "viewsourcetext": "तपाà¤\88à¤\81 यस पृष्ठको स्रोत हेर्न र प्रतिलिपी गर्न सक्नुहुन्छ ।",
        "viewyourtext": "यस पृष्ठमा रहेका '''तपाईंका सम्पादनहरू''' हेर्न या प्रतिलिपी गर्न सक्नुहुन्छ :",
        "protectedinterface": "यो पृष्ठले सफ्टवेयरको लागि अन्तरमोहडा पाठ प्रदान गर्दछ , र यसलाई दुरुपयोग हुनबाट बचाउन सुरक्षा प्रादन गरिएको छ।\nसम्पूर्ण विकिहरूका लागि अनुवादमा परिवर्तन गर्नको लागि [https://translatewiki.net/ translatewiki.net], प्रयोग गर्नुहोस् ,  मिडियाविकि स्थानियकरण परियोजना ।",
        "editinginterface": "<strong>चेतावनी:</strong> तपाईं यस पृष्ठलाई सम्पादन गर्नुहुँदैछ, जसले सफ्टवेयरको लागि \nइन्टरफेस सामग्रीहरू प्रदान गर्दछ।\nयस पृष्ठमा गरिएकोपरिवर्तनले यस विकिमा अरु प्रयोगकर्ताको इन्टरफेसको प्रदर्शनमा प्रभाव पार्नेछ ।",
        "createaccountmail": "कुनै अस्थाई र श्रिजित पासवर्ड प्रयोग गर्ने र खुलाईएको इमेलमा पठाउने",
        "createacct-realname": "वास्तविक नाम (ऐच्छिक)",
        "createacct-reason": "कारण",
-       "createacct-reason-ph": "à¤\95िन à¤¤à¤ªà¤¾à¤\88à¤\82 à¤¨à¤¯à¤¾à¤\81 à¤\96ाता à¤\96à¥\8bलिरहनà¥\81 à¤­à¤\8fà¤\95à¥\8b à¤¹à¥\8b ?",
+       "createacct-reason-ph": "तपाà¤\88à¤\81 à¤\95िन à¤\85रà¥\8dà¤\95à¥\8b à¤\96ाता à¤\96à¥\8bलिरहनà¥\81 à¤­à¤\8fà¤\95à¥\8b à¤\9b",
        "createacct-submit": "तपाईंको खाता सिर्जना गर्नुहोस",
        "createacct-another-submit": "खाता खोल्नुहोस्",
        "createacct-continue-submit": "खाता निर्माणलाई निरन्तरता दिनुहोस्",
        "whitelistedittext": "पाना सम्पादन गर्न तपाँईले $1 गर्नु पर्दछ।",
        "confirmedittext": "तपाईंले पृष्ठ संपादन गर्नअघि आफ्नो ई-मेल ठेगानाको पुष्टि गर्नुपर्छ।\nकृपया ई-मेल ठेगाना तयार गरी [[Special:Preferences|प्रयोगकर्ता अभिरूचि]] मार्फत मंजुर गराउनुहोस्।",
        "nosuchsectiontitle": "सेक्सन फेला परेन",
-       "nosuchsectiontext": "तपाà¤\88à¤\82 à¤¤à¥\8dयसà¥\8dतà¥\8b à¤\96णà¥\8dडà¤\95à¥\8b à¤¸à¤®à¥\8dपादन à¤\97रà¥\8dनà¥\87 à¤ªà¥\8dरयास à¤\97रà¥\8dनà¥\81भयà¥\8b à¤\9cà¥\8b à¤\85सà¥\8dतितà¥\8dवमा à¤\9bà¥\88न।\nयà¥\8b à¤¸à¤¾à¤°à¤¿à¤\8fà¤\95à¥\8b à¤\85थवा à¤®à¥\87à¤\9fाà¤\87à¤\8fà¤\95à¥\8b à¤¹à¥\81नà¥\81परà¥\8dà¤\9b à¤\9cब à¤¤à¤ªà¤¾à¤\88à¤\82 à¤¯à¤¸ à¤ªà¥\83षà¥\8dठलाà¤\88 à¤¹à¥\87रà¥\8dनà¥\81हà¥\81à¤\81दà¥\88थियà¥\8b।",
+       "nosuchsectiontext": "तपाà¤\88à¤\81लà¥\87 à¤¤à¥\8dयसà¥\8dतà¥\8b à¤\96णà¥\8dडà¤\95à¥\8b à¤¸à¤®à¥\8dपादन à¤\97रà¥\8dनà¥\87 à¤ªà¥\8dरयास à¤\97रà¥\8dनà¥\81भयà¥\8b à¤\9cà¥\81न à¤\9bà¥\88न।\nà¤\9cब à¤¤à¤ªà¤¾à¤\88à¤\82 à¤¯à¤¸ à¤ªà¥\83षà¥\8dठलाà¤\88 à¤¹à¥\87रà¥\8dनà¥\81हà¥\81à¤\81दà¥\88थियà¥\8b, à¤¯à¥\8b à¤¸à¤¾à¤°à¤¿à¤\8fà¤\95à¥\8b à¤\85थवा à¤®à¥\87à¤\9fाà¤\87à¤\8fà¤\95à¥\8b à¤¹à¥\81नà¥\81परà¥\8dà¤\9b।",
        "loginreqtitle": "प्रवेशगर्नु जरुरी छ।",
        "loginreqlink": "प्रवेश (लग ईन)",
        "loginreqpagetext": "अरु पृष्ठ हेर्न तपाईंले $1 गर्नुपर्छ ।",
        "revdelete-text-file": "हटाइएको फाइल अवतरण फाइल इतिहासमा देखाइनेछ तर तिनको सामग्री सार्वजनिक रूपले देखाइने छैन।",
        "logdelete-text": "हटाइएको प्रवेश घटनाहरू अहिले पनि लगमा देखाइनेछ तर तिनको सामग्रीको केहि भागलाई सार्वजनिक रूपले हेर्न सकिने छैन।",
        "revdelete-text-others": "अन्य प्रवन्धकहरू अहिले पनि लुकाइएको सामग्रीको उपयोग गर्नका लागि अझै अतिरिक्त प्रतिबन्ध सेट गरिरहेका छन्, जब सम्म यो अ-नष्ट गर्न सक्षम हुनेछ।",
-       "revdelete-confirm": "पà¥\81षà¥\8dà¤\9fि à¤\97रà¥\8dनà¥\81हà¥\8bसà¥\8d à¤\95ि à¤¤à¤ªà¤¾à¤\88à¤\82 à¤¯à¥\8b à¤\95ारà¥\8dय à¤\97रà¥\8dन à¤\9aाहनà¥\81हà¥\81नà¥\8dà¤\9b, à¤¤à¤ªà¤¾à¤\88à¤\82 à¤¯à¤¸à¤\95à¥\8b à¤ªà¤°à¤¿à¤£à¤¾à¤® à¤¦à¥\87à¤\96ि à¤\85वà¤\97त à¤¹à¥\81नà¥\81हà¥\81नà¥\8dà¤\9b, à¤° à¤¤à¤ªà¤¾à¤\88à¤\82 यो [[{{MediaWiki:Policy-url}}|नीति]] अनुसार गर्दै हुनुहुन्छ।",
+       "revdelete-confirm": "पà¥\81षà¥\8dà¤\9fि à¤\97रà¥\8dनà¥\81हà¥\8bसà¥\8d à¤\95ि à¤¤à¤ªà¤¾à¤\88à¤\81 à¤¯à¥\8b à¤\95ारà¥\8dय à¤\97रà¥\8dन à¤\9aाहनà¥\81हà¥\81नà¥\8dà¤\9b, à¤¤à¤ªà¤¾à¤\88à¤\81 à¤¯à¤¸à¤\95à¥\8b à¤ªà¤°à¤¿à¤£à¤¾à¤® à¤¦à¥\87à¤\96ि à¤\85वà¤\97त à¤¹à¥\81नà¥\81हà¥\81नà¥\8dà¤\9b, à¤° à¤¤à¤ªà¤¾à¤\88à¤\81 यो [[{{MediaWiki:Policy-url}}|नीति]] अनुसार गर्दै हुनुहुन्छ।",
        "revdelete-suppress-text": "लुकाउने प्रयोग <strong>मात्रै</strong> यी परिस्थितिमा हुनु पर्नेछ:\n* सम्भावित अपमानजनक जानकारी\n* अनुपयुक्त निजी जानकारी\n*: <em>घरको ठेगाना वा टेलिफोन नम्बर, राष्ट्रिय पहिचान नम्बर आदि।</em>",
        "revdelete-legend": "दृष्टि बन्देज मिलाउने",
        "revdelete-hide-text": "पुनरावलोकन पाठ",
        "timezoneregion-europe": "युरोप",
        "timezoneregion-indian": "हिन्द महासागर",
        "timezoneregion-pacific": "प्राशान्त महासागर",
-       "allowemail": "à¤\85रà¥\81 à¤ªà¥\8dरयà¥\8bà¤\97à¤\95रà¥\8dताहरà¥\82बाà¤\9f à¤ªà¥\8dरापà¥\8dत à¤¹à¥\81नà¥\87 à¤\88मà¥\87ल à¤¸à¤\95à¥\8dषम à¤\97रà¥\8dनुहोस् ।",
+       "allowemail": "à¤\85रà¥\82 à¤ªà¥\8dरयà¥\8bà¤\97à¤\95रà¥\8dताहरà¥\82लà¥\87 à¤®à¤²à¤¾à¤\88  à¤\87मà¥\87ल à¤ªà¤ à¤¾à¤\89न à¤¦à¤¿नुहोस् ।",
        "prefs-searchoptions": "खोज्ने",
        "prefs-namespaces": "नेमस्पेसेज",
        "default": "पूर्वनिर्धारित",
        "prefs-common-config": "साझा CSS/जाभा स्क्रिप्ट सबै त्वचा(स्किन)को लागि:",
        "prefs-reset-intro": "तपाईं यो पृष्ठलाई आफ्नो अभिरुचीहरू साइट पूर्वावस्थामा फर्काउन प्रयोग गर्न सक्नुहुन्छ । त्यस पछि यसलाई रद्द गर्न सक्नुहुन्न ।",
        "prefs-emailconfirm-label": "इ-मेल एकिन प्रक्रिया :",
-       "youremail": "à¤\88मà¥\87ल",
+       "youremail": "à¤\87मà¥\87ल:",
        "username": "{{GENDER:$1|प्रयोगकर्ता नाम}}:",
        "prefs-memberingroups": "निम्न {{PLURAL:$1|समूह | समूहहरू}}को {{GENDER:$2|सदस्य}} :",
        "prefs-memberingroups-type": "$1",
        "gender-male": "उसले विकि पृष्ठहरू सम्पादन गर्छ",
        "gender-female": "उनले विकि पृष्ठ सम्पादन गर्छिन",
        "prefs-help-gender": "यो जानकारी दिनु वैकल्पिक छ।\nयो सफ्टवेयरमा लिङ्गको आधारमा तपाईंको लागि सहि सम्बोधन गर्नको निमित्त हुन्छ।\nयो जानकारी सार्वजनिक गरिनेछ।",
-       "email": "à¤\88मेल",
+       "email": "à¤\87मेल",
        "prefs-help-realname": "वास्तविक नाम ऐच्छिक हो ।\nतपाईंले खुलाउनु भएको खण्डमा तपाईंको कामको श्रेय दिनको लागि यसको प्रयोग गरिने छ ।",
        "prefs-help-email": "इमेल ठेगाना ऐच्छिक हो, तर  प्रवेश शब्दको पुनर्स्थापनाका लागि आवश्यकता छ, तपाईंले प्रवेश शब्द त के भुल्नु हुन्थ्यो ।",
        "prefs-help-email-others": "तपाईंले यो पनि चयन गर्न सक्नुहुन्छ कि अरुले तपाईंको परिचय नपाई तपाईंसित तपाईंको प्रयोगकर्ता अथवा वार्तालाप पृष्ठको माध्यमले सम्पर्क राखुन् ।",
        "hidden-category-category-desc": "यस श्रेणीमा <code><nowiki>__HIDDENCAT__</nowiki></code> मा पृष्ठ पाठ हो, जुन पूर्व निर्धारित रूपले पृष्ठहरूमा श्रेणी लिङ्कहरूको बाकसमा हेरिनबाट रोक्ने गर्दछ।",
        "trackingcategories-nodesc": "कुनै विवरण उपलब्ध छैन।",
        "trackingcategories-disabled": "श्रेणी अक्षम गरियो",
-       "mailnologin": "à¤\88मेल पठाउने ठेगाना नै भएन ।",
-       "mailnologintext": "तपाà¤\88à¤\82लà¥\87 à¤\85रà¥\81 à¤ªà¥\8dरयà¥\8bà¤\97à¤\95रà¥\8dताहरà¥\82लाà¤\88 à¤\88मà¥\87ल à¤ªà¤ à¤¾à¤\89नà¤\95à¥\8b à¤²à¤¾à¤\97ि à¤\86फà¥\81 à¤ªà¤¹à¤¿à¤²à¥\87 [[Special:UserLogin|पà¥\8dरवà¥\87श(लà¤\97à¤\87न)à¤\97रà¥\87à¤\95à¥\8b]] à¤¹à¥\81नà¥\81परà¥\8dà¤\9b à¤° [[Special:Preferences|à¤\86फà¥\8dनà¥\8b à¤°à¥\8bà¤\9cाà¤\87हरà¥\82मा]] à¤\8fà¤\89à¤\9fा à¤µà¥\88ध à¤\88मेल ठेगाना भएको हुनुपर्छ।",
+       "mailnologin": "à¤\87मेल पठाउने ठेगाना नै भएन ।",
+       "mailnologintext": "तपाà¤\88à¤\81लà¥\87 à¤\85रà¥\81 à¤ªà¥\8dरयà¥\8bà¤\97à¤\95रà¥\8dताहरà¥\82लाà¤\88 à¤\87मà¥\87ल à¤ªà¤ à¤¾à¤\89नà¤\95à¥\8b à¤²à¤¾à¤\97ि à¤\86फà¥\82 à¤ªà¤¹à¤¿à¤²à¥\87 [[Special:UserLogin|पà¥\8dरवà¥\87श(लà¤\97à¤\87न)à¤\97रà¥\87à¤\95à¥\8b]] à¤¹à¥\81नà¥\81परà¥\8dà¤\9b à¤° [[Special:Preferences|à¤\86फà¥\8dनà¥\8b à¤°à¥\8bà¤\9cाà¤\87हरà¥\82मा]] à¤\8fà¤\89à¤\9fा à¤®à¤¾à¤¨à¥\8dय à¤\87मेल ठेगाना भएको हुनुपर्छ।",
        "emailuser": "यो प्रयोगकर्तालाई ई-मेल पठाउनुहोस्",
        "emailuser-title-target": "{{GENDER:$1|प्रयोगकर्ता}}लाई इमेल गर्ने",
        "emailuser-title-notarget": "प्रयोगकर्तालाई इमेल गर्नुहोस्",
        "defemailsubject": "{{SITENAME}} प्रयपोगकर्ता \"$1\" बाट इमेल",
        "usermaildisabled": "प्रयोगकर्ता इमेल निरस्त गरिएको",
        "usermaildisabledtext": "यस विकिमा तपाईं अरु प्रयोगकर्तालाई ई-मेल पठाउन सक्नुहुन्न",
-       "noemailtitle": "à¤\88मेल ठेगाना नभएको",
+       "noemailtitle": "à¤\87मेल ठेगाना नभएको",
        "noemailtext": "प्रयोगकर्ताले सही ई-मेल ठेगाना दर्शाएको छैन।",
        "nowikiemailtext": "यी प्रयोगकर्ताले अरु प्रयोगकर्ताहरूबाट ई-मेल स्वीकार नगर्ने छनोट गरेकाछन्।",
        "emailnotarget": "प्राप्तकर्ताको रुपमा नभएको अथवा अमान्य प्रयोगकर्ता।",
        "notanarticle": "सामाग्री सहितको पेज हैन",
        "notvisiblerev": "पूर्वावलोकन हटाइयो",
        "watchlist-details": "तपाईंको अवलोकन सूचीमा रहेका {{PLURAL:$1|$1 पृष्ठ|$1 पृष्ठहरू}} (तथा वार्तालाप पृष्ठहरू)।",
-       "wlheader-enotif": "à¤\88मेल जानकारी सक्रिय गरियो ।",
+       "wlheader-enotif": "à¤\87मेल जानकारी सक्रिय गरियो ।",
        "wlheader-showupdated": "तपाईंले पछिल्लो पल्ट भ्रमण गरेपछि परिवर्तन भएका पृष्ठहरूलाई <strong>गाढा<strong> गरेर देखाइएको छ ।",
        "wlnote": "$3 र $4 अनुसार विगत {{PLURAL:$2|घण्टामा|'''$2''' घण्टाहरूमा}} {{PLURAL:$1|गरिएको अन्तिम परिवर्तन तल दिइएकोछ|गरिएका अन्तिम  '''$1''' परिवर्तनहरू तल दिइएका छन्}}।",
        "wlshowlast": "पछिल्ला $2 दिनहरू $1 घण्टाहरू देखाउनुहोस्",
        "badipaddress": "अमान्य IP ठेगाना",
        "blockipsuccesssub": "निषेधकार्य सफल भयो",
        "blockipsuccesstext": "[[Special:Contributions/$1|$1]] निषेध गरिएको छ।<br />\nपुनरावलकोनको लागि [[Special:BlockList|आइपी निषेध सूची]] हेर्नहोस्।",
-       "ipb-blockingself": "तपाà¤\88à¤\82 à¤\86फà¥\88लाà¤\88 à¤°à¥\8bà¤\95 à¤²à¤\97ाà¤\89न à¤\97à¤\87रहनà¥\81 à¤­à¤\8fà¤\95à¥\8b à¤\9b! à¤\95à¥\87 à¤¤à¤ªà¤¾à¤\88à¤\82 à¤ªà¤\95à¥\8dà¤\95ा à¤¯à¤¸à¥\8b à¤\97रà¥\8dन à¤\9aाहनुहुन्छ?",
+       "ipb-blockingself": "तपाà¤\88à¤\81 à¤\86फà¥\88लाà¤\88 à¤°à¥\8bà¤\95 à¤²à¤\97ाà¤\89न à¤\97à¤\87रहनà¥\81 à¤­à¤\8fà¤\95à¥\8b à¤\9b! à¤\95à¥\87 à¤¤à¤ªà¤¾à¤\88à¤\81 à¤¯à¤¸à¥\8b à¤\97रà¥\8dन à¤¨à¤¿à¤¶à¥\8dà¤\9aित à¤¹à¥\81नुहुन्छ?",
        "ipb-confirmhideuser": "तपाईंले त्यस्तो प्रयोगकर्तालाई रोक लगाउन गइरहनु भएको छ जसको  \"प्रयोगकर्ता लुकाउने\" सक्रिय बनाइएको छ। यसो गर्दा सबै लग सूचीहरुबाट प्रयोगकर्ताको नाम गायब हुनेछ। के तपाईं पक्का यसो गर्न चाहनुहुन्छ?",
        "ipb-confirmaction": "यदि तपाईं विश्वासपूर्ण रूपले यहि गर्न चाहनुहुन्छ, भने कृपया \"{{int:ipb-confirm}}\" लाई क्षेत्रबाट छान्नुहोस्।",
        "ipb-edit-dropdown": "निषेध कारण सम्पादन गर्नुहोस्",
        "xffblockreason": "एउटा आइपि ठेगाना जुन X-Forwarded-For हेडरमा रहेको छ, या त तपाईंको हो वा त्यस प्रक्सी सर्भरको हो जसको प्रयोग तपाईं गर्दै हुनुहुन्छ र यस माथि प्रतिबन्ध लगाइएको छ। वास्तविक कारण थियो:$1",
        "cant-see-hidden-user": "तपाईंले रोक लगाउन खोज्नु भएको प्रयोगकर्ता पहिले नै रोकलगाइ लुकाइ सकिएको छ ।\nतपाईंसँग hideuser अधिकार नभएकोले यसलाई हेर्न या सम्पादन गर्न सक्नुहुन्न ।",
        "ipbblocked": "तपाईंले अरु प्रयोगकर्ताहरूलाई प्रतिबन्धित गर्न वा उनीहरूको प्रतिबन्ध खोल्न सक्नुहुन्न किनभने तपाईं आफै प्रतिबन्धित हुनुहुन्छ।",
-       "ipbnounblockself": "तपाà¤\88à¤\82 आफैले आफैलाई रोक खुलाउन सक्नुहुन्न ।",
+       "ipbnounblockself": "तपाà¤\88à¤\81 आफैले आफैलाई रोक खुलाउन सक्नुहुन्न ।",
        "lockdb": "डेटाबेस ताल्चामार्ने",
        "unlockdb": "डेटाबेसको ताल्चा खोल्ने",
        "lockdbtext": "डेटाबेसमा ताला लगाउनाले सबै प्रयोगकर्ता पृष्ठ सम्पादन, आफ्नो अभिरूचीमा परिवर्तन, आफ्नो ध्यानसूचीमा सम्पादन, र अन्य वस्तु जसको लागि डेटाबेसमा परिवर्तन गर्नु पर्ने हुन्छ, त्यसबाट वन्चित हुनेछ। कृपया यो सुनिश्चित गर्नुहोस् कि तपाईं यो गर्न चाहनुहुन्छ, र तपाईं रक्षण पश्चात ताला खोल्नुहुन्छ ।",
        "tooltip-p-logo": "मुख्य पृष्ठ",
        "tooltip-n-mainpage": "मुख्य पृष्ठमा जाने",
        "tooltip-n-mainpage-description": "मुख्य पृष्ठमा जानुहोस्",
-       "tooltip-n-portal": "à¤\86यà¥\8bà¤\9cनाà¤\95ा à¤¬à¤¾à¤°à¥\87मा, à¤¤à¤ªà¤¾à¤\88à¤\82 के गर्न सक्नुहुन्छ, सामग्री कहाँ भेट्टाउने",
+       "tooltip-n-portal": "à¤\86यà¥\8bà¤\9cनाà¤\95ा à¤¬à¤¾à¤°à¥\87मा, à¤¤à¤ªà¤¾à¤\88à¤\81 के गर्न सक्नुहुन्छ, सामग्री कहाँ भेट्टाउने",
        "tooltip-n-currentevents": "हालैको घटनाको बारेमा पृष्ठभूमि जानकारी पत्ता लगाउनुहोस्",
        "tooltip-n-recentchanges": "विकिमा गरिएका हालैका परिवर्तनहरूको सूची",
        "tooltip-n-randompage": "कुनै एक पृष्ठ खोल्ने",
        "confirmemail_subject": "{{SITENAME}} ई मेलl ठेगानाको पुष्टि",
        "confirmemail_body": "कसैले, सायद तपाईंले, आई पी ठेगाना $1बाट,\n{{SITENAME}}मा एउटा  खाता  \"$2\"को नाममा यस ई मेल ठेगानामा  पञ्जीकरण गरेकोछ।\n\nयो खाता साँच्ची नैं तपाईंको हो भनेर पुष्टि गर्न र {{SITENAME}}मा यो ई मेलका सुविधाहरु  सक्रिय गर्न तपाईंको ब्राउजरमा यो लिंक खोल्नुहोस्:\n\n$3 \n\nयदि त्यो खाता तपाईंले पञ्जीकरण गर्नु भएको *होइन* भनें, ई मेलको पुष्टिकरण रद्द गर्न यो लिंक पहिल्याउनुहोस्:\n\n$5\n\nयो पुष्टिकरणको समय  $4 मा सकिनेछ।",
        "confirmemail_body_changed": "कसैले, सायद तपाईंले, आई पी ठेगाना $1बाट,\n{{SITENAME}}मा \"$2\" नामको खाताको  ई मेल ठेगाना यस ठेगानामा  परिवर्तन गरेकोछ।\n\nयो खाता साँच्ची नैं तपाईंको हो भनेर पुष्टि गर्न र {{SITENAME}}मा यो ई मेलका सुविधाहरू  पुनः सक्रिय गर्न तपाईंको ब्राउजरमा यो लिंक खोल्नुहोस्:\n\n$3 \n\nयदि त्यो खाता तपाईंको *होइन* भने, ई मेल ठेगानाको पुष्टिकरण रद्द गर्न यो लिंक पहिल्याउनुहोस्:\n\n$5\n\nयो पुष्टिकरणको समय  $4 मा सकिनेछ।",
-       "confirmemail_body_set": "à¤\95सà¥\88लà¥\87, à¤¸à¤¾à¤¯à¤¦ à¤¤à¤ªà¤¾à¤\88à¤\82लà¥\87, à¤\86à¤\88पà¥\80 à¤ à¥\87à¤\97ाना $1बाà¤\9f,\n{{SITENAME}}मा \"$2\" à¤¨à¤¾à¤®à¤\95à¥\8b à¤\96ाताà¤\95à¥\8b à¤\88मà¥\87ल à¤ à¥\87à¤\97ाना à¤¯à¤¸ à¤ à¥\87à¤\97ानासित à¤\9cà¥\8bडà¥\87à¤\95à¥\8b à¤\9b।\n\nयà¥\8b à¤\96ाता à¤¸à¤¾à¤\81à¤\9aà¥\8dà¤\9aà¥\80 à¤¨à¥\88à¤\82 à¤¤à¤ªà¤¾à¤\88à¤\82à¤\95à¥\8b à¤¹à¥\8b à¤­à¤¨à¥\87र à¤ªà¥\81षà¥\8dà¤\9fि à¤\97रà¥\8dन à¤° {{SITENAME}}मा à¤¯à¥\8b à¤\88मà¥\87लà¤\95ा à¤¸à¥\81विधाहरà¥\82 à¤ªà¥\81नà¤\83 à¤¸à¤\95à¥\8dरिय à¤\97रà¥\8dन à¤¤à¤ªà¤¾à¤\88à¤\82à¤\95à¥\8b à¤¬à¥\8dराà¤\89à¤\9cरमा à¤¯à¥\8b à¤²à¤¿à¤\99à¥\8dà¤\95 à¤\96à¥\8bलà¥\8dनà¥\81हà¥\8bसà¥\8d:\n\n$3 \n\nयदि à¤¤à¥\8dयà¥\8b à¤\96ाता à¤¤à¤ªà¤¾à¤\88à¤\82à¤\95à¥\8b *हà¥\8bà¤\87न* à¤­à¤¨à¥\87, à¤\88मà¥\87ल à¤ à¥\87à¤\97ानाà¤\95à¥\8b à¤ªà¥\81षà¥\8dà¤\9fिà¤\95रण à¤°à¤¦à¥\8dद à¤\97रà¥\8dन à¤¯à¥\8b à¤²à¤¿à¤\99à¥\8dà¤\95 à¤ªà¤¹à¤¿à¤²à¥\8dयाà¤\89नà¥\81हà¥\8bसà¥\8d:\n\n$5\n\nयà¥\8b à¤ªà¥\81षà¥\8dà¤\9fिà¤\95रणà¤\95à¥\8b à¤¸à¤®à¤¯  $4 मा सकिनेछ।",
+       "confirmemail_body_set": "à¤\95सà¥\88लà¥\87, à¤¸à¤¾à¤¯à¤¦ à¤¤à¤ªà¤¾à¤\88à¤\81लà¥\87, à¤\86à¤\87पà¥\80 à¤ à¥\87à¤\97ाना $1 à¤¬à¤¾à¤\9f,\n\n{{SITENAME}}मा \"$2\" à¤¨à¤¾à¤®à¤\95à¥\8b à¤\96ाताà¤\95à¥\8b à¤\87मà¥\87ल à¤ à¥\87à¤\97ाना à¤¯à¤¸ à¤ à¥\87à¤\97ानासित à¤\9cà¥\8bडिà¤\8fà¤\95à¥\8b à¤\9b।\nयà¥\8b à¤\96ाता à¤¸à¤¾à¤\81à¤\9aà¥\8dà¤\9aà¥\88 à¤¤à¤ªà¤¾à¤\88à¤\81à¤\95à¥\8b à¤¹à¥\8b à¤­à¤¨à¥\87र à¤ªà¥\81षà¥\8dà¤\9fि à¤\97रà¥\8dन à¤° {{SITENAME}}मा à¤¯à¥\8b à¤\87मà¥\87लà¤\95ा à¤¸à¥\81विधाहरà¥\82 à¤ªà¥\81नà¤\83 à¤¸à¤\95à¥\8dरिय à¤\97रà¥\8dन à¤¤à¤ªà¤¾à¤\88à¤\81à¤\95à¥\8b à¤¬à¥\8dराà¤\89à¤\9cरमा à¤¯à¥\8b à¤²à¤¿à¤\99à¥\8dà¤\95 à¤\96à¥\8bलà¥\8dनà¥\81हà¥\8bसà¥\8d:\n\n$3 \n\nयदि à¤¤à¥\8dयà¥\8b à¤\96ाता à¤¤à¤ªà¤¾à¤\88à¤\81à¤\95à¥\8b *हà¥\8bà¤\87न* à¤­à¤¨à¥\87, à¤\87मà¥\87ल à¤ à¥\87à¤\97ानाà¤\95à¥\8b à¤ªà¥\81षà¥\8dà¤\9fिà¤\95रण à¤°à¤¦à¥\8dद à¤\97रà¥\8dन à¤¯à¥\8b à¤²à¤¿à¤\99à¥\8dà¤\95 à¤ªà¤¹à¤¿à¤²à¥\8dयाà¤\89नà¥\81हà¥\8bसà¥\8d:\n\n$5\n\nयà¥\8b à¤ªà¥\81षà¥\8dà¤\9fिà¤\95रणà¤\95à¥\8b à¤®à¥\8dयाद  $4 मा सकिनेछ।",
        "confirmemail_invalidated": "ई मेल ठेगाना रद्द भएको पुष्टिकरण",
        "invalidateemail": "इमेल यकिन कार्य रद्द गर्नुहोस्",
        "scarytranscludedisabled": "[अन्तरविकि दस्तावेज अन्तरकरण निस्क्रिय]",
        "tags-create-warnings-above": "निम्नलिखित {{PLURAL:$2|चेतावनी देखाइयो|चेतावनीहरू देखाइयो}} जब ट्याग \"$1\" बनाउने प्रयास गरिएको थियो:",
        "tags-create-warnings-below": "के तपाईं यो ट्याग बनाउने काम जारी राख्न चाहनु हुन्छ ?",
        "tags-delete-title": "ट्याग मेट्नुहोस्",
-       "tags-delete-explanation-initial": "तपाà¤\88à¤\82 ट्याग \"$1\" लाई डाटावेसबाट हटाउँदै हुनुहुन्छ ।",
+       "tags-delete-explanation-initial": "तपाà¤\88à¤\81 ट्याग \"$1\" लाई डाटावेसबाट हटाउँदै हुनुहुन्छ ।",
        "tags-delete-explanation-in-use": "ट्यागलाई {{PLURAL:$2|$2 संशोधन वा लग प्रविष्टि|सबै $2 संशोधन र/वा लग प्रविष्टिहरू}}बाट हटाइनेछ जहाँ अहिले त्यसको प्रयोग गरिंदै छ।",
        "tags-delete-explanation-warning": "यो क्रिया <strong>अपरिवर्तनीय</strong> हो र <strong>त्यसलाई परिवर्तन गर्न सकिंदैन</strong>, डेटाबेस प्रवन्धक पनि यसलाई केहि गर्न सक्दैनन्। विश्वासपूर्ण रूपले तपाईं तय गर्नुहोस् कि तपाईं यस ट्यागलाई हटाउन चाहनुहुन्छ।",
        "tags-delete-explanation-active": "<strong>ट्याग \"$1\" अहिले पनि सक्रिय छ, र यसको प्रयोग भविष्यमा पनि जारी रहनेछ।</strong> यसलाई रोकनका लागि, ती स्थानहरूमा जानुहोस जहाँ यस ट्यागको प्रयोग भइरहेको छ र त्यहाँ देखि यसलाई अक्षम गर्नुहोस।",
        "log-action-filter-block-block": "रोक्ने",
        "log-action-filter-block-unblock": "फुक्का गर्ने",
        "authmanager-userdoesnotexist": "प्रयोगकर्ता खाता \"$1\" दर्ता गरिएको छैन।",
-       "authmanager-email-label": "à¤\88मेल",
+       "authmanager-email-label": "à¤\87मेल",
        "authmanager-email-help": "इमेल ठेगाना",
        "authprovider-resetpass-skip-label": "छोड्नुहोस्",
        "edit-error-short": "त्रुटि: $1",
index fa5ee6f..19c9734 100644 (file)
        "linkaccounts": "Accounts koppelen",
        "linkaccounts-success-text": "Het account is gekoppeld.",
        "linkaccounts-submit": "Accounts koppelen",
+       "cannotunlink-no-provider-title": "Er zijn geen accounts om te ontkoppelen",
+       "cannotunlink-no-provider": "Er zijn geen gekoppelde accounts die ontkoppelt kunnen worden.",
        "unlinkaccounts": "Accounts ontkoppelen",
        "unlinkaccounts-success": "Het account is ontkoppeld.",
        "authenticationdatachange-ignored": "De wijziging van de authenticatiegegevens is niet afgehandeld. Misschien is er geen provider geconfigureerd?",
index c11c94a..df22576 100644 (file)
        "duplicate-args-category-desc": "ߞߙߊߞߏ ߞߟߌߟߌ ߟߎ߬ ߦߋ߫ ߞߐߜߍ ߘߐ߫߸ ߡߍ߲ ߠߎ߬ ߦߋ߫ ߘߊߘߐߡߌߘߊߞߎ߲ߢߊ߫ ߓߊߟߌߣߍ߲ ߠߎ߬ ߟߊߓߊ߯ߙߊ߫ ߟߊ߫߸ ߦߏ߫ <code><nowiki>{{foo|bar=1|bar=2}}</nowiki></code> ߥߟߊ߫ <code><nowiki>{{foo|bar|1=baz}}<nowiki></code>.",
        "expensive-parserfunction-warning": "<strong>ߖߊ߬ߛߙߋ߬ߡߊ߬ߟߊ</strong> ߞߐߜߍ ߣߌ߲߬ ߞߣߐߘߐ ߟߎ߬ ߘߐߞߍ߫ ߣߍ߲߫ ߞߎߙߎ߲ߞߎߙߎ߲ߟߊ߲߫ ߘߊߜߍߟߍ߲ߓߊ ߟߎ߬ ߗߋߘߊ ߞߟߌߟߌ ߟߎ߬ ߟߋ߬ ߟߊ߫. \n\nߕߎ߬ߡߊ߬ߘߐ߫ ߊ߬ ߘߌ߫ ߞߍ߫ $2 ߘߎ߰ߟߊ߫ \n{{PLURAL:$2|ߞߟߌߟߌ|ߞߟߌߟߌ ߟߎ߬}}߸ ߘߌ߫ ߞߍ߫ {{PLURAL:$1|ߦߋ߫ ߞߟߌߟߌ $1 ߟߋ߬ ߘߌ߫ ߡߎ߬ߕߎ߲߬|ߦߋ߫ ߞߟߌߟߌ ߟߎ߬ $1 ߟߋ߬ ߘߌ߫ ߡߎ߬ߕߎ߲߬}}.",
        "undo-failure": "ߡߊ߬ߦߟߍ߬ߡߊ߲߬ߠߌ߲ ߕߍ߫ ߣߊ߬ ߛߋ߫ ߟߊ߫ ߘߐߛߊ߬ ߟߊ߫߸ ߝߘߏ߬ߒ߬ߡߊ߬ߟߌ߬ ߡߊߦߟߍߡߊ߲ߠߌ߲ ߞߏߛߐ߲߬.",
+       "undo-summary-username-hidden": "ߟߊ߬ߓߊ߰ߙߊ߬ߟߊ߫ ߢߡߊߘߏ߲߰ߣߍ߲ ߠߎ߬ ߟߢߊ߬ߟߌ $1 ߘߐߛߊ߬",
+       "cantcreateaccount-text": "ߖߊ߬ߕߋ߬ߘߊ߬ ߛߌ߲ߘߟߌ ߞߊ߬ ߝߘߊ߫ ߟߊ߬ߓߊ߰ߙߊ߬ߟߊ ߣߌ߲߬ ߠߊ߫ IP ߛߊ߲߬ߓߊ߬ߕߐ߮ (<strong>$1</strong>) ߟߊ߫, ߏ߬ ߓߘߊ߫ ߓߊ߬ߟߌ߬ [[User:$3|$3]] ߓߟߏ߫.\n\nߞߎ߲߭ ߡߍ߲ ߦߌ߬ߘߊ߬ ߣߴߏ߬ ߟߊ߫ $3 ߓߟߏ߫߸ ߏ߬ ߦߋ߫ <em>$2</em> ߟߋ߬ ߘߌ߫",
        "viewpagelogs": "ߞߐߜߍ ߣߌ߲߬ ߜߊ߲߬ߞߎ߲߬ߠߌ߲ ߠߎ߬ ߦߋ߫",
        "nohistory": "ߡߊ߬ߦߟߍ߬ߡߊ߲߬ߠߌ߲ ߘߐ߬ߝߐ߬ ߛߌ߫ ߕߍ߫ ߞߐߜߍ ߣߌ߲߬ ߠߊ߫",
        "currentrev": "ߡߊ߬ߛߊ߬ߦߌ߲߬ߠߌ߲ ߕߊ߬ߡߌ߲߬ߣߍ߲",
        "rcfilters-filter-previousrevision-description": "ߡߊ߬ߦߟߍ߬ߡߊ߲߬ߠߌ߲ ߠߎ߬ ߓߍ߯ ߡߍ߲ ߠߎ߬ ߕߍ߫ \"ߟߢߊ߬ߟߌ߬ ߞߐ߯ߟߕߊ߫\" ߘߌ߫.",
        "rcfilters-filter-excluded": "ߊ߬ ߓߘߊ߫ ߟߊߘߏ߲߬ ߊ߬ ߘߐ߫",
        "rcfilters-tag-prefix-namespace-inverted": "<strong>:ߍ߲߬ߍ߲߫</strong> $1",
+       "rcfilters-view-namespaces-tooltip": "ߛߍ߲ߛߍ߲ߟߊ߲ ߞߐߝߟߌ ߕߐ߯ߛߓߍ ߞߣߍ ߡߊ߬",
+       "rcfilters-view-tags-tooltip": "ߛߍ߲ߛߍ߲ߟߊ߲ ߞߐߝߟߌ ߘߎ߲ߛߓߍ ߡߊߦߟߍ߬ߡߊ߲߬ߠߌ߲ ߡߊ߬",
+       "rcfilters-view-return-to-default-tooltip": "ߌ ߞߐߛߊ߬ߦߌ߬ ߛߍ߲ߛߍ߲ߟߊ߲ ߓߏ߬ߟߏ߲߬ߘߊ ߢߣߊߕߊߟߌ ߡߊ߬",
+       "rcfilters-view-tags-help-icon-tooltip": "ߛߌߦߊߡߊ߲߫ ߞߊ߬ߙߊ߲߬ ߞߎ߲߭ ߡߊߦߟߍ߬ߡߊ߲߬ߠߌ߲ ߞߊ߲߬",
+       "rcfilters-liveupdates-button": "ߟߏ߲ߘߐߦߊߟߌ ߝߊ߲ߞߊߢߊ߯",
+       "rcfilters-liveupdates-button-title-on": "ߟߏ߲ߘߐߦߊߟߌ ߝߊ߲ߞߊߢߊ߯ ߓߐ߫ ߊ߬ ߟߊ߫",
+       "rcfilters-liveupdates-button-title-off": "ߡߝߊ߬ߟߋ߲߬ߠߌ߲߬ ߞߎߘߊ߫ ߞߎߘߊ߫ ߟߊߓߊ߯ߙߊ߫ ߊ߬ߟߎ߬ ߞߍߢߊ ߡߊ߬",
+       "rcfilters-watchlist-markseen-button": "ߡߊ߬ߦߟߍ߬ߡߊ߲߬ߠߌ߲ ߓߍ߯ ߣߐ߬ߣߐ߬ ߦߋߣߍ߲ ߘߌ߫",
+       "rcfilters-watchlist-edit-watchlist-button": "ߌ ߟߊ߫ ߞߐߜߍ߫ ߡߊߝߟߍߣߍ߲ ߠߎ߬ ߛߙߍߘߍ ߡߊߦߟߍ߬ߡߊ߲߫",
        "rcfilters-target-page-placeholder": "ߞߐߜߍ ߕߐ߮ ߟߊߘߏ߲߬ (ߥߟߊ߫ ߦߌߟߡߊ)",
        "rcnotefrom": "ߘߎ߰ߟߊ ߘߐ߫ {{PLURAL:$5|is the change|are the changes}} ߞߊ߬ߦߌ߯ <strong>$3, $4</strong> (up to <strong>$1</strong> shown).",
        "rclistfromreset": "ߞߐߜߍ ߓߊߕߐߡߐ߲ߠߌ߲ ߡߊߦߟߍ߬ߡߊ߲߫",
        "uploadwarning": "ߟߊ߬ߦߟߍ߬ߟߌ ߖߊ߬ߛߙߋ߬ߡߊ߬ߟߊ",
        "uploadwarning-text": "ߞߐߕߐ߮ ߘߎ߰ߟߊ߬ߘߐ߫ ߞߊ߲ߛߓߍߟߌ ߡߊ߬ߦߟߍ߬ߡߊ߲߫ ߖߊ߰ߣߌ߲߬߸ ߞߵߊ߬ ߡߊߝߍߣߍ߲߫ ߕߎ߲߯.",
        "savefile": "ߞߐߕߐ߮ ߟߊߞߎ߲߬ߘߎ߬",
+       "uploaddisabled": "ߟߊ߬ߦߟߍ߬ߟߌ ߓߐ߫ ߣߴߊ߬ ߟߊ߫.",
+       "copyuploaddisabled": "ߟߊ߬ߦߟߍ߬ߟߌ ߞߍ URL ߟߊ߫߸ ߏ߬ ߓߐ߫ ߣߴߊ߬ ߟߊ߫.",
+       "uploaddisabledtext": "ߞߐߕߐ߮ ߟߊߦߟߍ߬ߟߌ ߓߐ߫ ߣߴߊ߬ ߟߊ߫.",
+       "php-uploaddisabledtext": "ߞߐߕߐ߮ ߟߊߦߟߍ߬ߟߌ ߓߐ߫ ߣߴߊ߬ ߟߊ߫ PHP ߘߐ߫.\nߞߐߕߐ߮ ߟߊߦߟߍ߬ߟߌ ߟߊ߬ߓߍ߲߬ߢߐ߲߰ߡߊ ߝߛߍ߬ߝߛߍ߫ ߖߊ߰ߣߌ߲߬.",
+       "upload-scripted-pi-callback": "ߊ߬ ߕߍߣߊ߬ ߛߐ߲߬ ߠߊ߫ ߞߐߕߐ߯ ߟߊߦߟߍ߬ ߟߊ߫ XML-stylesheet ߟߐ߲ߣߌ߲ߦߊ ߘߊ߲ߘߊߟߌ.",
        "upload-source": "ߞߐߕߐ߮ ߛߎ߲",
        "sourcefilename": "ߞߐߕߐ߮ ߕߐ߮ ߛߎ߲:",
        "sourceurl": "URL ߛߎ߲:",
        "backend-fail-read": "ߞߐߕߐ߮ ߕߴߛߋ߫ ߘߐߞߊ߬ߙߊ߲߬ ߠߊ߫   \"$1\".",
        "backend-fail-create": "ߊ߬ ߕߍ߫ ߣߊ߬ ߛߐ߲߬ ߠߊ߫ ߞߐߕߐ߮  \"$1\" ߛߓߍ߫ ߟߊ߫.",
        "backend-fail-maxsize": "ߊ߫ ߕߍ߫ ߣߊ߬ ߞߐߕߐ߮  \"$1\" ߛߓߍ߫ ߟߊ߫߸ ߓߊߏ߬ ߊ߬ ߓߏ߲߬ߓߊ߫ ߞߊ߬ ߕߊ߬ߡߌ߲߬ {{PLURAL:$2|ߝߙߐ߬ߢߐ߬ ߞߋߟߋ߲߫|ߝߙߐ߬ߢߐ ߟߎ߬ $2}}.",
+       "lockmanager-notlocked": "ߊ߬ ߕߍ߫ ߣߊ߬ ߛߐ߲߬ ߠߊ߫ \"$1\" ߟߊߞߊ߬ ߟߊ߫߸ ߊ߬ ߛߐ߰ߣߍ߲߬ ߕߍ߫.",
+       "lockmanager-fail-closelock": "ߊ߬ ߕߍ߫ ߣߊ߬ ߛߋ߫ ߟߊ߫ \"$1\" ߞߐߕߐ߮ ߛߐ߰ߣߍ߲ ߘߊߕߎ߲߯ ߠߊ߫.",
+       "lockmanager-fail-deletelock": "ߊ߬ ߕߍ߫ ߣߊ߬ ߛߋ߫ ߟߊ߫ \"$1\" ߞߐߕߐ߯ ߛߐ߰ߣߍ߲ ߖߏ߰ߛߌ߬ ߟߊ߫.",
        "uploadstash-errclear": "ߞߐߕߐ߯ ߗߌߙߏ߲ߣߍ߲ ߖߏ߰ߛߌ߬ߟߌ ߦߴߌ ߘߐ߫.",
        "uploadstash-bad-path-unknown-type": "ߛߎ߯ߦߊ߫  \"$1\" ߡߊߟߐ߲ߓߊߟߌ",
        "uploadstash-no-extension": "ߘߐ߬ߥߙߊ߬ߟߌ ߦߋ߫ ߝߏߦߊ߲ ߠߋ߬ ߘߌ߫.",
        "uploadstash-zero-length": "ߞߐߕߐ߮ ߦߋ߫ ߥߊ߲߬ߥߊ߲߬ ߘߐߞߏߟߏ߲ ߠߋ߬ ߘߌ߫.",
        "img-auth-nofile": "ߞߐߕߐ߮  \"$1\" ߕߍ߫ ߦߋ߲߬.",
+       "img-auth-noread": "ߟߊ߬ߛߐ߬ߘߐ߲߬ߠߌ߲ ߝߙߍ߫ ߕߍ߫ ߟߊ߬ߓߊ߰ߙߊ߬ߟߊ ߦߋ߫ ߞߊ߬ \"$1\" ߘߐߞߊ߬ߙߊ߲߬.",
+       "http-invalid-url": "URL: $1 ߓߍ߲߬ߓߊߟߌ",
        "http-request-error": "HTTP ߡߊ߬ߢߌ߬ߣߌ߲߬ߠߌ߲ ߓߘߊ߫ ߗߌߙߏ߲߫ ߝߎ߬ߕߎ߲߬ߕߌ߬ ߡߊߟߐ߲ߓߊߟߌ ߘߏ߫ ߞߏߛߐ߲߬.",
        "http-read-error": "HTTP ߘߐ߬ߞߊ߬ߙߊ߲߬ߠߌ߲ ߝߎ߬ߕߎ߲߬ߕߌ.",
        "http-timed-out": "HTTP ߡߊ߬ߢߌ߬ߣߌ߲߬ߠߌ߲ ߕߎ߬ߡߊ ߓߘߊ߫ ߕߊ߬ߡߌ߲߬.",
        "linkstoimage-redirect": "$1 (ߞߐߕߐ߯ ߟߊߞߎ߲߬ߛߌ߲߬ߣߍ߲߬) $2",
        "sharedupload-desc-here": "ߘߐ߬ߛߙߋ ߣߌ߲߬ ߦߋ߫ ߦߊ߲߬ ߠߋ߫ $1 ߖߊ߬ߕߋ߬ߘߐ߬ߛߌ߮ ߕߐ߭ ߟߎ߬ ߞߏ߬ߣߌ߲ ߘߌ߫ ߛߴߊ߬ ߟߊߓߊ߯ߙߊ߫ ߟߊ߫. ߊ߬ ߕߐ߯ ߛߓߍߟߌ ߦߙߐ [$2 ߞߐߕߐ߮ ߞߊ߲߬ߛߓߍߟߌ ߞߐߜߍ] ߟߋ߬ ߦߋ߫ ߘߎ߰ߟߊ ߘߐ߫ ߣߌ߲߬.",
        "filepage-nofile": "ߕߐ߮ ߣߌ߲߬ ߞߐߕߐ߯ ߛߎ߯ ߕߍ߫ ߦߋ߲߬",
+       "uploadnewversion-linktext": "ߞߐߕߐ߮ ߣߌ߲߬ ߛߎ߯ߦߊ߫ ߞߎߘߊ߫ ߟߊߦߟߍ߬",
        "shared-repo-from": "ߞߊ߬ ߝߘߊ߫: $1",
        "upload-disallowed-here": "ߌ ߕߍߣߊ߬ ߞߐߜߍ ߣߌ߲߬ ߞߊ߲߬ߛߓߍ߫ ߟߊ߫.",
+       "filerevert": "ߌ ߞߐߛߊ߬ߦߌ߬ $1",
+       "filerevert-legend": "ߞߐߕߐ߯ ߟߊߛߊ߬ߦߌ߲߬ߣߍ߲",
+       "filerevert-intro": "ߌ ߦߴߊ߬ ߞߊ߲߬ ߞߊ߬ ߞߐߕߐ߮ <strong>[[Media:$1|$1]]</strong> ߟߊߛߊ߬ߦߌ߬ ߦߊ߲߬ [$4 ߛߎ߯ߦߊ ߞߵߊ߬ ߕߊ߬ $3,$2]",
        "filerevert-comment": "ߊ߬ ߛߊߓߎ:",
+       "filerevert-defaultcomment": "ߓߘߊ߫ ߟߊߞߐߛߊ߬ߦߌ߬ ߛߎ߯ߦߊ ߡߊ߬ $2,$1, ($3)",
+       "filerevert-submit": "ߞߐߛߊߦߟߌ",
        "filedelete": "ߖߏ߰ߛߌ߬ߟߌ $1",
        "filedelete-legend": "ߞߐߕߐ߮ ߖߏ߰ߛߌ߬",
        "filedelete-comment": "ߊ߬ ߛߊߓߎ:",
        "unusedimages": "ߞߐߕߐ߯ ߟߊߓߊ߯ߙߊߓߊߟߌ",
        "wantedcategories": "ߦߌߟߡߊ߫ ߞߊ߬ߣߌ߲߬ߣߍ߲ ߠߎ߬",
        "wantedpages": "ߞߐߜߍ߫ ߜߋ߬ߟߎ߲߬ߣߍ߲ ߠߎ߬",
+       "wantedpages-badtitle": "ߞߎ߲߬ߕߐ߮ ߓߍ߲߬ߣߍ߲߬ ߕߍ߫ ߞߐߝߟߌ߫ ߦߌ߬ߘߊ߬ߣߍ߲: $1 ߘߐ߫",
+       "wantedfiles": "ߞߐߜߍ߫ ߜߋ߬ߟߎ߲߬ߣߍ߲ ߠߎ߬",
        "wantedtemplates": "ߞߙߊߞߏ ߞߊ߬ߣߌ߲߬ߣߍ߲ ߠߎ߬",
        "mostlinked": "ߛߘߌ߬ߜߋ߲߬ ߦߙߌߞߊ߫ ߦߋ߫ ߞߐߜߍ ߡߍ߲ ߠߎ߬ ߘߐ߫",
        "mostlinkedcategories": "ߛߘߌ߬ߜߋ߲߬ ߦߙߌߞߊ߫ ߦߋ߫ ߦߌߟߡߊ ߡߍ߲ ߠߎ߬ ߘߐ߫",
        "allpagessubmit": "ߥߊ߫",
        "allpages-hide-redirects": "ߟߊ߬ߞߎ߲߬ߛߌ߲߬ߠߌ߲ ߢߡߊߘߏ߲߰",
        "categories": "ߦߌߟߡߊ ߟߎ߬",
+       "categories-submit": "ߦߌ߬ߘߊ߬ߟߌ",
        "categoriesfrom": "ߦߌߟߡߊ ߟߎ߬ ߦߌ߬ߘߊ߬ߟߌ ߟߊߝߟߐ߫ ߣߌ߲߬ ߡߊ߬:",
        "deletedcontributions": "ߟߊ߬ߓߊ߰ߙߊ߬ߟߊ ߟߊ߫ ߓߟߏߡߊߜߍ߲ ߠߎ߬ ߓߘߊ߫ ߖߏ߬ߛߌ߬",
        "deletedcontributions-title": "ߟߊ߬ߓߊ߰ߙߊ߬ߟߊ ߟߊ߫ ߓߟߏߡߊߜߍ߲ ߓߘߊ߫ ߓߊ߲߫ ߖߏ߬ߛߌ߬ ߟߊ߫",
        "listusers-submit": "ߦߌ߬ߘߊ߬ߟߌ",
        "listusers-noresult": "ߟߊ߬ߓߊ߰ߙߊ߬ߟߊ ߕߴߦߋ߲߬",
        "activeusers-noresult": "ߟߊ߬ߓߊ߰ߙߊ߬ߟߊ ߕߴߦߋ߲߬",
+       "listgrouprights-group": "ߞߙߎ:",
+       "listgrouprights-rights": "ߞߌߣߌ߲",
+       "listgrouprights-helppage": "Help: ߞߙߎ ߤߊߞߍ",
        "listgrouprights-members": "(ߛߌ߲߬ߝߏ߲ ߠߎ߫ ߛߙߍߘߍ)",
+       "listgrouprights-addgroup": "ߟߊ߬ߘߏ߲߬ߠߌ߲ {{PLURAL:$2|ߞߙߎ|ߞߙߎ ߟߎ߬}}: $1",
+       "listgrouprights-removegroup": "ߛߋ߲߬ߓߐ߬ߟߌ {{PLURAL:$2|ߞߙߎ|ߞߙߎ ߟߎ߬}}: $1",
+       "listgrouprights-addgroup-all": "ߟߊߘߏ߲߬ ߞߙߎ ߓߍ߯ ߘߐ߫",
+       "listgrouprights-removegroup-all": "ߊ߬ ߛߋ߲߬ߓߐ߫ ߞߙߎ ߓߍ߯ ߘߐ߫",
+       "listgrouprights-addgroup-self": "ߟߊ߬ߘߏ߲߬ߠߌ߲ {{PLURAL:$2|ߞߙߎ|ߞߙߎ ߟߎ߬}} ߌ ߖߘߍ߬ߞߊ߬ߣߌ߲߬ ߖߊߕߋߘߊ: $1",
+       "listgrouprights-removegroup-self": "ߛߋ߲߬ߓߐ߬ߟߌ {{PLURAL:$2|ߞߙߎ|ߞߙߎ ߟߎ߬}} ߞߊ߬ ߝߘߊ߫ ߌ ߖߘߍ߬ߞߊ߬ߣߌ߲߬ ߖߊߕߋߘߊ ߟߊ߫: $1",
+       "listgrouprights-addgroup-self-all": "ߞߙߎ ߓߍ߯ ߟߊߘߏ߲߬ ߌ ߖߘߍ߬ߞߊ߬ߣߌ߲߬ ߖߊߕߋߘߊ ߘߐ߫",
+       "listgrouprights-removegroup-self-all": "ߞߙߎ ߓߍ߯ ߛߋ߲߬ߓߐ߫ ߌ ߖߘߍ߬ߞߊ߬ߣߌ߲߬ ߖߊߕߋߘߊ ߘߐ߫",
+       "listgrouprights-namespaceprotection-namespace": "ߕߐ߯ߛߓߍ ߞߣߍ:",
+       "listgrouprights-namespaceprotection-restrictedto": "ߞߌߣߌ߲߸ ߟߊ߬ߓߊ߰ߙߊ߬ߟߊ ߟߊߘߌ߬ߢߍ ߦߴߌ ߘߐ߫ ߞߵߊ߬ ߡߊߦߟߍ߬ߡߊ߲߫",
+       "listgrants-rights": "ߞߌߣߌ߲",
+       "trackingcategories-msg": "ߦߌߟߡߊ ߜߙߋ߬ߓߐ߬ߟߌ",
+       "trackingcategories-name": "ߗߋߛߓߍ ߕߐ߮",
+       "trackingcategories-nodesc": "ߞߊ߲߬ߛߓߍ߬ߟߌ߬ ߕߍ߫ ߦߋ߲߬",
+       "trackingcategories-disabled": "ߦߌߟߡߊ ߓߘߊ߫ ߓߘߊ߫ ߓߴߊ߬ ߟߊ߫",
+       "mailnologin": "ߗߋߟߌ ߛߊ߲߬ߓߊ߬ߕߐ߯ ߕߴߦߋ߲߬",
        "emailuser": "ߗߋߛߓߍ ߗߋ߫ ߣߌ߲߬ ߕߌ߭ ߡߊ߬",
+       "emailuser-title-target": "ߢߎߡߍߙߋ߲ߞߏ߲ߘߏ߫ ߟߊߕߊ߯ {{GENDER:$1|ߟߊߓߊ߯ߙߟߊ}} ߣߌ߲߬ ߡߊ߬",
+       "emailuser-title-notarget": "ߢߎߡߍߙߋ߲ߞߏ߲ߘߏ ߟߊ߬ߓߊ߰ߙߊ߬ߟߊ",
+       "defemailsubject": "{{SITENAME}} ߢߎߡߍߙߋ߲ߞߏ߲ߘߏ ߝߘߊߣߍ߲߫ ߟߊ߬ߓߊ߰ߙߊ߬ߟߊ  \"$1\" ߟߋ߬ ߟߊ߫",
+       "usermaildisabled": "ߢߎߡߍߙߋ߲ߞߏ߲ߘߏ ߟߊ߬ߓߊ߰ߙߊ߬ߟߌ ߓߘߊ߫ ߓߴߊ߬ ߟߊ߫",
+       "usermaildisabledtext": "ߌ ߕߍ߫ ߣߊ߬ ߛߋ߫ ߟߊ߫ ߢߎߡߍߙߋ߲ߞߏ߲ߘߏ ߗߋ߫ ߟߊ߫ ߟߊ߬ߓߊ߰ߙߊ߬ߟߊ ߘߏ ߟߎ߬ ߡߊ߬ ߥߞߌ ߣߌ߲߬ ߞߊ߲߬",
+       "noemailtitle": "ߢߎߡߍߙߋ߲ߞߏ߲ߘߏ ߛߊ߲߬ߓߊ߬ߕߐ߮ ߕߴߦߋ߲߬",
+       "noemailtext": "ߟߊ߬ߓߊ߰ߙߊ߬ߟߊ ߣߌ߲߬ ߡߊ߫ ߢߎߡߍߙߋ߲ߞߏ߲ߘߏ ߛߊ߲߬ߓߊ߬ߕߐ߮ ߓߍ߲߬ߓߍ߲ ߝߋ߲߫ ߡߊߕߍ߰ ߡߎߣߎ߲߬.",
+       "nowikiemailtext": "ߟߊ߬ߓߊ߰ߙߊ߬ߟߊ ߣߌ߲߬ ߟߊ߬ߓߊ߰ߙߊ߬ߟߊ ߘߏ ߟߎ߬ ߟߊ߫ ߢߎߡߍߙߋ߲ߞߏ߲ߘߏ ߗߋߟߌ ߟߊߟߐ߭ ߛߎߥߊ߲ߘߌ߫ ߊ߬ ߡߊ߬.",
        "emailusername": "ߟߊߓߊ߯ߙߊߟߊߕߐ߮:",
        "email-legend": "ߢߎߡߍߙߋ߲ ߗߋ߫  {{SITENAME}} ߜߘߍ ߟߊ߬ߓߊ߰ߙߊ߬ߟߊ ߘߏ ߟߎ߬ ߡߊ߬",
        "emailfrom": "ߞߊ߬ ߝߘߊ߫",
        "watchlist": "ߣߐ߬ߝߍ߬ߜߍ߲߬ߛߙߍߘߍ",
        "mywatchlist": "ߜߋ߬ߟߎ߲߬ߠߌ߲߬ ߛߙߍߘߍ",
        "watchlistfor2": "ߞߏߛߐ߲߬ $1 $2",
+       "watchnologin": "ߌ ߜߊ߲߬ߞߎ߲߬ߣߍ߲߬ ߕߍ߫",
+       "addwatch": "ߊ߬ ߟߊߘߏ߲߬ ߜߋ߬ߟߎ߲߬ߠߌ߲߬ ߛߙߍߘߍ ߘߐ߫",
+       "addedwatchtext": "\"[[:$1]]\" ߊ߬ ߣߴߊ߬ ߘߊߘߐߖߊߥߏ߫ ߞߐߜߍ ߓߘߊ߫ ߟߊߘߏ߲߬ ߌ ߟߊ߫ [[Special:Watchlist|watchlist]] ߘߐ߫.",
        "watch": "ߊ߬ ߘߐߜߍ߫",
+       "watchthispage": "ߞߐߜߍ ߣߌ߲߬ ߘߐߜߍ߫",
        "unwatch": "ߊ߬ ߞߍ߫ ߦߋߓߊߟߌ ߘߌ߫",
+       "unwatchthispage": "ߊ߬ ߘߐߜߍߟߌ ߟߊߟߐ߬",
+       "notanarticle": "ߊ߬ ߕߍ߫ ߞߣߐߘߐ߫ ߞߐߜߍ߫ ߘߌ߫",
+       "notvisiblerev": "ߟߊ߬ߓߊ߰ߙߊ߬ߟߊ߫ ߓߐߣߍ߲ߢߐ߲߰ߡߊ ߟߎ߬ ߟߊ߫ ߟߢߊ߬ߟߌ߬ ߞߐ߯ߟߕߊ ߟߎ߬ ߓߘߊ߫ ߖߏ߬ߛߌ߬",
        "watchlist-details": "ߌ ߟߊ߫ ߞߣߐ߬ߜߍ߲߬ ߛߙߍߘߍ ߟߎ߬ ߘߐ߫ {{PLURAL:$1|ߞߐߜߍ $1|ߞߐߜߍ ߟߎ߬ $1}}( ߊ߬ ߣߌ߫ ߢߊߝߐߟߌ߫ ߞߐߜߍ ߟߎ߬) ߓߍ߯ ߟߴߊ߬ ߞߣߐ߫.",
+       "wlheader-enotif": "ߢߎߡߍߙߋ߲ߞߏ߲ߘߏ ߟߊ߫ ߡߙߌߣߊ߲߫ ߦߌߘߊ ߓߘߊ߫ ߓߌ߬ߟߵߊ߬ ߟߊ߫.",
        "wlheader-showupdated": "ߞߐߜߍ ߡߍ߲ ߠߎ߬ ߡߊߦߟߍ߬ߡߊ߲߬ߣߍ߲߬ ߌ ߟߊ߫ ߡߊ߬ߝߍ߬ߣߍ߲߬ߠߌ߲ ߞߐ߫߸ ߏ߬ ߟߎ߫ ߟߋ߬ ߦߌ߬ߘߊ߬ߣߍ߲߫ <strong>ߛߓߍߘߋ߲߫ ߞߎ߲ߓߊ</strong> ߘߐ߫.",
        "wlnote": "ߘߎ߰ߟߊ ߘߐ߫ {{PLURAL:$1|is the last change|are the last <strong>$1</strong> changes}} ߞߐ߯ߟߕߊ ߘߐ߫ {{PLURAL:$2|hour|<strong>$2</strong> hours}}, as of $3, $4.",
        "wlshowlast": "ߕߎ߬ߡߊ߬ߙߋ߲ $1 ߞߐߟߕߊ $2 ߕߋ߬ߟߋ ߟߎ߬ ߦߌ߬ߘߊ߬",
+       "watchlist-hide": "ߊ߬ ߢߡߊߘߏ߲߰",
+       "watchlist-submit": "ߊ߬ ߦߌ߬ߘߊ߬",
+       "wlshowtime": "ߊ߬ ߞߊߞߊ߲߫ ߞߊ߬ ߟߊߓߊ߯ߙߊ߫ ߥߎ߬ߛߎ ߡߍ߲ ߞߘߐ߫",
+       "wlshowhideminor": "ߡߊ߬ߦߟߍ߬ߡߊ߲߬ߠߌ߲߬ ߘߋ߬ߣߍ߲ ߠߎ߬",
+       "wlshowhidebots": "ߓߏߕ",
+       "wlshowhideliu": "ߟߊߓߊ߯ߙߊߓߊ߯ ߛߙߍߘߍߦߊߣߍ߲ ߠߎ߬",
+       "wlshowhideanons": "ߡߊ߬ߦߟߍ߬ߡߊ߲߬ߠߊ߫ ߕߐ߯ߒߕߊ߲",
+       "wlshowhidepatr": "ߓߍ߬ߙߍ߲߬ߓߍ߬ߙߍ߲߬ߠߌ߲ ߡߊ߬ߦߟߍ߬ߡߊ߲߬ߠߌ߲",
+       "wlshowhidemine": "ߒ ߠߊ߫ ߡߊ߬ߦߟߍ߬ߡߊ߲߬ߠߌ߲ ߠߎ߬",
+       "wlshowhidecategorization": "ߞߐߜߍ ߦߌߟߡߊߦߊߟߌ",
        "watchlist-options": "ߞߣߐ߬ߜߍ߲߬ ߛߙߍߘߍ ߢߣߊߕߐߡߐ߲ ߠߎ߬",
+       "watching": "ߜߋ߬ߟߎ߲߬ߠߌ߲ ߦߴߌ ߘߐ߫...",
+       "unwatching": "ߦߋߓߊߟߌߦߊߟߌ ߦߴߌ ߘߐ߫...",
        "enotif_reset": "ߞߐߜߍ߫ ߞߎ߲ߝߊ߲ߓߐߣߍ߲ ߠߎ߬ ߓߍ߯ ߕߐ߰ߡߊ߬ߛߙߋ߫",
+       "enotif_impersonal_salutation": "{{SITENAME}} ߟߊߓߊ߯ߙߟߊ",
+       "enotif_subject_deleted": "{{SITENAME}} ߞߐߜߍ $1 ߓߘߊ߫ {{GENDER:ߖߏ߰ߛߌ߫|$2}} $2 ߓߟߏ߫",
+       "enotif_subject_created": "{{SITENAME}} ߞߐߜߍ $1 ߓߘߊ߫ {{GENDER:ߖߏ߰ߛߌ߫|$2}} $2 ߓߟߏ߫",
+       "enotif_subject_moved": "{{SITENAME}} ߞߐߜߍ $1 ߓߘߊ߫ {{GENDER:ߖߏ߰ߛߌ߫|$2}} $2 ߓߟߏ߫",
+       "enotif_subject_restored": "{{SITENAME}} ߞߐߜߍ $1 ߓߘߊ߫ {{GENDER:ߖߏ߰ߛߌ߫|$2}} $2 ߓߟߏ߫",
+       "enotif_subject_changed": "{{SITENAME}} ߞߐߜߍ $1 ߓߘߊ߫ {{GENDER:ߡߊߦߟߍ߬ߡߊ߲߫|$2}} $2 ߓߟߏ߫",
+       "enotif_body_intro_deleted": " {{SITENAME}} ߞߐߜߍ $1 ߓߘߊ߫ {{GENDER:ߖߏ߰ߛߌ߫|$2}} ߓߟߏ߫ $ߞߐߜߍ ߡߊߦߟߍ߬ߡߊ߲߬ ߕߎߡߘߊ $2 ߓߟߏ߫߸ ߣߌ߲߬ $3 ߦߋ߫.",
+       "enotif_body_intro_created": " {{SITENAME}} ߞߐߜߍ $1 ߓߘߊ߫ {{GENDER: ߛߌ߲ߘߌ߫|$2 ߓߟߏ߫}} $ߞߐߜߍ ߡߊߦߟߍ߬ߡߊ߲߬ ߕߎߡߘߊ ߟߊ߫ $2 ߓߟߏ߫߸ $3 ߘߐߜߍ߫ ߕߋ߲߭ߕߋ߲߭ ߟߢߊ߬ߟߌ߬ ߞߏ ߘߐ߫.",
+       "enotif_body_intro_moved": " {{SITENAME}} ߞߐߜߍ ߓߘߊ߫ {{GENDER: ߖߏ߬ߛߌ߫|$2}} ߓߟߏ߫ $ߞߐߜߍ ߡߊߦߟߍ߬ߡߊ߲߬ ߕߎߡߘߊ ߟߊ߫ $2 ߓߟߏ߫߸ $3 ߘߐߜߍ߫ ߕߋ߲߭ߕߋ߲߭ ߟߢߊ߬ߟߌ߬ ߞߏ ߘߐ߫.",
+       "enotif_body_intro_restored": " {{SITENAME}} ߞߐߜߍ $1 ߓߘߊ߫ {{GENDER: ߟߊߞߎߣߎ߲߫|$2 ߓߟߏ߫}} $ߞߐߜߍ ߡߊߦߟߍ߬ߡߊ߲߬ ߕߎߡߘߊ ߟߊ߫ $2 ߓߟߏ߫߸ $3 ߘߐߜߍ߫ ߕߋ߲߭ߕߋ߲߭ ߟߢߊ߬ߟߌ߬ ߞߏ ߘߐ߫.",
+       "enotif_body_intro_changed": " {{SITENAME}} ߞߐߜߍ $1 ߓߘߊ߫ {{GENDER: ߡߊߦߟߍ߬ߡߊ߲߫|$2 ߓߟߏ߫}} $ߞߐߜߍ ߡߊߦߟߍ߬ߡߊ߲߬ ߕߎߡߘߊ ߟߊ߫ $2 ߓߟߏ߫߸ $3 ߘߐߜߍ߫ ߕߋ߲߭ߕߋ߲߭ ߟߢߊ߬ߟߌ߬ ߞߏ ߘߐ߫.",
+       "enotif_lastvisited": "ߡߊ߬ߦߟߍ߬ߡߊ߲߬ߠߌ߲ ߓߍ߯ ߞߵߊ߬ ߕߊ߬ ߌ ߟߊ߫ ߓߐߒߡߊߟߌ߫ ߟߊߓߊ߲ ߡߊ߬߸߫ ߊ߬ ߦߋ߫ $1",
+       "enotif_lastdiff": "ߖߐ߲߬ߛߊ߫ ߌ ߘߌ߫ ߡߊ߬ߦߟߍ߬ߡߊ߲߬ߠߌ߲ ߣߌ߲߬ ߦߋ߫߸ $1 ߘߐߜߍ߫",
+       "enotif_anon_editor": "ߡߊ߬ߦߟߍ߬ߡߊ߲߬ߠߊ߫ ߕߐ߯ߒߕߊ߲ $1",
+       "enotif_minoredit": "ߣߌ߲߬ ߦߋ߫ ߡߊ߬ߦߟߍ߬ߡߊ߲߬ߠߌ߲߬ ߘߋ߬ߣߍ߲ ߘߏ߫ ߟߋ߬ ߘߌ߫",
+       "deletepage": "ߞߐߜߍ ߖߏ߰ߛߌ߬",
+       "confirm": "ߊ߬ ߟߊߛߙߋߦߊ߫",
+       "excontent": "ߞߣߐߘߐ ߕߘߍ߬ ߦߋ߫: \"$1\"",
+       "excontentauthor": "ߞߣߐߘߐ ߕߘߍ߬ ߦߋ߫: \"$1\"߸ ߊ߬ ߣߌ߫ ߡߊ߬ߦߟߍ߬ߡߊ߲߬ߠߊ߫ ߞߋߟߋ߲ ߕߘߍ߬ ߦߋ߫ \"[[Special:Contributions/$2|$2]]\" \" ([[User talk:$2|talk]])",
+       "exbeforeblank": "ߞߣߐߘߐ ߕߘߍ߬ ߦߋ߫ \"$1\" ߟߋ߬ ߘߌ߫ ߖߏ߬ߛߌ߬ߟߌ ߢߍ߫.",
+       "delete-confirm": "ߖߏ߰ߛߌ߬ߟߌ $1",
+       "delete-legend": "ߊ߬ ߖߏ߬ߛߌ߬",
        "dellogpage": "ߜߊ߲߬ߞߎ߲߬ߠߌ߲ ߖߏ߬ߛߌ߬ߣߍ߲",
+       "rollback": "ߡߊ߬ߦߟߍ߬ߡߊ߲߬ߠߌ߲ ߠߊߞߐߛߊ߬ߦߌ߬",
+       "rollback-confirmation-confirm": "ߊ߬ ߟߊߛߙߋߦߊ߫ ߖߊ߰ߣߌ߲߫:",
+       "rollback-confirmation-yes": "ߟߊߞߐߛߊ߬ߦߌ߬",
+       "rollback-confirmation-no": "ߊ߬ ߘߐߛߊ߬",
        "rollbacklink": "ߟߊߞߐߛߊ߬ߦߌ߬",
        "rollbacklinkcount": "ߛߋ߬ߦߌ߲߬ ߌ ߣߐ߭ߡߊ߬ $1 {{PLURAL:$1|ߡߊߦߟߍ߬ߡߊ߲߫|ߟߎ߬ ߡߊߦߟߍ߬ߡߊ߲߫}}",
+       "changecontentmodel-title-label": "ߞߐߜߍ ߞߎ߲߬ߕߐ߮",
+       "changecontentmodel-model-label": "ߞߣߐߘߐ߫ ߛߎ߯ߦߊ߫ ߞߎߘߊ",
+       "changecontentmodel-reason-label": "ߊ߬ ߛߊߓߎ:",
+       "changecontentmodel-submit": "ߊ߬ ߡߊߦߟߍ߬ߡߊ߲߫",
+       "changecontentmodel-success-title": "ߞߣߐߘߐ ߛߎ߯ߦߊ ߓߘߊ߫ ߡߊߦߟߍ߬ߡߊ߲߫",
        "protectlogpage": "ߜߊ߲߬ߞߎ߲߬ߠߌ߲߬ ߠߊߞߊ߲ߘߊߣߍ߲",
        "protectedarticle": "ߟߊ߬ߞߊ߲߬ߘߊ߬ߣߍ߲ \"[[$1]]\"",
        "modifiedarticleprotection": "ߟߊ߬ߞߊ߲߬ߘߊ߬ߟߌ ߞߊߓߋ ߡߊߦߟߍ߬ߡߊ߲߬ \"[[$1]]\"",
+       "protectcomment": "ߊ߬ ߛߊߓߎ:",
+       "protectexpiry": "ߊ߬ ߛߕߊ ߓߘߊ߫ ߝߊ߫:",
+       "protect_expiry_invalid": "ߊ߬ ߛߕߊ ߝߊ߫ ߕߎߡߊ ߓߍ߲߬ߣߍ߲߬ ߕߍ߫.",
+       "protect_expiry_old": "ߊ߬ ߛߕߊ ߝߊ߫ ߕߎߡߊ ߓߘߊ߫ ߕߊ߬ߡߌ߲߬.",
+       "protect-unchain-permissions": "ߟߊ߬ߞߊ߲߬ߘߊ߬ߟߌ ߢߣߊߕߊߟߌ ߣߊ߬ߕߐ ߘߊߦߟߍ߬",
+       "protect-text": "ߌ ߘߌ߫ ߛߋ߫ ߞߐߜߍ ߟߎ߫ ߟߊ߬ߞߊ߲߬ߘߊ߬ߟߌ ߞߊߓߋ ߦߋ߫ ߟߴߊ߬ ߣߌ߫ ߞߵߊ߬ ߡߊߦߟߍ߬ߡߊ߲߫ ߦߊ߲߬ <strong>$1</strong>.",
        "protect-default": "ߟߊ߬ߓߊ߰ߙߊ߬ߟߊ ߟߎ߬ ߓߍ߯ ߟߊߘߌ߬ߢߍ߬",
        "restriction-edit": "ߊ߬ ߡߊߦߟߍ߬ߡߊ߲߬",
        "restriction-move": "ߕߐ߯ߦߊ߫",
index f505038..2aa18bc 100644 (file)
        "rcfilters-clear-all-filters": "Wyczyść filtry",
        "rcfilters-show-new-changes": "Zobacz zmiany od $1",
        "rcfilters-search-placeholder": "Filtruj zmiany (użyj menu lub wyszukaj według nazwy filtra)",
+       "rcfilters-search-placeholder-mobile": "Filtry",
        "rcfilters-invalid-filter": "Nieprawidłowy filtr",
        "rcfilters-empty-filter": "Brak aktywnych filtrów. Wyświetlane są wszystkie zmiany.",
        "rcfilters-filterlist-title": "Filtry",
        "linkaccounts": "Połącz konta",
        "linkaccounts-success-text": "Konto zostało połączone.",
        "linkaccounts-submit": "Połącz konta",
+       "cannotunlink-no-provider-title": "Nie ma żadnych przyłączonych kont do odłączenia",
+       "cannotunlink-no-provider": "Nie ma żadnych przyłączonych kont, które mogą zostać odłączone.",
        "unlinkaccounts": "Odłącz konta",
        "unlinkaccounts-success": "Konta zostały odłączone.",
        "authenticationdatachange-ignored": "Zmiana danych uwierzytelniających nie została obsłużona. Być może nie skonfigurowano dostawcy.",
index d96cbbe..2104e3f 100644 (file)
        "rcfilters-clear-all-filters": "Limpar todos os filtros",
        "rcfilters-show-new-changes": "Ver as novas alterações desde $1",
        "rcfilters-search-placeholder": "Filtrar mudanças (usar o menu ou procurar o nome do filtro)",
+       "rcfilters-search-placeholder-mobile": "Filtros",
        "rcfilters-invalid-filter": "Filtro inválido",
        "rcfilters-empty-filter": "Nenhum filtro ativo. Todas as contribuições são mostradas.",
        "rcfilters-filterlist-title": "Filtros",
        "linkaccounts": "Vincular contas",
        "linkaccounts-success-text": "A conta foi vinculada.",
        "linkaccounts-submit": "Vincular contas",
+       "cannotunlink-no-provider-title": "Não há contas vinculadas para desvincular",
+       "cannotunlink-no-provider": "Não há contas vinculadas que possam ser desvinculadas.",
        "unlinkaccounts": "Desassociar contas",
        "unlinkaccounts-success": "A conta foi desassociada.",
        "authenticationdatachange-ignored": "A alteração de dados de autenticação não foi processada. Talvez nenhum provedor tenha sido configurado?",
index 7756b2d..c0e184d 100644 (file)
        "rcfilters-clear-all-filters": "Limpar todos os filtros",
        "rcfilters-show-new-changes": "Ver novas alterações desde $1",
        "rcfilters-search-placeholder": "Filtrar mudanças (usar o menu ou procurar o nome do filtro)",
+       "rcfilters-search-placeholder-mobile": "Filtros",
        "rcfilters-invalid-filter": "Filtro inválido",
        "rcfilters-empty-filter": "Não há filtros ativos. São mostradas todas as contribuições.",
        "rcfilters-filterlist-title": "Filtros",
        "linkaccounts": "Associar contas",
        "linkaccounts-success-text": "A conta foi associada.",
        "linkaccounts-submit": "Associar contas",
+       "cannotunlink-no-provider-title": "Não há contas associadas para desassociar",
+       "cannotunlink-no-provider": "Não há contas associadas que se possa desassociar.",
        "unlinkaccounts": "Desassociar contas",
        "unlinkaccounts-success": "A conta foi desassociada.",
        "authenticationdatachange-ignored": "A alteração dos dados de autenticação não foi realizada. Talvez o fornecedor não tenha sido configurado?",
index 081bb66..033be0b 100644 (file)
        "rcfilters-clear-all-filters": "Title for the button that clears all filters",
        "rcfilters-show-new-changes": "Label for the button to show new changes. Parameters:\n* $1 - timestamp from which new changes are available. It indicates that clicking the refresh link will bring changes newer than (or equal to) this timestamp. It is formatted according to the user's date, time and timezone preferences",
        "rcfilters-search-placeholder": "Placeholder for the filter search input. The first \"Filter\" is a verb, and the second \"filter\" is a noun.",
+       "rcfilters-search-placeholder-mobile": "Placeholder for the filter search input for mobile devices.",
        "rcfilters-invalid-filter": "A label for an invalid filter.",
        "rcfilters-empty-filter": "Placeholder for the filter list when no filters were chosen.",
        "rcfilters-filterlist-title": "Title for the filters list.\n{{Identical|Filter}}",
        "changecontentmodel": "Title of the change content model special page",
        "changecontentmodel-legend": "Legend of the fieldset on the change content model special page",
        "changecontentmodel-title-label": "Label for the input field where the target page title should be entered\n{{Identical|Page title}}",
+       "changecontentmodel-current-label": "Label for the current content model",
        "changecontentmodel-model-label": "Label of the dropdown listing available content model types the user can change a page to",
        "changecontentmodel-reason-label": "{{Identical|Reason}}",
        "changecontentmodel-submit": "Label of the form \"submit\" button for [[Special:ChangeContentModel]]\n{{Identical|Change}}",
        "block-log-flags-angry-autoblock": "Used as a block log flag in [[Special:Log/block]].\n{{Related|Block-log-flags}}",
        "block-log-flags-hiddenname": "Used as a block log flag in [[Special:Log/block]] and in [[Special:Block]].\n\n{{Related|Block-log-flags}}",
        "range_block_disabled": "Used as error message in [[Special:Block]].\n\nSee also:\n* {{msg-mw|Range block disabled}}\n* {{msg-mw|Ip range invalid}}\n* {{msg-mw|Ip range toolarge}}",
+       "ipb-prevent-user-talk-edit": "Used as error message in [[Special:Block]] if invalid options are selected regarding \"Edit own user talk\".\n\nSee also:\n{{msg-mw|ipb-disableusertalk}}",
        "ipb_expiry_invalid": "Used as error message in [[Special:Block]].",
        "ipb_expiry_old": "Used as error message in [[Special:Block]], if the expiry time is in the past.\n{{Identical|protect_expiry_old}}",
        "ipb_expiry_temp": "Warning message displayed on [[Special:Block]] if the option \"hide username\" is selected but the expiry time is not infinite.",
        "move-page-legend": "Legend of the fieldset around the input form of [[Special:MovePage/testpage]].\n\nSee also:\n* {{msg-mw|newtitle|label for new title}}\n* {{msg-mw|movereason|label for textarea}}\n* {{msg-mw|movetalk|label for checkbox}}\n* {{msg-mw|move-leave-redirect|label for checkbox}}\n* {{msg-mw|fix-double-redirects|label for checkbox}}\n* {{msg-mw|move-subpages|label for checkbox}}\n* {{msg-mw|move-talk-subpages|label for checkbox}}\n* {{msg-mw|move-watch|label for checkbox}}\n{{Identical|Move page}}",
        "movepagetext": "Introduction shown when moving a page ([[Special:MovePage]]).\n\nSpecial pages mentioned: {{msg-mw|Doubleredirects}}, {{msg-mw|Brokenredirects}}\n\nSee also:\n* {{msg-mw|Movepagetext-noredirectfixer}}",
        "movepagetext-noredirectfixer": "A variant of the following message ''Movepagetext'' displayed when the automatic redirect fixer is not enabled.\n\nSpecial pages mentioned: {{msg-mw|Doubleredirects}}, {{msg-mw|Brokenredirects}}\n\nSee also:\n* {{msg-mw|Movepagetext}}",
+       "movepagetext-noredirectsupport": "A variant of the following message ''Movepagetext'' displayed when the content model of the page being moved does not support redirects.\n\nSee also:\n* {{msg-mw|Movepagetext}}",
        "movepagetalktext": "Text on the special 'Move page'. This text only appears if the talk page is not empty.",
        "moveuserpage-warning": "Used as warning in [[Special:MovePage]], when moving a user page.",
        "movecategorypage-warning": "Used as warning in [[Special:MovePage]], when moving a category page.",
        "linkaccounts": "Title of the special page [[Special:LinkAccounts]] which allows the user to connect the local user accounts with external ones such as Google or Facebook.",
        "linkaccounts-success-text": "Text shown on top of the form after a successful action.",
        "linkaccounts-submit": "Text of the main submit button on [[Special:LinkAccounts]] (when there is one)",
+       "cannotunlink-no-provider-title": "Error page title shown when the user visits [[Special:UnlinkAccounts]] but there is no external account that could be unlinked.",
+       "cannotunlink-no-provider": "Error message shown when the user visits [[Special:UnlinkAccounts]] but there is no external account that could be unlinked.",
        "unlinkaccounts": "Title of the special page [[Special:UnlinkAccounts]] which allows the user to remove linked remote accounts.",
        "unlinkaccounts-success": "Account unlinking form success message",
        "authenticationdatachange-ignored": "Shown when authentication data change was unsuccessful due to configuration problems.\n\nCf. e.g. {{msg-mw|Passwordreset-ignored}}.",
index db1bf05..585a532 100644 (file)
        "blockedtext": "<strong>'U nome de l'utende o l'indirizze IP ha state bloccate.</strong>\n\n'U blocche ha state fatte da $1.\n'U mutive date jè <em>$2</em>.\n\n* 'U Blocche accumenze: $8\n* 'U Blocche spicce: $6\n* Tipe de blocche: $7\n\nTu puè condatta $1 o n'otre [[{{MediaWiki:Grouppage-sysop}}|amministratore]] pe 'ngazzarte sus a 'u blocche.\nTu non ge puè ausà 'u strumende \"{{int:emailuser}}\" senza ca mitte n'indirizze email valide jndr'à le\n[[Special:Preferences|preferenze tune]] e ce è state bloccate sus a l'use sue.\nL'IP ca tine mò jè $3 e 'u codece d'u blocche jè #$5.\nPe piacere mitte ste doje 'mbormaziune ce manne 'na richieste de sblocche.",
        "autoblockedtext": "L'indirizze IP tue ha state automaticamende blocchete purcè ha state ausete da n'otre utende, ca avère state blocchete da $1.\n'U mutive date jè 'u seguende:\n\n:''$2''\n\n* Inizie d'u blocche: $8\n* Scadenze d'u blocche: $6\n* Blocche 'ndise: $7\n\nTu puè cundattà $1 o une de l'otre [[{{MediaWiki:Grouppage-sysop}}|amministrature]] pe parà de stu probbleme.\n\nVide Bbuene ca tu non ge puè ausà 'a funziona \"manne n'e-mail a stu utende\" senze ca tu tìne 'n'indirizze e-mail valide e reggistrete jndr'à seziona [[Special:Preferences|me piace accussì]] e tu non ge sinde blocchete da ausarle.\n\nL'indirizze IP corrende jè $3, e 'u codece d'u blocche jè #$5.\nPe piacere mitte tutte le dettaglie ca ponne essere utile pe le richieste tune.",
        "blockednoreason": "nisciune mutive",
+       "blockedtext-composite-no-ids": "'U 'ndirizze IP tune jesse jndr'à 'nu sacche de liste gnore",
+       "blockedtext-composite-reason": "Stonne attive cchiù blocche sus a 'u cunde tune e/o indirizze IP",
        "whitelistedittext": "Tu ha $1 pàggene da cangià.",
        "confirmedittext": "Tu a confermà l'indirizze e-mail tue apprime de cangià le pàggene.\nPe piacere mitte e validesce l'indirizze e-mail tue ausanne le [[Special:Preferences|preferenze de l'utende]].",
        "nosuchsectiontitle": "Non ge pozze acchià 'a sezione",
        "action-changetags": "Aggiunge e live arbitrariamende tag sus a le revisiune individuale e vôsce de l'archivije",
        "action-deletechangetags": "scangille le tag da 'u database",
        "action-purge": "aggiorne sta pàgene",
+       "action-editprotected": "cange le pàggene prutette cumme a \"{{int:protect-level-sysop}}\"",
+       "action-editsemiprotected": "cange le pàggene prutette cumme a \"{{int:protect-level-autoconfirmed}}\"",
        "action-editinterface": "cange l'inderfacce utende",
        "action-editusercss": "cange 'u CSS de l'otre utinde",
        "action-edituserjson": "cange 'u JSON de l'otre utinde",
        "rcfilters-restore-default-filters": "Repristine le filtre de base",
        "rcfilters-clear-all-filters": "Pulizze tutte le filtre",
        "rcfilters-show-new-changes": "'Ndruche le cangiaminde nuève da $1",
+       "rcfilters-search-placeholder-mobile": "Filtre",
        "rcfilters-invalid-filter": "Filtre invalide",
        "rcfilters-empty-filter": "Nisciune filtre attive. Tutte le condrebbute avènene fatte 'ndrucà.",
        "rcfilters-filterlist-title": "Filtre",
        "linkaccounts": "Colleghe le cunde",
        "linkaccounts-success-text": "'U cunde ha state collegate.",
        "linkaccounts-submit": "Colleghe le cunde",
+       "cannotunlink-no-provider-title": "Non ge stonne cunde collegate da scollegà.",
+       "cannotunlink-no-provider": "Non ge stonne cunde collegate ca ponne essere scollegate.",
        "unlinkaccounts": "Scolleghe le cunde",
        "unlinkaccounts-success": "'U cunde ha state scollegate.",
        "specialmute": "Citte",
index 6267b0d..28e4801 100644 (file)
                        "Саша Волохов",
                        "Serhio Magpie",
                        "ЛингвоЧел",
-                       "OlegVeliky"
+                       "OlegVeliky",
+                       "Saimongoltinio"
                ]
        },
        "tog-underline": "Подчёркивание ссылок:",
        "search-interwiki-more": "(ещё)",
        "search-interwiki-more-results": "ещё результаты",
        "search-relatedarticle": "Связанный",
+       "search-invalid-sort-order": "Порядок сортировки в $1 не распознан, будет применена сортировка по умолчанию.  Действительные заказы на сортировку: $2",
+       "search-unknown-profile": "Профиль поиска $1 не распознан, будет применен профиль поиска по умолчанию.",
        "searchrelated": "связанный",
        "searchall": "все",
        "showingresults": "Ниже {{PLURAL:$1|1=показан <strong>1</strong> результат|показан  <strong>$1</strong> результат|показано <strong>$1</strong> результата|показаны <strong>$1</strong> результатов}}, начиная с № <strong>$2</strong>.",
        "rcfilters-clear-all-filters": "Очистить все фильтры",
        "rcfilters-show-new-changes": "Последние изменения, начиная с $1",
        "rcfilters-search-placeholder": "Фильтровать изменения (используйте меню или ищите по названию фильтра)",
+       "rcfilters-search-placeholder-mobile": "Фильтры",
        "rcfilters-invalid-filter": "Недопустимый фильтр",
        "rcfilters-empty-filter": "Нет активных фильтров. Показываются все правки.",
        "rcfilters-filterlist-title": "Фильтры",
        "wlshowhideanons": "анонимных участников",
        "wlshowhidepatr": "проверенные правки",
        "wlshowhidemine": "мои правки",
-       "wlshowhidecategorization": "каÑ\82егоÑ\80изаÑ\86иÑ\8e Ñ\81Ñ\82Ñ\80аниÑ\86",
+       "wlshowhidecategorization": "изменение Ñ\81оÑ\81Ñ\82ава ÐºÐ°Ñ\82егоÑ\80ий",
        "watchlist-options": "Настройки списка наблюдения",
        "watching": "Добавление в список наблюдения…",
        "unwatching": "Удаление из списка наблюдения…",
        "linkaccounts": "Связать учётные записи",
        "linkaccounts-success-text": "Учётная запись была связана.",
        "linkaccounts-submit": "Связать учётные записи",
+       "cannotunlink-no-provider-title": "У вас нет связанных учётных записей, чтобы их отвязать",
+       "cannotunlink-no-provider": "У вас нет связанных учётных записей, которые можно было бы отвязать.",
        "unlinkaccounts": "Отвязать учётные записи",
        "unlinkaccounts-success": "Учётная запись была отвязан.",
        "authenticationdatachange-ignored": "Изменение данных для проверки подлинности не было обработано. Может быть, не был настроен ни один провайдер?",
        "edit-error-short": "Ошибка: $1",
        "edit-error-long": "Ошибки:\n\n$1",
        "specialmute": "Откл. уведомления",
-       "specialmute-success": "Ð\98зменениÑ\8f Ð±Ñ\8bли Ñ\83Ñ\81пеÑ\88но Ñ\81деланы. Просмотрите всех отключённых участников на [[Special:Preferences|ваших настройках]].",
+       "specialmute-success": "Ð\98зменениÑ\8f Ð¿Ð¾ Ð¾Ñ\82клÑ\8eÑ\87ениÑ\8e Ñ\83ведомлений Ð±Ñ\8bли Ñ\81оÑ\85Ñ\80анены. Просмотрите всех отключённых участников на [[Special:Preferences|ваших настройках]].",
        "specialmute-submit": "Подтвердить",
        "specialmute-label-mute-email": "Отключить эл. почту от этого участника",
        "specialmute-header": "Пожалуйста, выберите настройки уведомлений для {{GENDER:$1|участника|участницы}} <b>{{BIDI:[[User:$1|$1]]}}</b>.",
        "specialmute-error-invalid-user": "Указанное вами имя участника не может быть найдено.",
        "specialmute-email-footer": "Для управления настройками эл. почты {{GENDER:$2|участника|участницы}} {{BIDI:$2}}, пожалуйста, посетите <$1>.",
-       "specialmute-login-required": "Пожалуйста, войдите, чтобы совершить изменения.",
+       "specialmute-login-required": "Пожалуйста авторизируйтесь, чтобы управлять отключением уведомлений.",
        "mute-preferences": "Настройки выключения",
        "revid": "версия $1",
        "pageid": "ID страницы $1",
        "passwordpolicies-policy-passwordnotinlargeblacklist": "Пароль не может соответствовать какому-либо из 100 000 самых часто используемых паролей.",
        "passwordpolicies-policyflag-forcechange": "необходимо изменить при входе",
        "passwordpolicies-policyflag-suggestchangeonlogin": "предложить изменение при входе",
+       "mycustomjsredirectprotected": "У вас нет прав на редактирование этой JavaScript-страницы, так как она является перенаправлением и не указывает внутрь вашего пространства участника.",
        "easydeflate-invaliddeflate": "Предоставленное содержимое не спущено надлежащим образом",
        "unprotected-js": "По соображениям безопасности JavaScript нельзя загружать с незащищённых страниц. Пожалуйста, создавайте скрипты только в пространстве имён MediaWiki: или как подстраницы участника.",
        "userlogout-continue": "Вы хотите выйти?"
index 92e10f0..d424923 100644 (file)
        "missingcommenttext": "Задайте коментарь",
        "missingcommentheader": "'''Припомянутя:''' Не задали сьте тему/надпис про тот коментарь.\nКідь іщі раз кликнете на „$1“, буде ваша едітація записана і без того.",
        "summary-preview": "Опис буде:",
-       "subject-preview": "Нагляд темы/надпису:",
+       "subject-preview": "Нагляд темы:",
        "blockedtitle": "Хоснователь є блокованый",
        "blockedtext": "'''Ваше хосновательске мено або IP адреса была заблокована.'''\n\nБлокованя зробив $1.\nПричіна є ''$2''.\n\n* Початок блокованя: $8\n* Конец блокованя: $6\n* Блокованый хоснователь: $7\n\nМожете контактовати $1 або іншого [[{{MediaWiki:Grouppage-sysop}}|адміністратора]].\nУважте, же не можете поужыти 'Послати e-mail', кідь не маєте платну адресу ел. пошты во вашых [[Special:Preferences|наставлїня конта]] і кідь вам тота можность не была заказана.\nВаша IP адреса є $3, і ID блокованя є #$5.\nПросиме Вас, пиште тоты детайлы во вшыткых запытах на адміністратора.",
        "autoblockedtext": "'''Ваше хосновательске мено або IP адреса была заблокована.'''\n\nБлокованя зробив $1.\nПричіна є ''$2''.\n\n* Початок блокованя: $8\n* Конец блокованя: $6\n* Блокованый хоснователь: $7\n\nМожете контактовати $1 або іншого [[{{MediaWiki:Grouppage-sysop}}|адміністратора]].\nУважте, же не можете поужыти 'Послати e-mail', кідь не маєте платну адресу ел. пошты во вашых [[Special:Preferences|наставлїня конта]] і кідь вам тота можность не была заказана.\nВаша IP адреса є $3, і ID блокованя є #$5.\nПросиме Вас, пиште тоты детайлы во вшыткых запытах на адміністратора.",
        "prefs-rc": "Послїднї зміны",
        "prefs-watchlist": "Слїдованы сторінкы",
        "prefs-editwatchlist": "Едітованя списку слїдованых сторінок",
+       "prefs-editwatchlist-label": "Едітованя записів списку слїдованых сторінок:",
+       "prefs-editwatchlist-raw": "Едітованя списку слїдованых сторінок як тексту",
        "prefs-watchlist-days": "Чісло днїв зображеных в слїдованых сторінках:",
        "prefs-watchlist-days-max": "Максімално $1 {{PLURAL:$1|день|днї|днїв}}",
        "prefs-watchlist-edits": "Чісло едітовань зображеных в списку слїдованых сторінок:",
        "right-reupload-own": "Переписованя файлів ладованых од себе самого",
        "right-reupload-shared": "Наладованя локалных файлів жебы перекрыли тотых в сполочнім усховищу",
        "right-upload_by_url": "Наладовованя файлів з URL адрес",
-       "right-purge": "Очіщіня кешу про сторінкы без потверджовачого діалоґу",
+       "right-purge": "Очіщіня кешу сторінкы",
        "right-autoconfirmed": "Не тыкать ся лімітів основаных на IP",
        "right-bot": "Быти поважованый за автоматічный процес",
        "right-nominornewtalk": "Невыписованя новых повідомлїнь по малых управах діскузной сторінкы",
        "right-browsearchive": "Гляданя вымазаных сторінок",
        "right-undelete": "Обновлїня вымазаных сторінок",
        "right-suppressrevision": "Перегляд і обновованя ревізій схованых перед адміністраторами",
+       "right-viewsuppressed": "Перегляд ревізій схованых перед вшыткыма",
        "right-suppressionlog": "Перегляд пріватных записів",
        "right-block": "Блокованя іншых хоснователїв од едітованя",
        "right-blockemail": "Блокованя хоснователя од посыланя е-пошты",
        "right-editmyusercss": "Едітовати вашы властны хосновательскы CSS файлы.",
        "right-editmyuserjson": "Едітовати вашы властны хосновательскы JSON файлы",
        "right-editmyuserjs": "Едітовати вашы властны хосновательскы JavaScript файлы",
+       "right-editmyuserjsredirect": "Едітованя властных хосновательскых JavaScript файлів, котры суть напрямлїня",
        "right-viewmywatchlist": "Перезераня властного списку слїдованых сторінок",
        "right-editmywatchlist": "Едітованя властного списку слїдованых сторінок. Усвідомте собі, же дакотры дїї будуть до нёго придавати сторінкы і без такого права.",
        "right-viewmyprivateinfo": "Перезераня властных пріватных даных (напр. імейлова адреса, правдиве імя)",
        "right-sendemail": "Посыланя пошты іншым хоснователям",
        "right-managechangetags": "Вытворїня і (де)актівація [[Special:Tags|значок]]",
        "right-applychangetags": "Придаваня [[Special:Tags|значок]] до властных змін",
+       "right-changetags": "Придаваня любовольных [[Special:Tags|значок]] на окремы ревізії і лоґы і іх одобераня",
+       "right-deletechangetags": "Мазаня [[Special:Tags|значок]] з датабазы",
        "grant-group-email": "Послати ел. пошту",
        "grant-blockusers": "Блоковати і одблоковати хоснователїв",
        "grant-createaccount": "Закладаня конт",
+       "grant-createeditmovepage": "Вытваряти, едітовати і переменовати сторінкы",
+       "grant-delete": "Мазати сторінкы, ревізії і лоґы",
        "grant-editmywatchlist": "Едітовати ваш список слїдованых сторінок",
+       "grant-editpage": "Едітовати екзістуючі сторінкы",
        "grant-editprotected": "Едітованя замкнутых сторінок",
        "grant-uploadfile": "Наладовати новы файлы",
        "grant-viewdeleted": "Перегляд вымазаных файлів і сторінок",
        "action-viewmyprivateinfo": "перезерати вашы пріватны даны",
        "action-editmyprivateinfo": "едітовати вашы пріватны інформації",
        "action-editcontentmodel": "едітовати модел обсягу сторінкы",
+       "action-changetags": "придавати і одоберати любовольны значкы на окремы ревізії і лоґы",
        "action-apihighlimits": "хоснованя высшых лімітів в API запытах",
        "action-autoconfirmed": "не тыкать ся лімітів основаных на IP",
        "action-bigdelete": "мазаня сторінок з довгов історіёв",
        "action-editmyusercss": "едітовати вашы властны хосновательскы CSS файлы",
        "action-editmyuserjson": "едітовати вашы властны хосновательскы JSON файлы",
        "action-editmyuserjs": "едітовати вашы властны хосновательскы JavaScript файлы",
+       "action-editmyuserjsredirect": "едітованя властных хосновательскых JavaScript файлів, котры суть напрямлїня",
        "action-hideuser": "блокованя мена хоснователя і ёго схованя",
        "action-ipblock-exempt": "обходжіня блокованя IP адрес, їх россягів і автоблокованя",
        "action-unblockself": "одблоковати самого себе",
        "recentchanges-submit": "Вказати",
        "rcfilters-activefilters-hide": "Сховати",
        "rcfilters-activefilters-show": "Вказати",
+       "rcfilters-quickfilters": "Уложены фільтры",
+       "rcfilters-savedqueries-defaultlabel": "Уложены фільтры",
        "rcfilters-savedqueries-rename": "Переменовати",
+       "rcfilters-savedqueries-setdefault": "Наставити як дефолтный",
+       "rcfilters-savedqueries-unsetdefault": "Одобрати як дефолтный",
        "rcfilters-savedqueries-remove": "Вымазати",
        "rcfilters-savedqueries-new-name-label": "Назва",
        "rcfilters-savedqueries-apply-label": "Створїня філтра",
+       "rcfilters-savedqueries-apply-and-setdefault-label": "Створити дефолтный філтер",
+       "rcfilters-savedqueries-cancel-label": "Зрушыти",
+       "rcfilters-savedqueries-add-new-title": "Уложыти сучасны наставлїня філтрів",
+       "rcfilters-restore-default-filters": "Обновити дефолтны філтры",
+       "rcfilters-clear-all-filters": "Зняти вшыткы філтры",
+       "rcfilters-filterlist-title": "Філтры",
+       "rcfilters-filter-editsbyself-label": "Вашы зміны",
+       "rcfilters-filter-editsbyother-label": "Зміны остатніх",
+       "rcfilters-filter-user-experience-level-registered-label": "Зареґістрованы",
+       "rcfilters-filter-user-experience-level-unregistered-label": "Незареґістрованы",
+       "rcfilters-filter-reviewstatus-auto-label": "Автоматічно перевірены",
+       "rcfilters-filter-minor-label": "Малы едітації",
+       "rcfilters-filter-major-label": "Немалы едітації",
+       "rcfilters-filtergroup-watchlist": "Слїдованы сторінкы",
+       "rcfilters-filter-watchlist-watched-label": "На списку слїдованых сторінок",
+       "rcfilters-filter-watchlist-notwatched-label": "Не на списку слїдованых сторінок",
+       "rcfilters-filter-watchlistactivity-seen-label": "Зображены зміны",
+       "rcfilters-filter-pageedits-label": "Едітації сторінок",
+       "rcfilters-filter-newpages-label": "Вытворіня сторінок",
+       "rcfilters-filter-categorization-label": "Зміны катеґорій",
+       "rcfilters-filtergroup-lastrevision": "Актуалны ревізії",
+       "rcfilters-filter-lastrevision-label": "Актуална ревізія",
+       "rcfilters-filter-previousrevision-label": "Не послїдня ревізія",
+       "rcfilters-filter-excluded": "Вылучены",
+       "rcfilters-tag-prefix-namespace-inverted": "<strong>:не</strong> $1",
+       "rcfilters-exclude-button-off": "Вылучити зволене",
+       "rcfilters-exclude-button-on": "Зволены вылучены",
+       "rcfilters-view-tags": "Означены едітації",
        "rcnotefrom": "Долов суть вказаны зміны од <strong>$2</strong> (до <strong>$1</strong>).",
        "rclistfrom": "Вказати едітованя почінаючі з $3 $2.",
        "rcshowhideminor": "$1 маленькы едітованя",
        "rcshowhidemine": "$1 мої едітованя",
        "rcshowhidemine-show": "Вказати",
        "rcshowhidemine-hide": "Сховати",
-       "rclinks": "Вказати послїднї $1 зміны за $2 днїв",
+       "rcshowhidecategorization-show": "Вказати",
+       "rcshowhidecategorization-hide": "Сховати",
+       "rclinks": "Вказати послїднї $1 змін за $2 днїв",
        "diff": "різн.",
        "hist": "історія",
        "hide": "сховати",
        "license-nopreview": "(Нагляд недоступный)",
        "upload_source_url": " (платна, публічно доступна інтернет-адреса)",
        "upload_source_file": " (файл на вашім компютерї)",
+       "listfiles-delete": "змазати",
        "listfiles-summary": "Тота шпеціална сторінка указує вшыткы наладованы файлы.",
        "listfiles_search_for": "Глядати файл по назві:",
+       "listfiles-userdoesnotexist": "Хосновательске конто „$1“ не є реґістроване.",
        "imgfile": "файл",
        "listfiles": "Список файлів",
        "listfiles_thumb": "Мініатура",
        "listfiles_size": "Розмір (в байтах)",
        "listfiles_description": "Опис",
        "listfiles_count": "Верзії",
-       "listfiles-show-all": "Включати стары верзії образчіків",
+       "listfiles-show-all": "Включати стары верзії файлів",
        "listfiles-latestversion": "Актуална верзія",
        "listfiles-latestversion-yes": "Гей",
        "listfiles-latestversion-no": "Нї",
        "randomincategory": "Трафункова сторінка в катеґорії",
        "randomincategory-invalidcategory": "\" $1 \" не є платна назва катеґорії.",
        "randomincategory-nopages": "В [[:Category:$1|катеґорії $1]] не суть ниякы сторінкы.",
+       "randomincategory-category": "Катеґорія:",
+       "randomincategory-legend": "Трафункова сторінка в катеґорії",
+       "randomincategory-submit": "Перейти",
        "randomredirect": "Трафункове напрямлїня",
        "randomredirect-nopages": "Простор назв „$1“ не обсягує жадны напрямлїня.",
        "statistics": "Штатістіка",
        "pageswithprop-prophidden-binary": "значіня бінарной властности є сховане ($1)",
        "doubleredirects": "Двоїты напрямлїня",
        "doubleredirectstext": "На тій сторінцї є список напрямлїн ведучіх на далшы напрямлїня.\nКаждый рядок обсягує одказ на перше і друге напрямлїня і ку тому ціль другого напрямлїня, котрый звычайно вказує мено „реалной“ цілёвой сторінкы, на котру бы ся мало перше напрямлїня одказовати.\n<del>Перечаркнуты</del> положкы уж были вырїшены.",
-       "double-redirect-fixed-move": "Сторінка [[$1]] была переменована, нынї напрямлює на [[$2]]",
+       "double-redirect-fixed-move": "Сторінка [[$1]] была переменована.\nБыла автоматічно актуалізована і нынї напрямлює на [[$2]].",
        "double-redirect-fixed-maintenance": "Корекція двоїтого напрямлїня з [[$1]] на [[$2]].",
        "double-redirect-fixer": "Оправарь напрямлїнь",
        "brokenredirects": "Перерваны напрямлїня",
        "ninterwikis": "$1 {{PLURAL:$1|міджіязыковый одказ|міджіязыковы одказы|міджіязыковых одказів}}",
        "nlinks": "$1 {{PLURAL:$1|одказ|одказы|одказів}}",
        "nmembers": "$1 {{PLURAL:$1|обєкт|обєкты|обєктів}}",
+       "nmemberschanged": "$1 → $2 {{PLURAL:$2|обєкт|обєкты|обєктів}}",
        "nrevisions": "$1 {{PLURAL:$1|ревізія|ревізії|ревізій}}",
        "nimagelinks": "Хоснованый ня $1 {{PLURAL:$1|сторінка|сторінкы|сторінках}}",
        "ntransclusions": "хоснованый ня $1 {{PLURAL:$1|сторінка|сторінкы|сторінках}}",
        "mostrevisions": "Сторінкы з найвеце ревізіями",
        "prefixindex": "Вшыткы сторінкы з початком назв",
        "prefixindex-namespace": "Вшыткы сторінкы з префіксом (простор назв $1)",
+       "prefixindex-submit": "Вказати",
        "prefixindex-strip": "Одсїчі початок назвы в списку",
        "shortpages": "Курты статї",
        "longpages": "Найдовшы статті",
        "deadendpages": "Слїпы сторінкы",
        "deadendpagestext": "Наслїдуючі сторінкы не одказують на жадну іншу сторінку {{grammar:2sg|{{SITENAME}}}}.",
        "protectedpages": "Замкнуты сторінкы",
+       "protectedpages-filters": "Філтры:",
        "protectedpages-indef": "Лем замкы на нестановлено",
        "protectedpages-cascade": "Лем каскадовы замкы",
+       "protectedpages-noredirect": "Сховати напрямлїня",
        "protectedpagesempty": "Жадна сторінка не є замкнута з тыма параметрами.",
        "protectedpages-timestamp": "Часова значка",
+       "protectedpages-page": "Сторінка",
        "protectedpages-expiry": "Кінчіть",
        "protectedpages-reason": "Прічіна",
        "protectedpages-unknown-timestamp": "Не є знаме",
        "protectedpages-unknown-performer": "Незнамый хоснователь",
        "protectedtitles": "Замкнуты назвы сторінок",
        "protectedtitlesempty": "Жадна назва не є замкнута з тыма параметрами.",
+       "protectedtitles-submit": "Зобразити назвы",
        "listusers": "Список хоснователїв",
        "listusers-editsonly": "Вказати лем хоснователїв з едітованями",
        "listusers-creationsort": "Сортовати за датумом створїня",
        "usereditcount": "$1 {{PLURAL:$1|едітованя|едітованя|едітовань}}",
        "usercreated": "{{GENDER:$3|Реґістрованый|Реґістрована|Реґістрованый(а)}} $1 в $2",
        "newpages": "Новы сторінкы",
+       "newpages-submit": "Вказати",
        "newpages-username": "Мено хоснователя:",
        "ancientpages": "Сторінкы, якы ся найдовше не едітовали",
        "move": "Переменовати",
        "pager-older-n": "{{PLURAL:$1|старша|старшы|старшых}} $1",
        "suppress": "Ревізор",
        "querypage-disabled": "Тота шпеціална сторінка є  заблокована про проблемы з выконом.",
+       "apihelp-no-such-module": "Модул „$1“ не найдженый.",
+       "apisandbox-reset": "Вычістити",
+       "apisandbox-retry": "Повторити",
        "booksources": "Жрідла книг",
        "booksources-search-legend": "Гляданя інформації про книгы",
        "booksources-text": "Ниже є список одказів на серверы продаваючі книгы, або котры можуть мати далшы інформації о книгах, котры глядате.",
        "booksources-invalid-isbn": "Дане ISBN ся здасть быти неплатне. Перевірте го з оріґіналным жрідлом.",
        "specialloguserlabel": "Супроводник:",
-       "speciallogtitlelabel": "Цїль (назва або хоснователь):",
+       "speciallogtitlelabel": "Цїль (назва або {{ns:user}}:хосновательске мено):",
        "log": "Лоґы",
+       "logeventslist-submit": "Вказати",
+       "logeventslist-patrol-log": "Книга перевіреных едітовань",
        "all-logs-page": "Вшыткы публічны записы",
        "alllogstext": "Комбіноване зображіня вшыткых доступных протоколёвачіх записів про {{grammar:4sg|{{SITENAME}}}}.\nЗображіня можете зужыти выбером тіпу запису, мена хоснователя (залежыть на великости букв) або зазначеной сторінкы (тыж залежыть на великости букв).",
        "logempty": "Протокол не обсягує жаден одповідаючій запис.",
        "log-title-wildcard": "Глядати назвы зачінаючі ся з тым текстом",
        "showhideselectedlogentries": "Вказати/сховати зволены записы лоґу.",
+       "checkbox-select": "Выбрати: $1",
+       "checkbox-all": "Вшыткы",
+       "checkbox-none": "Жадне",
+       "checkbox-invert": "Інвертовати",
        "allpages": "Вшыткы сторінкы",
        "nextpage": "Далша сторінка ($1)",
        "prevpage": "Попередня сторінка ($1)",
        "cachedspecial-viewing-cached-ts": "Позерати собі кешовану верзію той сторінкы, котра могла стратити актуалность.",
        "cachedspecial-refresh-now": "Вказати найновшы.",
        "categories": "Катеґорії",
+       "categories-submit": "Вказати",
        "categoriespagetext": "{{PLURAL:$1|Наслїдуюча катеґорія|Наслїдуючі катеґорії}} {{PLURAL:$1|обсягує|обсягують}} сторінкы або файлы.\nНевказаны суть то [[Special:UnusedCategories|нехоснованы катеґорії]].\nПосмотьте ся тыж на [[Special:WantedCategories|жаданы катеґорії]].",
        "categoriesfrom": "Вказати сторінкы, што ся зачінають на:",
        "deletedcontributions": "Вымазаны приспевкы хоснователя",
        "watchlistanontext": "Про перезераня ці управу списку слїдованых сторінок ся мусите $1.",
        "watchnologin": "Не сьте приголошеный(а)",
        "addwatch": "Придати до списку слїдованых сторінок",
-       "addedwatchtext": "Сторінка «[[:$1]]» была придана до вашого [[Special:Watchlist|списку слїдованых]].\nБудучі зміны той статї і з нёв повязаной сторінков діскузії будуть зображены у тім списку.",
+       "addedwatchtext": "Сторінка «[[:$1]]» і її сторінка діскузії была придана до вашого [[Special:Watchlist|списку слїдованых]].",
+       "addedwatchtext-talk": "Сторінка «[[:$1]]» і її повязана сторінка была придана до вашого [[Special:Watchlist|списку слїдованых]].",
+       "addedwatchtext-short": "Сторінка «$1» была придана до вашого списку слїдованых.",
        "removewatch": "Вымазати зо списку слїдованых сторінок",
-       "removedwatchtext": "Сторінка «[[:$1]]» вымазана з вашого [[Special:Watchlist|списку слїдованых сторінок]].",
+       "removedwatchtext": "Сторінка «[[:$1]]» і її сторінка діскузії вымазана з вашого [[Special:Watchlist|списку слїдованых сторінок]].",
+       "removedwatchtext-talk": "Сторінка «[[:$1]]» і її привязана сторінка вымазана з вашого [[Special:Watchlist|списку слїдованых сторінок]].",
+       "removedwatchtext-short": "Сторінка «$1» была вымазана з вашого списку слїдованых.",
        "watch": "Слїдовати",
        "watchthispage": "Слїдовати тоту сторінку",
        "unwatch": "Зрушыти слїдованя",
        "wlheader-enotif": "Упозорнїня  ел. поштов є запнуте.",
        "wlheader-showupdated": "Сторінкы, котры ся змінили од вашой послїднёй навщівы суть вказаны '''грубо'''",
        "wlnote": "Ниже є {{PLURAL:$1|остатня зміна|остатнї $1 зміны|остатнїх $1 змін}} за {{PLURAL:$2|остатнїй|остатнї|остатнїх}} <b>$2</b> {{PLURAL:$2|годину|годины|годин}} до do $4, $3.",
-       "wlshowlast": "Вказати зміны за послїднїх $1 годин $2 днїв",
+       "wlshowlast": "Вказати послїднїх $1 годин $2 днїв",
+       "watchlist-hide": "Сховати",
+       "watchlist-submit": "Вказати",
+       "wlshowhideminor": "малы едітації",
+       "wlshowhidebots": "боты",
+       "wlshowhideliu": "реґістрованы хоснователї",
+       "wlshowhideanons": "анонімны хоснователї",
+       "wlshowhidepatr": "перевірены едітованя",
        "watchlist-options": "Наставлїна списку слїдованых",
        "watching": "Придаваня до списку слїдованя...",
        "unwatching": "Одобратя зо списку слїдованя...",
        "enotif_body_intro_moved": "В $PAGEEDITDATE {{gender:$2|переменовав|переменовала}} $2 на {{grammar:6sg|{{SITENAME}}}} сторінку $1, посмотьте актуалну верзію на $3 .",
        "enotif_body_intro_restored": "В $PAGEEDITDATE {{gender:$2|обновив|обновила}} $2 на {{grammar:6sg|{{SITENAME}}}} сторінку $1, посмотьте актуалну верзію на $3 .",
        "enotif_body_intro_changed": "В $PAGEEDITDATE {{gender:$2|змінив|змінила}} $2 на {{grammar:6sg|{{SITENAME}}}} сторінку $1, посмотьте актуалну верзію на $3 .",
-       "enotif_lastvisited": "Ð\92идÑ\8cÑ\82е $1 Ð¿Ñ\80о Ñ\81пиÑ\81ок Ð²Ñ\88Ñ\8bÑ\82кÑ\8bÑ\85 Ð·Ð¼Ñ\96н Ð¾Ð´ Ð¼Ð¸Ð½Ñ\83лой Ð½Ð°Ð²Ñ\89Ñ\96вÑ\8b.",
+       "enotif_lastvisited": "Ð\9fÑ\80о Ñ\81пиÑ\81ок Ð²Ñ\88Ñ\8bÑ\82кÑ\8bÑ\85 Ð·Ð¼Ñ\96н Ð¾Ð´ Ð¼Ð¸Ð½Ñ\83лой Ð½Ð°Ð²Ñ\89Ñ\96вÑ\8b Ð²Ð¸Ð´Ñ\8cÑ\82е $1",
        "enotif_lastdiff": "Тоту зміну видьте на $1",
        "enotif_anon_editor": "анонімный хоснователь $1",
        "enotif_body": "Честованый хоснователю $WATCHINGUSERNAME,\n\n$PAGEINTRO $NEWPAGE\n\nЗгорнутя едітованя: $PAGESUMMARY $PAGEMINOREDIT\n\nХоснователя, котрый зміну учінив, можете контактовати:\nімейлом: $PAGEEDITOR_EMAIL\nна вікі: $PAGEEDITOR_WIKI\n\nПокы сторінку не навщівите, не будуть вам заганяны далшы повідомлїня о змінах той сторінкы. Тыж собі можете вынуловати признакы у своїм списку слїдованых сторінок.\n\n\tПоздравує вас ваш посылач голошіня {{grammar:2sg|{{SITENAME}}}}\n\n--\nЗмінити наштелёваня імейловых повідомлїнь можете на\n{{canonicalurl:{{#special:Preferences}}}}\n\nНаштелёваня слїдованых сторінок можете змінити на\n{{canonicalurl:Special:Watchlist/edit}}\n\nСторінку можете зо своїх слїдованых вышмарити на\n$UNWATCHURL\n\nПорада і контакт:\n$HELPPAGE",
+       "enotif_minoredit": "Тото є мала зміна",
        "created": "створена",
        "changed": "змінена",
        "deletepage": "Змазати сторінку",
        "confirm": "Потверджіня",
        "excontent": "обсяг быв: „$1“",
-       "excontentauthor": "обсяг быв: „$1“ (і єдиным приспівателём быв „[[Special:Contributions/$2|$2]]“)",
+       "excontentauthor": "обсяг быв: „$1“, і єдиным приспівателём быв „[[Special:Contributions/$2|$2]]“ ([[User talk:$2|діскузія]])",
        "exbeforeblank": "обсяг перед выпорожнїнём быв: „$1“",
        "delete-confirm": "Змазаня  $1",
        "delete-legend": "Вымазати",
-       "historywarning": "'''Варованя:''' Сторінка, котру хочете змазати, має історію з приближно $1 {{PLURAL:$1|ревізії|ревізіями}}:",
+       "historywarning": "<strong>Варованя:</strong> Сторінка, котру хочете змазати, має історію з $1 {{PLURAL:$1|ревізії|ревізіями}}:",
+       "historyaction-submit": "Вказати ревізії",
        "confirmdeletetext": "Рыхтуєте ся вымазати сторінку і вшыткы єй лоґы едітовань.\nПросиме Вас, потвердьте, же справды тото хочете зробити, повно розумієте наслїдкы і же робите тото в одповідности з [[{{MediaWiki:Policy-url}}|правилами]].",
        "actioncomplete": "Дїя выконана",
        "actionfailed": "Операція ся не вдала",
        "dellogpage": "Лоґ вымазаня",
        "dellogpagetext": "Ту є список послїднїх змазаных сторінок.",
        "deletionlog": "Лоґ вымазаня",
+       "log-name-create": "Лоґ вытворіня сторінок",
+       "log-description-create": "Ту є список послїднїх вытвореных сторінок.",
+       "logentry-create-create": "$1 {{GENDER:$2|вытворив|вытворила}} сторінку $3",
        "reverted": "Обновлїня скоршой верзії",
        "deletecomment": "Причіна:",
        "deleteotherreason": "Інша/додаткова причіна:",
        "confirm-watch-top": "Придати тоту сторінку до списку слїдованых?",
        "confirm-unwatch-button": "ОК",
        "confirm-unwatch-top": "Выняти тоту сторінку зо слїдованых?",
+       "confirm-rollback-top": "Вратити едітованя той сторінкы?",
+       "confirm-mcrrestore-title": "Обновити ревізію",
+       "confirm-mcrundo-title": "Вернути зміну",
        "imgmultipageprev": "← попередня сторінка",
        "imgmultipagenext": "далша сторінка →",
        "imgmultigo": "Перейти!",
        "special-characters-group-khmer": "Кгмерськы",
        "mw-widgets-dateinput-placeholder-day": "РРРР-ММ-ДД",
        "mw-widgets-dateinput-placeholder-month": "РРРР-ММ",
-       "log-action-filter-all": "Вшыткы"
+       "log-action-filter-all": "Вшыткы",
+       "mycustomjsredirectprotected": "Немаєте права едітовати тоту сторінку з JavaScript-ом, бо є напрямлїня мимо ваш хосновательскый простор."
 }
index ae331b9..e9f3a4f 100644 (file)
@@ -11,7 +11,8 @@
                        "Uharteko",
                        "Taxandru",
                        "Macofe",
-                       "Zoranzoki21"
+                       "Zoranzoki21",
+                       "Pierpao"
                ]
        },
        "tog-underline": "Sutalìnia sos ligòngios",
        "listfiles-delete": "Cantzella",
        "listfiles_search_for": "Chirca figuras pro nùmene:",
        "imgfile": "file",
-       "listfiles": "Lista de is files",
+       "listfiles": "Lista de is documentos",
        "listfiles_thumb": "Miniatura",
        "listfiles_date": "Data",
        "listfiles_name": "Nùmene",
index 64968da..15ee048 100644 (file)
        "tog-extendwatchlist": "تازه ترين بدران سموريون تبديليون ڏيکارڻ لاءِ زير نظر فهرست کي وسيع ڪريو.",
        "tog-numberheadings": "سُرخين کي خودڪاراً نمبر ڏيو",
        "tog-editondblclick": "ٻٽي ڪلڪ تي صفحا سنواريو",
-       "tog-editsectiononrightclick": "Ú\80اڱÙ\86 Ø¬Ù\8a Ø¹Ù\86Ù\88اÙ\86 Ú©Ù\8a Ø±Ø§Ø¦Ù\8aÙ½ ÚªÙ\84Úª ÚªØ±Ù\8a ØªØ±Ù\85Ù\8aÙ\85 ÚªØ±Ú» Ù\88ارÙ\8a Ø³Ù\87Ù\88Ù\84ت Ú©Ù\8a ÚªØ§Ø±Ú¯Ø± Ø¨Ú»Ø§Ù\8aÙ\88.",
+       "tog-editsectiononrightclick": "Ú\80اڱÙ\86 Ø¬Ù\8a Ø¹Ù\86Ù\88اÙ\86 Ú©Ù\8a Ø±Ø§Ø¦Ù\8aÙ½ ÚªÙ\84Úª ÚªØ±Ù\8a Ø³Ù\86Ù\88ارڻ Ù\88ارÙ\8a Ø³Ú¾Ù\88Ù\84ت Ú©Ù\8a ÚªØ§Ø±Ú¯Ø± Ø¨Ú»Ø§Ù\8aÙ\88",
        "tog-watchcreations": "منھنجا سرجيل صفحا ۽ منھنجا چاڙھيل فائيل منھنجي نظر ۾ فھرست تي رکو",
        "tog-watchdefault": "منھنجا ترميميل صفحا ۽ فائيل  منھنجي نظر ۾ فھرست ۾ رکو",
        "tog-watchmoves": "جيڪي صفحا ۽ فائيل آءُٗ چوريان، سي منهنجي نظر ۾ فھرست ۾ شامل ڪريو",
        "tog-watchdeletion": "آءُٗ جيڪي صفحا ۽ فائيل  ڊاهيان، سي منهنجي نظر ۾ فھرست تي رکو",
        "tog-watchuploads": "منھنجا نوان چاڙهيل فائيلس نظر ۾ فھرست ۾ شامل ڪريو",
        "tog-watchrollback": "انهن صفحن کي منهنجي نظر ۾ فھرست تي رکو، جن ۾ تبديلين کي مون واپس ورايو آهي",
-       "tog-minordefault": "سÙ\85Ù\88رÙ\8aÙ\86 ØªØ¨Ø¯Ù\8aÙ\84Ù\8aÙ\86 Ú©Ù\8a Ø¨Ù\86اÙ\86 Ú\86ئÙ\8a Ù\85عÙ\85Ù\88Ù\84Ù\8a ØªØ±Ù\85Ù\8aÙ\85 ØªØµÙ\88ر ÚªØ±يو",
-       "tog-previewontop": "ترÙ\85Ù\8aÙ\85Ù\8a Ø¯Ù»Ù\8aØ¡Ù\8e Ù\85ٿاÙ\86 Ù¾Ù\8aØ´ نگاھ ڏيکاريو",
-       "tog-previewonfirst": "پهرين ترميم تي پيش نگاھ ڏيکاريو",
+       "tog-minordefault": "سÙ\85Ù\88رÙ\8aÙ\86 Ø³Ù\86Ù\88ارÙ\86 Ú©Ù\8a Ù¾Ø§Ú»Ù\85رادÙ\88 Ù\85عÙ\85Ù\88Ù\84Ù\8a Ø·Ù\88ر Ù\86شاÙ\86 Ù\84ڳايو",
+       "tog-previewontop": "سÙ\86Ù\88ار Ø¯Ù»Ù\8aØ¡Ù\8e Ù\85ٿاÙ\86 Ù¾Ù\8aØ´-نگاھ ڏيکاريو",
+       "tog-previewonfirst": "پھرين سنوار تي پيش-نگاھ ڏيکاريو",
        "tog-enotifwatchlistpages": "منهنجي نظر ۾ فھرست اندر شامل ڪنهن صفحي يا فائيل ۾ تبديل پيش اچي مون کي برقٽپال اماڻيو",
        "tog-enotifusertalkpages": "منهنجي مباحثي صفحي ۾ تبديليءَ جي صورت ۾ مون کي برقٽپال اماڻيو",
        "tog-enotifminoredits": "صفحن ۾ معمولي ترميمن جي صورت ۾ بہ مون کي برقٽپال ڪريو",
        "tog-oldsig": "توھان جو موجوده دستخط:",
        "tog-fancysig": "صحيح کي وڪيٽيڪسٽ سمجھو (ڪنھن خوڪار ڳنڍڻي کانسواءِ)",
        "tog-uselivepreview": "صفحي کي ٻيھر اتارڻ سواءِ پيش نگاھ ڏيکاريو",
-       "tog-forceeditsummary": "جÙ\8aÚªÚ\8fÙ\87Ù\86 ØªØ±Ù\85Ù\8aÙ\85Ù\8a تَتُ خالي ڇڏيل هجي تہ مونکي چتاءُ ڏيو",
-       "tog-watchlisthideown": "Ù\86ظر Û¾ Ù\81ھرست Ù\85اÙ\86 Ù\85Ù\86Ú¾Ù\86جÙ\88Ù\86 ØªØ±Ù\85Ù\8aÙ\85ون لڪايو",
-       "tog-watchlisthidebots": "Ù\86ظر Û¾ Ù\81ھرست ØªØ§Ù\86 Ø¨Ù\88Ù½ Ø¬Ù\88Ù\86 ØªØ±Ù\85Ù\8aÙ\85ون لڪايو",
-       "tog-watchlisthideminor": "Ù\86ظر Û¾ Ù\81ھرست ØªØ§Ù\86 Ù\85عÙ\85Ù\88Ù\84Ù\8a ØªØ±Ù\85Ù\8aÙ\85ون لڪايو",
-       "tog-watchlisthideliu": "داخÙ\84 Ù¿Ù\8aÙ\84 Ù\88اپرائÙ\8aÙ\86دÚ\99Ù\86 Ø¬Ù\88Ù\86 ØªØ±Ù\85Ù\8aÙ\85ون نظر ۾ فھرست مان لڪايو",
-       "tog-watchlisthideanons": "Ù\86ظر Û¾ Ù\81ھرست ØªØ§Ù\86 Ø§Ú»Ú\84اتÙ\84 Ù\88اپرائÙ\8aÙ\86دÚ\99 Ø¬Ù\88Ù\86 ØªØ±Ù\85Ù\8aÙ\85ون لڪايو",
-       "tog-watchlisthidepatrolled": "Ù\86ظر Û¾ Ù\81ھرست Ù\85اÙ\86 Ú¯Ø´Øª ÚªÙ\8aÙ\84 ØªØ±Ù\85Ù\8aÙ\85ون لڪايو",
+       "tog-forceeditsummary": "جÙ\8aÚªÚ\8fÙ\87Ù\86 Ø³Ù\86Ù\88ار Ø¬Ù\88 تَتُ خالي ڇڏيل هجي تہ مونکي چتاءُ ڏيو",
+       "tog-watchlisthideown": "Ù\86ظر Û¾ Ù\81ھرست Ù\85اÙ\86 Ù\85Ù\86Ú¾Ù\86جÙ\88Ù\86 Ø³Ù\86Ù\88ارون لڪايو",
+       "tog-watchlisthidebots": "Ù\86ظر Û¾ Ù\81ھرست ØªØ§Ù\86 Ø¨Ù\88Ù½ Ø¬Ù\88Ù\86 Ø³Ù\86Ù\88ارون لڪايو",
+       "tog-watchlisthideminor": "Ù\86ظر Û¾ Ù\81ھرست ØªØ§Ù\86 Ù\85عÙ\85Ù\88Ù\84Ù\8a Ø³Ù\86Ù\88ارون لڪايو",
+       "tog-watchlisthideliu": "داخÙ\84 Ù¿Ù\8aÙ\84 Ù\88اپرائÙ\8aÙ\86دÚ\99Ù\86 Ø¬Ù\88Ù\86 Ø³Ù\86Ù\88ارون نظر ۾ فھرست مان لڪايو",
+       "tog-watchlisthideanons": "Ù\86ظر Û¾ Ù\81ھرست ØªØ§Ù\86 Ø§Ú»Ú\84اتÙ\84 Ù\88اپرائÙ\8aÙ\86دÚ\99 Ø¬Ù\88Ù\86 Ø³Ù\86Ù\88ارون لڪايو",
+       "tog-watchlisthidepatrolled": "Ù\86ظر Û¾ Ù\81ھرست Ù\85اÙ\86 Ú¯Ø´Øª ÚªÙ\8aÙ\84 Ø³Ù\86Ù\88ارون لڪايو",
        "tog-watchlisthidecategorization": "صفحن جا زمرا لڪايو",
        "tog-ccmeonemails": "ٻين واپرائيندڙن ڏانھن منهنجي موڪليل برقٽپال جو پرت مون کي اماڻيو",
        "tog-diffonly": "تفاوت هيٺان صفحي جو مواد نہ ڏيکاريو",
        "tog-showhiddencats": "لڪل زمرا ڏيکاريو",
        "tog-norollbackdiff": "واپس ورائڻ کان پوءِ تفاوت نہ ڏيکاريو",
-       "tog-useeditwarning": "مونکي خبردار ڪريو جڏهن مان هڪ ترميم وارو صفحو بغير تبديلين سانڍڻ جي ڇڏيان",
+       "tog-useeditwarning": "جڏهن مان ڪو سنوار وارو صفحو بغير تبديلين سانڍڻ جي ڇڏيان تہ مونکي چِتاءُ ڏيو",
        "tog-prefershttps": "هميشہ محفوظ ڪنيڪشن استعمال ڪريو جڏهن داخل ٿيل هجو",
        "underline-always": "سدائين",
        "underline-never": "ڪڏهن بہ نہ",
        "view": "ڏسو",
        "view-foreign": "$1 تي ڏسو",
        "edit": "سنواريو",
-       "edit-local": "مقامي تشريح کي ترميميو",
+       "edit-local": "مقامي تشريح سنواريو",
        "create": "سرجيو",
        "create-local": "مقامي تشريح ڏيو",
        "delete": "ڊاھيو",
-       "undelete_short": "اڻڊاهيو {{PLURAL:$1|هڪ ترميم|$1 ترميمون}}",
-       "viewdeleted_short": "Ú\8fسÙ\88 {{PLURAL:$1|Ù\87Úª Ú\8aاٺÙ\84 ØªØ±Ù\85Ù\8aÙ\85|$1 Ú\8aاٺÙ\84 ØªØ±Ù\85Ù\8aÙ\85ون}}",
+       "undelete_short": "{{PLURAL:$1|هڪ سنوار|$1 سنوارون}} اڻڊاھيو",
+       "viewdeleted_short": "Ú\8fسÙ\88 {{PLURAL:$1|Ù\87Úª Ú\8aاٿÙ\84 Ø³Ù\86Ù\88ار|$1 Ú\8aاٿÙ\84 Ø³Ù\86Ù\88ارون}}",
        "protect": "تحفظيو",
        "protect_change": "مٽايو",
        "unprotect": "تحفظ بدلايو",
        "currentevents-url": "Project:ھاڻوڪا واقعا",
        "disclaimers": "غيرجوابداريناما",
        "disclaimerpage": "Project:عام غيرجوابدارينامو",
-       "edithelp": "مدد براءِ ترميم",
+       "edithelp": "سنوارڻ ۾ مدد",
        "helppage-top-gethelp": "مدد",
        "mainpage": "مک صفحو",
        "mainpage-description": "مک صفحو",
        "confirmable-no": "نہ",
        "thisisdeleted": "$1 کي ڏسندئو يا بحاليندئو؟",
        "viewdeleted": "$1 ڏسندئو؟",
-       "restorelink": "{{PLURAL:$1|Ù\87ÚªÚ\99Ù\8a Ú\8aاٿÙ\84 ØªØ±Ù\85Ù\8aÙ\85|$1 Ú\8aاٿÙ\84 ØªØ±Ù\85Ù\8aÙ\85ون}}",
+       "restorelink": "{{PLURAL:$1|Ù\87ÚªÚ\99Ù\8a Ú\8aاٿÙ\84 Ø³Ù\86Ù\88ار|$1 Ú\8aاٿÙ\84 Ø³Ù\86Ù\88ارون}}",
        "feedlinks": "روان رسد:",
        "site-rss-feed": "$1 آرايسايس روان رسد",
        "site-atom-feed": "$1 اڻو روان رسد",
        "viewsourcetext": "توهان هن صفحي جو ڪوڊ ڏسي ۽ نقل ڪري سگھو ٿا.",
        "protectedinterface": "هي صفحو سافٽ ويئر جو انٽرفيس متعين ڪري ٿو ۽ غلط استعال کان بچڻ لاءِ ان کي تحفظيو ويو آهي.\nتمام وڪي ۾ ترجمو شامل ڪرڻ لاءِ يا هن ۾ تبديلي ڪرڻ لاءِ ميڊياوڪي ترجمو [https://translatewiki.net/ translatewiki.net] استعمال ڪيو.",
        "namespaceprotected": "توهان کي نانءُپولار <strong>$1</strong> جا صفحا سنوارڻ جا اختيار ناهن.",
-       "sitecssprotected": "اوهان وٽ سيـ.ايسـ.ايسـ صفحي کي ترميم ڪرڻ جا اختيار ناهن، ڇو تہ اهڙيون ترميمون سڀني گھمندڙن کي متاثر ڪري سگھن ٿيون.",
+       "sitecssprotected": "اوهان وٽ ھن سيايسايس صفحي کي سنوارڻ جي اجازت ناھي، ڇو تہ ان سان سڀني گھمندڙ متاثر ٿي سگھن ٿا.",
        "mycustomcssprotected": "توهان کي هيءُ CSS صفحو سنوارڻ جي اجازت نہ آهي.",
        "mycustomjsprotected": "توهان کي هيءُ جاوا اسڪرپٽ صفحو سنوارڻ جي اجازت حاصل ڪانهي.",
        "myprivateinfoprotected": "توهان کي پنهنجي ذاتي معلومات سنوارڻ جي اجازت حاصل نہ آهي.",
        "createacct-another-submit": "کاتو کوليو",
        "createacct-continue-submit": "کاتو کولڻ جاري رکو",
        "createacct-another-continue-submit": "کاتو کولڻ جاري رکو",
-       "createacct-benefit-heading": "{{SITENAME}} توھان جھڙن سڄڻن ٺاھيو آھي.",
-       "createacct-benefit-body1": "{{PLURAL:$1|ترÙ\85Ù\8aÙ\85|ترÙ\85Ù\8aÙ\85ون}}",
+       "createacct-benefit-heading": "{{SITENAME}} کي توھان جھڙن ماڻھن ٺاھيو آھي.",
+       "createacct-benefit-body1": "{{PLURAL:$1|سÙ\86Ù\88ار|سÙ\86Ù\88ارون}}",
        "createacct-benefit-body2": "{{PLURAL:$1|صفحو|صفحا}}",
        "createacct-benefit-body3": "ھاڻوڪا {{PLURAL:$1|ڀاڱيدار}}",
        "badretype": "توهان جو ڄاڻايل ڳجھولفظ درست نہ آهي.",
        "noname": "توهان جو ڄاڻايل واپرائيندڙ-نانءُ ناقابل ڪار آهي.",
        "loginsuccesstitle": "داخل ٿيل",
        "loginsuccess": "'''هاڻي توهان {{SITENAME}} تي بطور \"$1\" داخل ٿيل آهيو.'''",
-       "nosuchuser": "\"$1\" نالي سان ڪو بہ واپرائيندڙ نہ آهي.\nننڍن وڏن اکرن ۾ امتياز ڪرڻ لازمي آهي. \nهِجي چڪاسيو، يا [[Special:CreateAccount|نئون کاتو تخليق ڪريو]].",
+       "nosuchuser": "\"$1\" نالي سان ڪو بہ واپرائيندڙ نہ آهي.\nننڍن وڏن اکرن ۾ امتياز ڪرڻ لازمي آهي. \nهِجي چڪاسيو، يا [[Special:CreateAccount|نئون کاتو کوليو]].",
        "nosuchusershort": "\"$1\" نالي ڪو بہ واپرائيندڙ ناهي.\nپنھنجي هِجي جي پڪ ڪندا.",
        "nouserspecified": "توهان کي ڪو واپرائيندڙ-نان‎ءُ ڄاڻائڻو پوندو.",
        "login-userblocked": "هيءُ واپرائيندڙ بندشيل آهي. داخل ٿيڻ جي اجازت نٿي ڏجي.",
        "password-name-match": "توهان جو ڳجھولفظ توهان جي واپرائيندڙ-نانءَ کان مختلف هجڻ لازمي آھي.",
        "mailmypassword": "ڳجھولفظ ٻيھر مقرر ڪريو",
        "passwordremindertitle": "{{SITENAME}} لاءِ نئون عارضي ڳجھولفظ",
-       "passwordremindertext": "ÚªÙ\86Ú¾Ù\86 (شاÙ\8aد ØªÙ\88ھاÙ\86 Ø¢Ø¡Ù\90Ù¾Ù\8a Ù¾ØªÙ\8a $1 ØªØ§Ù\86) Ø§Ø³Ø§Ù\86 Ú©Ù\8a {{SITENAME}} ($4) Ù\84اءÙ\90 Ù\86ئÙ\88Ù\86 Ú³Ø¬Ú¾Ù\88Ù\84Ù\81ظ Ø§Ù\85اڻڻ Ø¬Ù\8a Ú¯Ù\8fھرÙ\8e ÚªØ¦Ù\8a.\"$2\" Ù\88اپرائÙ\8aÙ\86دÚ\99 Ù\84اءÙ\90 Ú¾Úª Ú³Ø¬Ú¾Ù\88Ù\84Ù\81ظ ØªØ®Ù\84Ù\8aÙ\82 ÚªÙ\8aÙ\88 Ù\88Ù\8aÙ\88 آهي \"$3\" تي ترتيب ڏنو ويو ھو. جيڪڏھن اھو توھان جو ارادو ھيو، تہ ھاڻي توھان کي ھينئر ئي داخل ٿي پنھنجو ڳجھولفظ تبديل ڪرڻ گھرجي.\nتوھان جو عارضي ڳجھولفظ {{PLURAL:$5|هڪ ڏينھُن|$5 ڏينھَن}} ۾ ختم ٿيندو.\n\nجيڪڏھن اھا گُھرَ اوھان نہ ڪئي ھئي، يا ھاڻي اوھان کي پنھنجو ڳجھولفظ ياد اچي ويو آھي ۽ توھان ان کي تبديل ڪرڻ نٿا چاھيو، تہ توھان ھن نياپي کي نظر انداز ڪندي پنھنجو پراڻو ڳجھولفظ ئي استعمال ڪري سگھو ٿا.",
+       "passwordremindertext": "ÚªÙ\86Ú¾Ù\86 (آءÙ\90Ù\8a Ù¾ØªÙ\8a $1 ØªØ§Ù\86) Ø§Ø³Ø§Ù\86 Ú©Ù\8a {{SITENAME}} ($4) Ù\84اءÙ\90 Ù\86ئÙ\88Ù\86 Ú³Ø¬Ú¾Ù\88Ù\84Ù\81ظ Ø§Ù\85اڻڻ Ø¬Ù\8a Ú¯Ù\8fھرÙ\8e ÚªØ¦Ù\8a.\"$2\" Ù\88اپرائÙ\8aÙ\86دÚ\99 Ù\84اءÙ\90 Ú¾Úª Ú³Ø¬Ú¾Ù\88Ù\84Ù\81ظ Ø³Ø±Ø¬Ù\8aÙ\88 Ù\88Ù\8aع آهي \"$3\" تي ترتيب ڏنو ويو ھو. جيڪڏھن اھو توھان جو ارادو ھيو، تہ ھاڻي توھان کي ھينئر ئي داخل ٿي پنھنجو ڳجھولفظ تبديل ڪرڻ گھرجي.\nتوھان جو عارضي ڳجھولفظ {{PLURAL:$5|هڪ ڏينھُن|$5 ڏينھَن}} ۾ ختم ٿيندو.\n\nجيڪڏھن اھا گُھرَ اوھان نہ ڪئي ھئي، يا ھاڻي اوھان کي پنھنجو ڳجھولفظ ياد اچي ويو آھي ۽ توھان ان کي تبديل ڪرڻ نٿا چاھيو، تہ توھان ھن نياپي کي نظر انداز ڪندي پنھنجو پراڻو ڳجھولفظ ئي استعمال ڪري سگھو ٿا.",
        "noemail": "واپرائيندڙ \"$1\" جو ڪو بہ برقٽپال پتو درج ٿيل ناهي.",
        "noemailcreate": "توھان کي قابلڪار برقٽپال پتو مھيا ڪرڻو پوندو.",
        "passwordsent": "واپرائيندڙ \"$1\" لاءِ ھڪ نئون ڳجھولفظ برقٽپال ذريعي اماڻيو ويو آهي.\nمھرباني ڪري اھو حاصل ڪرڻ بعد داخل ٿيندا.",
        "cannotchangeemail": "هن وڪيءَ تي کاتيدار جو برقٽپال پتو بدلائي نہ ٿو سگھجي.",
        "emaildisabled": "هيءَ سرزمين برقٽپال اماڻي نہ ٿي سگھي.",
        "accountcreated": "کاتو کلي چڪو",
-       "accountcreatedtext": "واپرائيندڙ کاتو براءِ [[{{ns:User}}:$1|$1]] ([[{{ns:User talk}}:$1|بحث]]) جي لاءِ تخليق ٿي چڪو آهي.",
+       "accountcreatedtext": "[[{{ns:User}}:$1|$1]] ([[{{ns:User talk}}:$1|بحث]]) لاءِ واپرائيندڙ کلي چڪو آهي.",
        "createaccount-title": "{{SITENAME}} تي کاتو کولڻ",
        "login-throttled": "توهان تازو ئي داخل ٿيڻ جون هيڪانديون گھڻيون ڪوششون ڪيون آهن. مهرباني ڪري $1 لاءِ ترسي پوءِ وري ڪوشش ڪريو.",
        "login-abort-generic": "توهان جو داخل ٿيڻ ناڪام ويو - بند ڪيل",
        "hr_tip": "افقي لڪير (ڪفايت سان استعمال ڪريو)",
        "summary": "تَتُ:",
        "subject": "موضوع:",
-       "minoredit": "Ú¾Ù\8aØ¡Ù\8e Ú¾Úª Ù\85عÙ\85Ù\88Ù\84Ù\8a ØªØ±Ù\85Ù\8aÙ\85 آھي",
+       "minoredit": "Ú¾Ù\8aØ¡Ù\8e Ú¾Úª Ù\85عÙ\85Ù\88Ù\84Ù\8a Ø³Ù\86Ù\88ار آھي",
        "watchthis": "هيءُ صفحو نظر ۾ رکو",
        "savearticle": "صفحو سانڍيو",
        "savechanges": "تبديليون سانڍيو",
        "preview": "پيش نگاھ",
        "showpreview": "پيش نگاھ",
        "showdiff": "تبديليون ڏيکاريو",
-       "blankarticle": "<strong>چتاءُ:</strong> اوهان خيالي صفحو تخليق ڪري رهيا آهيو.\nجيڪڏهن اوهان «$1» تي ٻيهر ڪلڪ ڪريو، ته هي صفحو بغير ڪنهن مواد جي ڇاپيو ويندو.",
-       "anoneditwarning": "<strong>چتاءُ:</strong> توھان داخل ٿيل نہ آھيو. توھان جو آءِپي پتو عوامي طور ظاھر ٿيندو جي توھان ڪي ترميمون ڪريو ٿا. جيڪڏھن توھان <strong>[$1 داخل ٿيو]</strong> ٿا يا <strong>[$2 کاتو کوليو]</strong> ٿا، تہ ٻين فائدن سان گڏ توھان جون ترميمون توھان جي واپرائيندڙ-نانءَ سان منسوب ڪيون وينديون.",
+       "blankarticle": "<strong>چِتاءُ:</strong> اوهان خالي صفحو سرجي رهيا آهيو.\nجيڪڏهن اوهان «$1» تي ٻيھر ڪلڪ ڪيو، تہ هي صفحو بغير ڪنھن مواد جي سرجيو ويندو.",
+       "anoneditwarning": "<strong>چِتاءُ:</strong> توھان داخل ٿيل نہ آھيو. توھان جو آءِپي پتو عوامي طور ظاھر ٿيندو جي توھان ڪي سنوارون ڪريو ٿا. جيڪڏھن توھان <strong>[$1 داخل ٿيو]</strong> ٿا يا <strong>[$2 کاتو کوليو]</strong> ٿا، تہ ٻين فائدن سان گڏ توھان جون سنوارون توھان جي واپرائيندڙ-نانءَ سان منسوب ڪيون وينديون.",
        "anonpreviewwarning": "توهان داخل ٿيل نہ آهيو. جيڪڏهن توهان صفحي ۾ تبديليون سانڍيون تہ اهڙين تبديلين ساڻ توهان جو آءِپي پتو درج ڪيو ويندو.",
        "missingcommenttext": "براءِ مھرباني ڪو تاثر درج ڪندا.",
        "summary-preview": "تت جي پيش نگاھ:",
        "accmailtitle": "ڳجھولفظ اماڻجي چڪو",
        "newarticle": "(نئون)",
        "newarticletext": "توھان اھڙي صفحي جو ڳنڍڻو وٺي ھتي پھتا آھيو، جيڪو اڃا وجود نٿو رکي.\nاھڙو صفحو جوڙڻ لاءِ، ھيٺين دٻي ۾ لکڻ شروع ڪريو (وڌيڪ ڄاڻڻ لاءِ [$1 امدادي صفحو] ڏسندا).\nجي توھان ھتي غلطيءَ ۾ اچي ويا آهيو، تہ رڳو پنھنجي جھانگُوءَ جي <strong>back</strong> بٽڻ تي ٽڙڪ ڪريو.",
-       "noarticletext": "Ù\81Ù\8aâ\80\8cاÙ\84Ù\88Ù\82ت Ù\87Ù\86 ØµÙ\81Ø­Ù\8a Ø§Ù\86در ÚªÙ\88 Ø¨Û\81 Ù½Ù\8aڪسٽ Ù\86Û\81 Ø¢Ù\87Ù\8a.\nتÙ\88Ù\87اÙ\86 Ù»Ù\8aÙ\86 ØµÙ\81Ø­Ù\86 Û¾ [[Special:Search/{{PAGENAME}}|search Ø³Ø§Ú³Ù\8a Ø¹Ù\86Ù\88اÙ\86 Ø¬Ù\8a Ú³Ù\88Ù\84ا]] ÚªØ±Ù\8a Ø³Ú¯Ú¾Ù\88 Ù¿Ø§Ø\8c  \n<span class=\"plainlinks\">[{{fullurl:{{#Special:Log}}|page={{FULLPAGENAMEE}}}} Ù\84اڳاپÙ\8aÙ\84 Ù\84اگس Û¾ Ú³Ù\88Ù\84Ù\8aÙ\88\8c\nÙ\8aا [{{fullurl:{{FULLPAGENAME}}|action=edit}} Ù\87Ù\8aØ¡Ù\8f ØµÙ\81Ø­Ù\88 ØªØ±Ù\85Ù\8aÙ\85يو]</span>.",
-       "noarticletext-nopermission": "Ú¾Ù\86 Ù\88Ù\82ت Ú¾Ù\86 ØµÙ\81Ø­Ù\8a Û¾  ÚªØ§ Ø¨Û\81 Ù\84کت Ù\86Û\81 Ø¢Ú¾Ù\8a.\nتÙ\88ھاÙ\86 Ù»Ù\8aÙ\86 ØµÙ\81Ø­Ù\86 Û¾ [[Special:Search/{{PAGENAME}}|Ú¾Ù\86 ØµÙ\81Ø­Ù\8a Ø¬Ù\8a Ø¹Ù\86Ù\88اÙ\86 Ø³Ø§Ù\86 Ú³Ù\88Ù\84ا ÚªØ±Ù\8a Ø³Ú¯Ú¾Ù\88 Ù¿Ø§]]Ø\8c Ù\8aا <span class=\"plainlinks\">[{{fullurl:{{#Special:Log}}|page={{FULLPAGENAMEE}}}} Ù\84اڳاپÙ\8aÙ\84 Ù\84اگس Ú³Ù\88Ù\84Ù\8aÙ\88]</span>Ø\8c Ù¾Ø± ØªÙ\88ھاÙ\86 Ú©Ù\8a Ø§Ù\86 Ø¬Ù\8a ØªØ®Ù\84Ù\8aÙ\82 ÚªØ±ڻ جي اجازت نہ آھي.",
+       "noarticletext": "Ù\81Ù\8aâ\80\8cاÙ\84Ù\88Ù\82ت Ù\87Ù\86 ØµÙ\81Ø­Ù\8a Ø§Ù\86در ÚªÙ\88 Ø¨Û\81 Ù½Ù\8aڪسٽ Ù\86Û\81 Ø¢Ù\87Ù\8a.\nتÙ\88Ù\87اÙ\86 Ù»Ù\8aÙ\86 ØµÙ\81Ø­Ù\86 Û¾ [[Special:Search/{{PAGENAME}}|search Ø³Ø§Ú³Ù\8a Ø¹Ù\86Ù\88اÙ\86 Ø¬Ù\8a Ú³Ù\88Ù\84ا]] ÚªØ±Ù\8a Ø³Ú¯Ú¾Ù\88 Ù¿Ø§Ø\8c  \n<span class=\"plainlinks\">[{{fullurl:{{#Special:Log}}|page={{FULLPAGENAMEE}}}} Ù\84اڳاپÙ\8aÙ\84 Ù\84اگس Û¾ Ú³Ù\88Ù\84Ù\8aÙ\88\8c\nÙ\8aا [{{fullurl:{{FULLPAGENAME}}|action=edit}} Ù\87Ù\8aØ¡Ù\8f ØµÙ\81Ø­Ù\88 Ø³Ø±Ø¬يو]</span>.",
+       "noarticletext-nopermission": "Ú¾Ù\86 Ù\88Ù\82ت Ú¾Ù\86 ØµÙ\81Ø­Ù\8a Û¾  ÚªØ§ Ø¨Û\81 Ù\84کت Ù\86Û\81 Ø¢Ú¾Ù\8a.\nتÙ\88ھاÙ\86 Ù»Ù\8aÙ\86 ØµÙ\81Ø­Ù\86 Û¾ [[Special:Search/{{PAGENAME}}|Ú¾Ù\86 ØµÙ\81Ø­Ù\8a Ø¬Ù\8a Ø¹Ù\86Ù\88اÙ\86 Ø³Ø§Ù\86 Ú³Ù\88Ù\84ا ÚªØ±Ù\8a Ø³Ú¯Ú¾Ù\88 Ù¿Ø§]]Ø\8c Ù\8aا <span class=\"plainlinks\">[{{fullurl:{{#Special:Log}}|page={{FULLPAGENAMEE}}}} Ù\84اڳاپÙ\8aÙ\84 Ù\84اگس Ú³Ù\88Ù\84Ù\8aÙ\88]</span>Ø\8c Ù¾Ø± ØªÙ\88ھاÙ\86 Ú©Ù\8a Ø§Ù\86 Ø¬Ù\8a Ø³Ø±Ø¬ڻ جي اجازت نہ آھي.",
        "missing-revision": "صفحي \"{{FULLPAGENAME}}\" جو نمبر #$1 وجود نٿو رکي.\n\nاڪثر اهو تڏهن ٿيندو آهي، جڏهن اوهان ڪنهن پراڻي ڳنڍڻي تان اچو يا صفحو [{{fullurl:{{#Special:Log}}/delete|page={{FULLPAGENAMEE}}}} ڊاٺو] ويو هجي.\n\n.",
        "userpage-userdoesnotexist-view": "واپرائيندڙ کاتو $1 درج ٿيل نہ آهي.",
        "blocked-notice-logextract": "هيءُ واپرائيندڙ في‌الحال بندشيل آهي.\nتازو بندش لاگ حوالي طور پيش ڪجي ٿو:",
        "updated": "(تجديديل)",
        "note": "<strong>نوٽ:</strong>",
-       "previewnote": "<strong>Ù\87Ù\8aØ¡Ù\8e Ù\81Ù\82Ø· Ù¾Ù\8aØ´ Ù\86گاھ Ø¢Ù\87Ù\8a.</strong>\nتÙ\88ھاÙ\86 Ø¬Ù\88Ù\86 ØªØ±Ù\85Ù\8aÙ\85ون اڃان نہ سانڍيون ويون آھن!",
-       "continue-editing": "ترÙ\85Ù\8aÙ\85 گاھ ڏانھن وڃو",
+       "previewnote": "<strong>Ù\8aاد Ø±Ú©Ù\88 Ù\87Ù\8aØ¡Ù\8e Ù\81Ù\82Ø· Ù¾Ù\8aØ´-Ù\86گاھ Ø¢Ù\87Ù\8a.</strong>\nتÙ\88ھاÙ\86 Ø¬Ù\88Ù\86 Ø³Ù\86Ù\88ارون اڃان نہ سانڍيون ويون آھن!",
+       "continue-editing": "سÙ\86Ù\88ار گاھ ڏانھن وڃو",
        "editing": "$1 سنواريندي",
        "creating": "$1 سرجيندي",
-       "editingsection": "ترÙ\85Ù\8aÙ\85 Ú¾Ù\8aÙº $1 (سÙ\8aڪشÙ\86)",
-       "editingcomment": "ترميم هيٺ $1 (نئون سيڪشن)",
-       "editconflict": "ترÙ\85Ù\8aÙ\85Ù\8a تڪرار: $1",
+       "editingsection": "سÙ\86Ù\88ار Ú¾Ù\8aÙº $1 (Ú\80اڱÙ\88)",
+       "editingcomment": "(نئون ڀاڱو) $1 سنواريندي",
+       "editconflict": "سÙ\86Ù\88ار تڪرار: $1",
        "yourtext": "توهان جو متن",
        "storedversion": "سانڍيل مسودو",
        "yourdiff": "تفاوت",
-       "copyrightwarning": "ياد رکندا ته {{SITENAME}} لاءِ سموريون ڀاڱيداريون $2 تحت پڌريون ڪجن ٿيون (تفصيلن لاءِ $1 ڏسندا). اوهان جي تحرير کي {{SITENAME}} جي قائدن تحت ترميمي سگهجي ٿو. جيڪڏهن اوهان نه ٿا چاهيو ته اوهان جي لکڻين کي بي رحميءَ سان ترميميو وڃي يا ورهائي عام ڪيو وڃي ته پوءِ پنهنجي لکڻي هتي جمع نه ڪرايو. پنهنجو مواد هتي جمع ڪرڻ جو مطلب هوندو ته توهان کي جمع ڪرايل مواد جي مفت فراهمي ۽ کُليل تبديليءَ تي ڪو به اعتراز ناهي.<br />\nتوهان اهڙي پڪ ڏيڻ جا پابند پڻ آهيو ته توهان جو جمع ڪرايل مواد توهان جو پنهنجو لکيل آهي يا وري توهان ڪنهن مفت وسيلي تان ڪاپي ڪيو آهي.\n'''تحفظيل حق ۽ واسطا رکندڙ مواد واسطيدار مالڪ کان اڳواٽ اجازت وٺڻ کان سواءِ هتي جمع نه ڪريو.'''",
-       "copyrightwarning2": "Ù\8aاد Ø±Ú©Ù\86دا ØªÛ\81 {{SITENAME}} Ù\84اءÙ\90 Ø³Ù\85Ù\88رÙ\8aÙ\88Ù\86 Ú\80اڱÙ\8aدارÙ\8aÙ\86 Ú©Ù\8a Ù»Ù\8aا Ú\80اڱÙ\8aدار Ø³Ù\86Ù\88ارÙ\8aØ\8c Ø¨Ø¯Ù\84ائÙ\8aØ\8c Ù\8aا Ú\8aاÙ\87Ù\8a Ø³Ú¯Ú¾Ù\86 Ù¿Ø§. Ø¬Ù\8aÚªÚ\8fÙ\87Ù\86 Ø§Ù\88Ù\87اÙ\86 Ù\86Û\81 Ù¿Ø§ Ú\86اÙ\87Ù\8aÙ\88 ØªÛ\81 Ø§Ù\88Ù\87اÙ\86 Ø¬Ù\8a Ù\84Ú©Ú»Ù\8aÙ\86 Ú©Ù\8a Ø¨Ù\8a Ø±Ø­Ù\85Ù\8aØ¡Ù\8e Ø³Ø§Ù\86 ØªØ±Ù\85Ù\8aÙ\85Ù\8aÙ\88 وڃي يا ورهائي عام ڪيو وڃي تہ پوءِ پنهنجي لکڻي هتي جمع نہ ڪرايو.</br>\nتوهان اهڙي پڪ ڏيڻ جا پابند پڻ آهيو تہ توهان جو جمع ڪرايل مواد توهان جو پنهنجو لکيل آهي يا وري توهان ڪنهن اهڙي ئي مفت عوامي وسيلي تان ڪاپي ڪيو آهي. (تفصيلن لاءِ $1 ڏسندا).\n\n<strong>تحفظيل حق ۽ واسطا رکندڙ مواد واسطيدار مالڪ کان اڳواٽ اجازت وٺڻ بنان هتي جمع نہ ڪريو.</strong>",
+       "copyrightwarning": "ياد رکندا تہ {{SITENAME}} لاءِ سموريون ڀاڱيداريون $2 تحت پڌريون ڪجن ٿيون (تفصيلن لاءِ $1 ڏسندا). اوهان جي تحرير کي {{SITENAME}} جي قائدن تحت سنواري سگهجي ٿو. جيڪڏهن اوهان نٿا چاهيو تہ اوهان جي لکڻين کي بي رحميءَ سان سنواريو وڃي يا ورهائي عام ڪيو وڃي تہ پوءِ پنهنجي لکڻي هتي جمع نہ ڪرايو. پنهنجو مواد هتي جمع ڪرڻ جو مطلب هوندو تہ توهان کي جمع ڪرايل مواد جي مفت فراهمي ۽ کُليل تبديليءَ تي ڪوبہ اعتراز ناهي.<br />\nتوهان اهڙي پڪ ڏيڻ جا پابند پڻ آهيو تہ توهان جو جمع ڪرايل مواد توهان جو پنهنجو لکيل آهي يا وري توهان ڪنهن مفت وسيلي تان ڪاپي ڪيو آهي.\n'''تحفظيل حق ۽ واسطا رکندڙ مواد واسطيدار مالڪ کان اڳواٽ اجازت وٺڻ کان سواءِ هتي جمع نہ ڪريو.'''",
+       "copyrightwarning2": "Ù\8aاد Ø±Ú©Ù\86دا ØªÛ\81 {{SITENAME}} Ù\84اءÙ\90 Ø³Ù\85Ù\88رÙ\8aÙ\88Ù\86 Ú\80اڱÙ\8aدارÙ\8aÙ\86 Ú©Ù\8a Ù»Ù\8aا Ú\80اڱÙ\8aدار Ø³Ù\86Ù\88ارÙ\8aØ\8c Ø¨Ø¯Ù\84ائÙ\8aØ\8c Ù\8aا Ú\8aاÙ\87Ù\8a Ø³Ú¯Ú¾Ù\86 Ù¿Ø§. Ø¬Ù\8aÚªÚ\8fÙ\87Ù\86 Ø§Ù\88Ù\87اÙ\86 Ù\86Û\81 Ù¿Ø§ Ú\86اÙ\87Ù\8aÙ\88 ØªÛ\81 Ø§Ù\88Ù\87اÙ\86 Ø¬Ù\8a Ù\84Ú©Ú»Ù\8aÙ\86 Ú©Ù\8a Ø¨Ù\8a Ø±Ø­Ù\85Ù\8aØ¡Ù\8e Ø³Ø§Ù\86 Ø³Ù\86Ù\88ارÙ\8aÙ\88Ù\86 وڃي يا ورهائي عام ڪيو وڃي تہ پوءِ پنهنجي لکڻي هتي جمع نہ ڪرايو.</br>\nتوهان اهڙي پڪ ڏيڻ جا پابند پڻ آهيو تہ توهان جو جمع ڪرايل مواد توهان جو پنهنجو لکيل آهي يا وري توهان ڪنهن اهڙي ئي مفت عوامي وسيلي تان ڪاپي ڪيو آهي. (تفصيلن لاءِ $1 ڏسندا).\n\n<strong>تحفظيل حق ۽ واسطا رکندڙ مواد واسطيدار مالڪ کان اڳواٽ اجازت وٺڻ بنان هتي جمع نہ ڪريو.</strong>",
        "protectedpagewarning": "<strong>چتاءُ: هيءَ صفحو اهڙيءَ ريت تحفظيو ويو آهي جو فقط منتظمين ئي ان کي سنواري سگھن ٿا. </strong>\nتازه ترين لاگ حوالي طور پيش ڪجي ٿو:",
        "semiprotectedpagewarning": "<strong>نوٽ:</strong> هيءَ صفحو اهڙيءَ ريت تحفظيو ويو آهي جو فقط خودڪار نموني پڪ ڪيل واپرائيندڙ ئي ان کي سنواري سگھن ٿا.\nتازه ترين لاگ حوالي طور پيش ڪجي ٿو:",
        "templatesused": "هن صفحي تي استعمال ٿيندڙ {{PLURAL:$1|سانچو|سانچا}}:",
        "edit-conflict": "سنوار تڪرار",
        "postedit-confirmation-created": "هيءُ صفحو سرجي چڪو آهي.",
        "postedit-confirmation-restored": "هيءُ صفحو بحالجي چڪو آهي.",
-       "postedit-confirmation-saved": "توهان جي ترميم سانڍجي چڪي آهي.",
+       "postedit-confirmation-saved": "توھان جي سنوار سانڍجي وئي ھئي.",
        "edit-already-exists": "نئون صفحو سرجي نہ سگھيو. اهو اڳ ۾ ئي وجود رکي ٿو.",
        "invalid-content-data": "ناقابل ڪار موادي اعداد",
        "content-not-allowed-here": "\"$1\" مواد جي هن صفحي [[:$2]] جي جڳھ \"$3\" تي اجازت ناھي.",
        "content-json-empty-array": "خالي اري",
        "duplicate-args-warning": "چتاءُ: [[:$2]] کي [[:$1]] ڪال ڪري رهيو آهي، جنهن منجھہ ’$3‘ نيم‌پيما لاءِ هڪ کان وڌيڪ قدر ڄاڻايل آهن. فقط آخري ڄاڻايل قدر استعمال ڪيو ويندو.",
        "parser-template-loop-warning": "سانچو چڪر لڌو ويو: [[$1]]",
-       "cantcreateaccount-text": "هن آءِ پي پتي تان کاتي جي تخليق تي يُوز (<strong>$1</strong>)  [[User:$3|$3]] روڪ لڳائي آهي.\n\n$3 جو ڄاڻايل سبب آهي <em>$2</em> آهي.",
+       "cantcreateaccount-text": "هن آءِپي پتي تان کاتي جي کولڻ تي (<strong>$1</strong>)  [[User:$3|$3]] بندش وڌل آهي.\n\n$3 جو ڄاڻايل سبب ھي <em>$2</em> آهي.",
        "cantcreateaccount-range-text": "آءِپي پتن جي حد <strong>$1</strong> ۾ [[User:$3|$3]] کاتو کولڻ تي روڪ لڳائي وئي آهي،$4 جنهن ۾ توهان جو آءِپي پتو بہ (<strong>$4</strong>)،  پڻ شامل آهي. \n\n$3 ان روڪَ جو سبب \"$2\" ڄاڻايو آهي.",
        "viewpagelogs": "هن صفحي جا لاگس ڏسو",
        "nohistory": "هن صفحي جي ڪا بہ سوانح نہ آهي.",
        "currentrev": "هاڻوڪو مسودو",
        "currentrev-asof": "$1 جو تازو ترين مسودو",
        "revisionasof": "$1 وارو پرت",
-       "revision-info": "$1 ØªÙ\8a {{GENDER:$6|$2}}$7 Ø¬Ù\8a ØªØ±Ù\85Ù\8aÙ\85 بعد مسودو",
+       "revision-info": "$1 Ø¬Ù\88 {{GENDER:$6|$2}}$7 Ø¬Ù\8a Ø³Ù\86Ù\88ار بعد مسودو",
        "previousrevision": "←اڃا پراڻو پرت",
        "nextrevision": "اڃا نئون پرت→",
        "currentrevisionlink": "هاڻوڪو پرت",
        "history-show-deleted": "رڳو ڊاٺل مسودا",
        "histfirst": "اوائلي ترين",
        "histlast": "تازه ترين",
-       "historysize": "({{PLURAL:$1|1 بائيٽ|$1 بائيٽس}})",
+       "historysize": "({{PLURAL:$1|1 بائيٽ|$1 بائيٽون}})",
        "historyempty": "خالي",
-       "history-feed-title": "ترميمي سوانح",
-       "history-feed-description": "وڪي جي هن صفحي جي ترميمي سوانح",
+       "history-feed-title": "ورجاءُ سوانح",
+       "history-feed-description": "وڪي جي هن صفحي جي ورجاءُ سوانح",
        "history-feed-item-nocomment": "$2 تي $1",
-       "rev-deleted-comment": "(ترÙ\85Ù\8aÙ\85Ù\8a تَتُ ھٽايل)",
+       "rev-deleted-comment": "(سÙ\86Ù\88ار Ø¬Ù\88 تَتُ ھٽايل)",
        "rev-deleted-user": "(واپرائيندڙ-نانءُ ڊاٿو ويو)",
        "rev-deleted-event": "(لاگ تفصيل هٽايا ويا)",
-       "rev-deleted-user-contribs": "[Ù\88اپرائÙ\8aÙ\86دÚ\99\86اÙ\86Ø¡Ù\8f Ù\8aا Ø¢Ø¡Ù\90Ù¾Ù\90Ù\8a Ù¾ØªÙ\88 Ù\85Ù\90ٽاÙ\8aÙ\88 Ù\88Ù\8aÙ\88 - Ú\80اڱÙ\8aدارÙ\8aÙ\86 Ù\85اÙ\86 ØªØ±Ù\85Ù\8aÙ\85Ù\88Ù\86 Ù\84ڪاÙ\8aÙ\88Ù\86 Ù\88Ù\8aÙ\88Ù\86]",
-       "rev-suppressed-no-diff": "توهان اهو تفاوت ڏسي نہ ٿا سگھو، ڇاڪاڻ تہ ڪا هڪ ترميم <strong> ڊهي چڪي </strong> آهي.",
+       "rev-deleted-user-contribs": "[Ù\88اپرائÙ\8aÙ\86دÚ\99\86اÙ\86Ø¡Ù\8f Ù\8aا Ø¢Ø¡Ù\90Ù¾Ù\90Ù\8a Ù¾ØªÙ\88 Ù\85Ù\90ٽاÙ\8aÙ\88 Ù\88Ù\8aÙ\88 - Ú\80اڱÙ\8aدارÙ\8aÙ\86 Ù\85اÙ\86 Ø³Ù\86Ù\88ار Ù\84ڪائÙ\8a Ù\88ئÙ\8a]",
+       "rev-suppressed-no-diff": "توهان اهو تفاوت نٿا ڏسي سگھو، ڇاڪاڻ تہ مسودن مان ڪو ھڪ <strong>ڊاھيو ويو آھي</strong>.",
        "rev-delundel": "نمائش تبديل ڪريو",
        "rev-showdeleted": "ڏيکاريو",
        "revisiondelete": "مسوادا ڊاهيو/اڻ‌ڊاهيو",
        "revdelete-legend": "نمائش جون پابنديون ترتيب ڪريو",
        "revdelete-hide-image": "فائيل جو مواد لڪايو",
        "revdelete-hide-name": "هدف ۽ نيمپيما لڪايو",
-       "revdelete-hide-comment": "ترÙ\85Ù\8aÙ\85 جو تتُ",
+       "revdelete-hide-comment": "سÙ\86Ù\88ار جو تتُ",
        "revdelete-hide-user": "ايڊيٽر جو واپرائيندڙ-نانءُ/آءِپِي پتو",
        "revdelete-hide-restricted": "منتظمن توڙي ٻين کان مليل اعداد دٻايو",
        "revdelete-radio-same": "(نہ بدلايو)",
        "mergehistory-box": "ٻن صفحن جي ڀيرن کي ضم ڪريو:",
        "mergehistory-from": "ذريعہ صفحو:",
        "mergehistory-into": "مقصود صفحو:",
-       "mergehistory-list": "ضÙ\85ائتÙ\8a ØªØ±Ù\85Ù\8aÙ\85 سوانح",
-       "mergehistory-go": "ضم ڪرڻ لائق ترميمون ڏيکاريو",
+       "mergehistory-list": "ضÙ\85ائتÙ\8a Ø³Ù\86Ù\88ار سوانح",
+       "mergehistory-go": "ضم ڪرڻ جوڳيون سنوارون ڏيکاريو",
        "mergehistory-submit": "ڀيرن کي ضم ڪريو",
        "mergehistory-empty": "ڪي بہ ڀيرا ضم ڪري نہ ٿا سگھجن.",
        "mergehistory-no-source": "مصدر صفحو $1 وجود نٿو رکي.",
        "mergehistory-reason": "سبب:",
        "mergelog": "ضم لاگ",
        "revertmerge": "اڻ ضم",
-       "history-title": "\"$1\" جي ترميمي سوانح",
+       "history-title": "\"$1\" جي ورجاءُ سوانح",
        "difference-title": "\"$1\" جي مسودن ۾ تفاوت",
        "difference-title-multipage": "صفحن \"$1\" ۽ \"$2\" ۾ تفاوت",
        "difference-multipage": "(صفحن درميان تفاوت)",
        "shown-title": "$1 {{PLURAL:$1|نتيجو|نتيجا}} في صفحو ڏيکاريو",
        "viewprevnext": "ڏسو ($1 {{int:pipe-separator}} $2) ($3)",
        "searchmenu-exists": "<strong>ھن وڪيءَ تي \"[[:$1]]\" نالي ھڪ صفحو آھي.</strong> {{PLURAL:$2|0=|ٻيا لڌل ڳولا نتيجا پڻ ڏسو.}}",
-       "searchmenu-new": "<strong>ھن وڪيءَ تي صفحو تخليق ڪريو \"[[:$1]]\"!</strong> {{PLURAL:$2|0=|توھان جي ڳولا سان لڌل صفحو ڏسو.|لڌل ڳولا نتيجا پڻ ڏسو.}}",
+       "searchmenu-new": "<strong>ھن وڪيءَ تي صفحو \"[[:$1]]\" سرجيو!</strong> {{PLURAL:$2|0=|توھان جي ڳولا ۾ لڌل صفحو پڻ ڏسو.|ڳولا نتيجا پڻ ڏسو.}}",
        "searchprofile-articles": "موادي صفحا",
        "searchprofile-images": "گھڻذريعات",
        "searchprofile-everything": "ھر شَيءِ",
        "stub-threshold-disabled": "غيرفعال",
        "recentchangesdays": "تازين تبديلين ۾ ڏيکارڻ جي لاءِ ڏينهن:",
        "recentchangesdays-max": "وڌ ۾ وڌ $1 {{PLURAL:$1|ڏينهن}}",
-       "recentchangescount": "عدÙ\85 Ù¾Ù\8aرÙ\88Ù\8a Ø¬Ù\8a ØµÙ\88رت Û¾ Ú\8fÙ\8aکارڻ Ø¬Ù\8a Ù\84اءÙ\90 ØªØ±Ù\85Ù\8aÙ\85Ù\88Ù\86:",
+       "recentchangescount": "تازÙ\8aÙ\86 ØªØ¨Ø¯Ù\8aÙ\84Ù\8aÙ\86Ø\8c ØµÙ\81Ø­Ù\86 Ø¬Ù\8a Ø³Ù\88اÙ\86Ø­Ø\8c Û½ Ù\84اگس Û¾ Ú\8fÙ\8aکارڻ Ù\84اءÙ\90 Ø³Ù\86Ù\88ارÙ\86 Ø¬Ù\88 Ø®Ù\88دڪار Ú\8fÙ\86Ù\84 Ø§Ù\86Ú¯:",
        "prefs-help-recentchangescount": "وڌ ۾ وڌ انگ: 1000",
        "savedprefs": "توھان جون ترجيحون سانڍجي چڪيون آھن.",
        "savedrights": "{{GENDER:$1|$1}} جا واپرائيندڙ گروھ سانڍجي چڪا آھن.",
        "timezonelegend": "ٽائيم زون:",
        "localtime": "مقامي وقت:",
        "timezoneuseserverdefault": "وڪي عدم پيروي استعمال ڪريو ($1)",
-       "timezoneuseoffset": "ٻيو (ڄاڻايو)",
+       "timezoneuseoffset": "Ù»Ù\8aÙ\88 (Ú¾Ù\8aÙº Ú\84اڻاÙ\8aÙ\88)",
        "servertime": "سَروَر پٽاندر وقت:",
        "guesstimezone": "جھانگُوءَ مان ڀريو",
        "timezoneregion-africa": "آفريڪا",
        "userrights-groupsmember-auto": "رڪن واجبي:",
        "userrights-groupsmember-type": "$1",
        "userrights-reason": "سبب:",
-       "userrights-no-interwiki": "توهان کي ٻين وڪيز تي واپرائيندڙ حقن ۾ ترميم ڪرڻ جو حق حاصل نہ آهي.",
+       "userrights-no-interwiki": "توهان کي ٻين وڪين تي واپرائيندڙ حق سنوارڻ جي اجازت ناھي.",
        "userrights-nodatabase": "اعداخانو $1 يا تہ وجود نہ ٿو رکي يا تہ اهو مقامي اعدادخانو نہ آهي.",
        "userrights-changeable-col": "گروپَ جيڪي توهان تبديل ڪري سگھو ٿا",
        "userrights-unchangeable-col": "گروپَ جيڪي توهان تبديل نٿا ڪري سگھو",
        "right-import": "ٻين وڪيز کان صفحا درآمديو",
        "right-importupload": "ڪو فائيل چاڙهي صفحا درآمديو",
        "right-patrol": "ٻين جي ترميمن کي گشت-ڪيل طور نشان لڳايو",
-       "right-autopatrol": "سÙ\86دس ØªØ®Ù\84Ù\8aÙ\82 ÚªÙ\8aÙ\84 Ù\85ضÙ\85Ù\88Ù\86 Ù¾Ø§Ú»Ù\85رادÙ\88 Ú¯Ø´Øª ÚªÙ\8aÙ\84 Ú\84اڻاÙ\8aا Ù\88Ù\8aÙ\86دا Ø¢Ù\87Ù\86.",
+       "right-autopatrol": "سÙ\86دس Ø³Ù\86Ù\88ارÙ\88Ù\86 Ù¾Ø§Ú»Ù\85رادÙ\88 Ú¯Ø´Øª ÚªÙ\8aÙ\84 Ø·Ù\88ر Ù\86شاÙ\86 Ù\84Ú³Ù\84 Ø¢Ú¾Ù\86",
        "right-mergehistory": "صفحن جي سوانح سنواريو",
        "right-userrights": "واپرائيندڙ جا سڀ حق سنواريو",
        "right-userrights-interwiki": "ٻين وڪين تي واپرائيندڙن جا حق سنواريو",
        "grant-basic": "بنيادي حقَ",
        "grant-viewdeleted": "ڊَٺَلَ فائيلَ ۽ صفحا ڏسو",
        "grant-viewmywatchlist": "پنھنجي نظر ۾ فھرست ڏسو",
-       "newuserlogpage": "Ù\88اپرائÙ\8aÙ\86دÚ\99 Ø¬Ù\88 ØªØ®Ù\84Ù\8aÙ\82 لاگ",
+       "newuserlogpage": "Ù\88اپرائÙ\8aÙ\86دÚ\99 Ø¬Ù\88 Ø³Ø±Ø¬Ú» لاگ",
        "rightslog": "واپرائيندڙ حق لاگ",
        "action-read": "هي صفحو پڙهو",
        "action-edit": "هن صفحي کي سسنواريو",
-       "action-createpage": "Ú¾Ù\8a ØµÙ\81Ø­Ù\88 ØªØ®Ù\84Ù\8aÙ\82 ÚªØ±يو",
-       "action-createtalk": "Ú¾Ù\8a Ù\85باحثÙ\8a ØµÙ\81Ø­Ù\88 ØªØ®Ù\84Ù\8aÙ\82 ÚªØ±يو",
+       "action-createpage": "Ú¾Ù\8a ØµÙ\81Ø­Ù\88 Ø³Ø±Ø¬يو",
+       "action-createtalk": "Ú¾Ù\8a Ù\85باحثÙ\8a ØµÙ\81Ø­Ù\88 Ø³Ø±Ø¬يو",
        "action-createaccount": "ھي واپرائيندڙ کاتو سرجيو",
        "action-history": "هن صفحي جي سوانح ڏسو",
-       "action-minoredit": "Ù\87Ù\86 ØªØ±Ù\85Ù\8aÙ\85 کي معمولي طور نشان لڳايو",
+       "action-minoredit": "Ù\87Ù\86 Ø³Ù\86Ù\88ار کي معمولي طور نشان لڳايو",
        "action-move": "هيءَُ صفحو چوريو",
        "action-move-subpages": "هيءُ صفحو، ۽ ان جا ذيلي صفحا چوريو",
        "action-move-categorypages": "زمرن جا صفحا چوريو",
        "action-suppressionlog": "هيءُ ذاتي لاگ ڏسو",
        "action-block": "هن واپرائيندڙ کي سنوارڻ کان بندشيو",
        "action-protect": "هن صفحي جي تحفظاتي سطح بدلايو",
-       "action-rollback": "ڪنهن مخصوص صفحي تي آخري ترميم ڪندڙ واپرائيندڙ جي سمورين ترميمن کي ترت واپس ورايو",
+       "action-rollback": "ڪنھن مخصوص صفحي تي آخري سنوار ڪندڙ واپرائيندڙ جي سمورين سنوارن کي ترت واپس-ورايو",
        "action-import": "ٻي ڪنهن وڪي کان صفحا درآمد ڪريو",
        "action-importupload": "ڪو فائيل چاڙهي صفحا درآمديو",
        "action-patrol": "ٻين جي ترميمن کي گشت-ڪيل طور نشان لڳايو",
        "enhancedrc-history": "سوانح",
        "recentchanges": "تازيون تبديليون",
        "recentchanges-legend": "تازين تبديلين جا چارا",
-       "recentchanges-summary": "Ú¾Ù\86 ØµÙ\81Ø­Ù\8a ØªÙ\8a Ù\88ÚªÙ\8aØ¡Ù\8e Û¾ ÚªÙ\8aÙ\84 ØªØ§Ø²Ù\8aÙ\88Ù\86 ØªØ±Ù\8aÙ\86 ØªØ±Ù\85Ù\8aÙ\85ون ڏيکاريو.",
+       "recentchanges-summary": "Ú¾Ù\86 ØµÙ\81Ø­Ù\8a ØªÙ\8a Ù\88ÚªÙ\8aØ¡Ù\8e Û¾ ÚªÙ\8aÙ\84 ØªØ§Ø²Ù\8aÙ\88Ù\86 ØªØ±Ù\8aÙ\86 Ø³Ù\86Ù\88ارون ڏيکاريو.",
        "recentchanges-noresult": "ڏنل عرصي ۾ ڪي بہ تبديليون ھنن ڪسوٽين سان نٿيون ملن.",
        "recentchanges-feed-description": "ۡهن روان رسد ۾ آيل تازيون تبديليون لهو",
-       "recentchanges-label-newpage": "Ù\87Ù\86 ØªØ±Ù\85Ù\8aÙ\85 Ø³Ø§Ù\86 Ù\86ئÙ\88Ù\86 ØµÙ\81Ø­Ù\88 Ø¬Ú\99يو",
-       "recentchanges-label-minor": "Ú¾Ù\8aØ¡Ù\8e Ú¾Úª Ù\85عÙ\85Ù\88Ù\84Ù\8a ØªØ±Ù\85Ù\8aÙ\85 آھي",
-       "recentchanges-label-bot": "Ú¾Ù\8aØ¡Ù\8e ØªØ±Ù\85Ù\8aÙ\85 بوٽ عمل ۾ آندي",
-       "recentchanges-label-unpatrolled": "Ú¾Ù\86 ØªØ±Ù\85Ù\8aÙ\85 Ø¬Ù\88 Ø§Ú\83اÙ\86 گشت نہ ڪيو ويو آھي",
-       "recentchanges-label-plusminus": "ھن صفحي جي ماپ ۾ ھيترين بائيٽس جو ڦير آيو آھي",
+       "recentchanges-label-newpage": "Ù\87Ù\86 Ø³Ù\86Ù\88ار Ú¾Úª Ù\86ئÙ\88Ù\86 ØµÙ\81Ø­Ù\88 Ø³Ø±Ø¬يو",
+       "recentchanges-label-minor": "Ú¾Ù\8aØ¡Ù\8e Ú¾Úª Ù\85عÙ\85Ù\88Ù\84Ù\8a Ø³Ù\86Ù\88ار آھي",
+       "recentchanges-label-bot": "Ú¾Ù\8aØ¡Ù\8e Ø³Ù\86Ù\88ار بوٽ عمل ۾ آندي",
+       "recentchanges-label-unpatrolled": "Ú¾Ù\86 Ø³Ù\86Ù\88ار Ø¬Ù\88 Ø§Ú\83ا گشت نہ ڪيو ويو آھي",
+       "recentchanges-label-plusminus": "ھن صفحي جي ماپ ۾ ھيترين بائيٽن جو ڦير آيو",
        "recentchanges-legend-heading": "<strong>ڪنجي:</strong>",
        "recentchanges-legend-newpage": "{{int:recentchanges-label-newpage}} (پڻ ڏسو [[Special:NewPages|نون صفحن جي فھرست]])",
        "recentchanges-submit": "ڏيکاريو",
        "rcfilters-filter-user-experience-level-experienced-label": "تجربيڪار واپرائيندڙ",
        "rcfilters-filtergroup-automated": "خودڪار ڀاڱيداريون",
        "rcfilters-filter-bots-label": "بوٽ",
-       "rcfilters-filter-bots-description": "Ø®Ù\88دڪار Ø§Ù\88زارÙ\86 Ù¾Ø§Ø±Ø§Ù\86 ÚªÙ\8aÙ\84 ØªØ±Ù\85Ù\8aÙ\85ون.",
+       "rcfilters-filter-bots-description": "Ø®Ù\88دڪار Ø§Ù\88زارÙ\86 Ù¾Ø§Ø±Ø§Ù\86 ÚªÙ\8aÙ\84 Ø³Ù\86Ù\88ارون.",
        "rcfilters-filter-humans-label": "انسان (بوٽ نہ)",
-       "rcfilters-filter-humans-description": "اÙ\86ساÙ\86Ù\8a Ø§Ù\8aÚ\8aÙ\8aٽرÙ\86 Ù¾Ø§Ø±Ø§Ù\86 ÚªÙ\8aÙ\84 ØªØ±Ù\85Ù\8aÙ\85ون.",
-       "rcfilters-filter-minor-label": "Ù\85عÙ\85Ù\88Ù\84Ù\8a ØªØ±Ù\85Ù\8aÙ\85ون",
-       "rcfilters-filter-major-label": "غير معمولي ترميمون",
-       "rcfilters-filter-major-description": "Ù\85عÙ\85Ù\88Ù\84Ù\8a Ø·Ù\88ر Ù\86شاÙ\86 Ù\86Û\81 Ù\84Ú³Ù\84 ØªØ±Ù\85Ù\8aÙ\85ون.",
+       "rcfilters-filter-humans-description": "اÙ\86ساÙ\86Ù\8a Ø³Ù\86Ù\88رÙ\8aÙ\86دÚ\99Ù\86 Ù¾Ø§Ø±Ø§Ù\86 ÚªÙ\8aÙ\84 Ø³Ù\86Ù\88ارون.",
+       "rcfilters-filter-minor-label": "Ù\85عÙ\85Ù\88Ù\84Ù\8a Ø³Ù\86Ù\88ارون",
+       "rcfilters-filter-major-label": "غير-معمولي سنوارون",
+       "rcfilters-filter-major-description": "Ù\85عÙ\85Ù\88Ù\84Ù\8a Ø·Ù\88ر Ù\86شاÙ\86 Ù\86Û\81 Ù\84Ú³Ù\84 Ø³Ù\86Ù\88ارون.",
        "rcfilters-filter-watchlist-watched-label": "نظر ۾ فھڙست تي",
        "rcfilters-filter-watchlist-watched-description": "توھان جي نظر ۾ فھرست ۾ صفحن ۾ تبديليون.",
        "rcfilters-filter-watchlist-watchednew-label": "نيون نظر ۾ فھرست ۾ تبديليون",
        "rcfilters-filter-watchlist-notwatched-label": "نظر ۾ فھرست ۾ ناھي",
        "rcfilters-filtergroup-watchlistactivity": "نظر ۾ فھرست ۾ سرگرمي",
-       "rcfilters-filter-watchlistactivity-unseen-label": "اڻÚ\8fÙºÙ\84 ØªØ±Ù\85Ù\8aÙ\85ون",
-       "rcfilters-filter-watchlistactivity-seen-label": "Ú\8fÙºÙ\84 ØªØ±Ù\85Ù\8aÙ\85ون",
+       "rcfilters-filter-watchlistactivity-unseen-label": "اڻÚ\8fÙºÙ\84 Ø³Ù\86Ù\88ارون",
+       "rcfilters-filter-watchlistactivity-seen-label": "Ú\8fÙºÙ\84 Ø³Ù\86Ù\88ارون",
        "rcfilters-filtergroup-changetype": "تبديليءَ جو قِسم",
-       "rcfilters-filter-pageedits-label": "صÙ\81Ø­Ù\8a ØªØ±Ù\85Ù\8aÙ\85ون",
+       "rcfilters-filter-pageedits-label": "صÙ\81Ø­Ù\8a Ø³Ù\86Ù\88ارون",
        "rcfilters-filter-newpages-label": "صفحي تخليقون",
-       "rcfilters-filter-newpages-description": "Ù\86Ù\88اÙ\86 ØµÙ\81حا ÙºØ§Ú¾Ù\8aÙ\86دÚ\99 ØªØ±Ù\85Ù\8aÙ\85ون.",
+       "rcfilters-filter-newpages-description": "Ù\86Ù\88اÙ\86 ØµÙ\81حا ÙºØ§Ú¾Ù\8aÙ\86دÚ\99 Ø³Ù\86Ù\88ارون.",
        "rcfilters-filter-categorization-label": "زمري ۾ تبديليون",
        "rcfilters-filter-logactions-label": "لاگڊ عمل",
-       "rcfilters-view-tags": "Ù½Ù\8aÚ¯-Ù¿Ù\8aÙ\84 ØªØ±Ù\85Ù\8aÙ\85ون",
+       "rcfilters-view-tags": "Ù½Ù\8aÚ¯-Ù¿Ù\8aÙ\84 Ø³Ù\86Ù\88ارون",
        "rcfilters-liveupdates-button": "سڌي-سنئين تجديد",
        "rcnotefrom": "هيٺ {{PLURAL:$5|تبديلي آهي|تبديليون آهن}} کان <strong>$3, $4</strong> (تائين <strong>$1</strong> ) ڏيکاريل آهن.",
        "rclistfrom": "$2، $3 کان شروع ٿيندڙ نيون تبديليون ڏيکاريو",
-       "rcshowhideminor": "$1 Ù\85عÙ\85Ù\88Ù\84Ù\8a ØªØ±Ù\85Ù\8aÙ\85ون",
+       "rcshowhideminor": "$1 Ù\85عÙ\85Ù\88Ù\84Ù\8a Ø³Ù\86Ù\88ارون",
        "rcshowhideminor-show": "ڏيکاريو",
        "rcshowhideminor-hide": "لڪايو",
        "rcshowhidebots": "$1 بوٽس",
        "rcshowhideanons": "$1 نامعلوم واپرائيندڙَ",
        "rcshowhideanons-show": "ڏيکاريو",
        "rcshowhideanons-hide": "لڪايو",
-       "rcshowhidepatr": "$1 تاڻيل ترميمون",
+       "rcshowhidepatr": "$1 گشت-ڪيل سنوارون",
        "rcshowhidepatr-show": "ڏيکاريو",
        "rcshowhidepatr-hide": "لڪايو",
-       "rcshowhidemine": "Ù\85Ù\86Ú¾Ù\86جÙ\88Ù\86 ØªØ±Ù\85Ù\8aÙ\85ون $1",
+       "rcshowhidemine": "Ù\85Ù\86Ú¾Ù\86جÙ\88Ù\86 Ø³Ù\86Ù\88ارون $1",
        "rcshowhidemine-show": "ڏيکاريو",
        "rcshowhidemine-hide": "لڪايو",
        "rcshowhidecategorization": "$1 صفحاتي زمراڪاري",
        "newpageletter": "نئون",
        "boteditletter": "گ",
        "unpatrolledletter": "!",
-       "rc-change-size-new": "$1 {{PLURAL:$1|بائيٽ|بائيٽس}} تبديليءَ کانپوءِ",
+       "rc-change-size-new": "$1 {{PLURAL:$1|بائيٽ|بائيٽون}} تبديليءَ کانپوءِ",
        "newsectionsummary": "/* $1 */ نئون سيڪشن",
        "rc-enhanced-expand": "تفصيل ڏيکاريو",
        "rc-enhanced-hide": "تفصيل لڪايو",
-       "rc-old-title": "اصÙ\84 Û¾ \"$1\" Ø·Ù\88ر ØªØ®Ù\84Ù\8aÙ\82يل",
+       "rc-old-title": "اصÙ\84 Û¾ \"$1\" Ø·Ù\88ر Ø³Ø±Ø¬يل",
        "recentchangeslinked": "لاڳاپيل تبديليون",
        "recentchangeslinked-feed": "لاڳاپيل تبديليون",
        "recentchangeslinked-toolbox": "لاڳاپيل تبديليون",
        "randomredirect": "بلا ترتيب چورڻو",
        "statistics": "انگ اکر",
        "statistics-header-pages": "صفحي انگ اکر",
-       "statistics-header-edits": "ترÙ\85Ù\8aÙ\85Ù\8a Ø§Ù\86Ú¯ اکر",
+       "statistics-header-edits": "سÙ\86Ù\88ار Ø¬Ø§ Ø§Ù\86Ú¯-اکر",
        "statistics-header-users": "واپرائيندڙن جا انگ اکر",
        "statistics-header-hooks": "ٻيا انگ اکر",
        "statistics-articles": "موادي صفحا",
        "statistics-pages": "صفحا",
        "statistics-pages-desc": "وڪيءَ ۾ سڀ صفحا ٻشمول بحث صفحا، ڇوريل، وغيره.",
        "statistics-files": "چاڙھيل فائيل",
-       "statistics-edits": "{{SITENAME}} Ø¬Ù\8a ÙºÚ¾Ú» Ú©Ø§Ù\86 ØµÙ\81Ø­Ù\86 Û¾ ÚªÙ\8aÙ\84 Ø³Ù\85Ù\88رÙ\8aÙ\88Ù\86 ØªØ±Ù\85Ù\8aÙ\85ون",
-       "statistics-edits-average": "سراسرÙ\8a ØªØ±Ù\85Ù\8aÙ\85ون في صفحو",
+       "statistics-edits": "{{SITENAME}} Ø¬Ù\8a ÙºÚ¾Ú» Ú©Ø§Ù\86 ØµÙ\81Ø­Ù\86 Û¾ ÚªÙ\8aÙ\84 Ø³Ù\85Ù\88رÙ\8aÙ\88Ù\86 Ø³Ù\86Ù\88ارون",
+       "statistics-edits-average": "سراسرÙ\8a Ø³Ù\86Ù\88ارون في صفحو",
        "statistics-users": "کاتيدار واپرائيندڙَ",
        "statistics-users-active": "سرگرم واپرائيندڙ",
        "statistics-users-active-desc": "اھي واپرائيندڙ جن پوين {{PLURAL:$1|ڏينھن|$1 ڏينھن}} ۾ ڪو عمل سرانجام ڏنو آهي",
        "withoutinterwiki-legend": "اڳياڙي",
        "withoutinterwiki-submit": "ڏيکاريو",
        "fewestrevisions": "گھٽانگھٽ ترميميل صفحا",
-       "nbytes": "$1 {{PLURAL:$1|بائيٽ|بائيٽس}}",
+       "nbytes": "$1 {{PLURAL:$1|بائيٽ|بائيٽون}}",
        "ncategories": "$1 {{PLURAL:$1|زمرو|زمرا}}",
        "ninterwikis": "$1 {{PLURAL:$1|بين‌الوڪي}}",
        "nlinks": "$1 {{PLURAL:$1|ڳنڍڻو|ڳنڍڻا}}",
        "protectedtitles-submit": "عنوان ڏيکاريو",
        "listusers": "واپرائيندڙن جي فهرست",
        "listusers-editsonly": "صرف ترميمن وارا واپرائيندڙ ڏيکاريو",
-       "usereditcount": "$1 {{PLURAL:$1|ترÙ\85Ù\8aÙ\85|ترÙ\85Ù\8aÙ\85ون}}",
+       "usereditcount": "$1 {{PLURAL:$1|سÙ\86Ù\88ار|سÙ\86Ù\88ارون}}",
        "newpages": "نوان صفحا",
        "newpages-submit": "ڏيکاريو",
        "newpages-username": "واپرائيندڙ-نانءُ:",
        "watchlist-hide": "لڪايو",
        "watchlist-submit": "ڏيکاريو",
        "wlshowtime": "ڪيترو عرصو ڏيکارجي:",
-       "wlshowhideminor": "Ù\85عÙ\85Ù\88Ù\84Ù\8a ØªØ±Ù\85Ù\8aÙ\85",
+       "wlshowhideminor": "Ù\85عÙ\85Ù\88Ù\84Ù\8a Ø³Ù\86Ù\88ارÙ\88Ù\86",
        "wlshowhidebots": "بوٽس",
        "wlshowhideliu": "کاتيدار واپرائيندڙ",
        "wlshowhideanons": "گمنام واپررائيندڙ",
-       "wlshowhidepatr": "گشت-ÚªÙ\8aÙ\84 ØªØ±Ù\85Ù\8aÙ\85ون",
-       "wlshowhidemine": "Ù\85Ù\86Ú¾Ù\86جÙ\88Ù\86 ØªØ±Ù\85Ù\8aÙ\85ون",
+       "wlshowhidepatr": "گشت-ÚªÙ\8aÙ\84 Ø³Ù\86Ù\88ارون",
+       "wlshowhidemine": "Ù\85Ù\86Ú¾Ù\86جÙ\88Ù\86 Ø³Ù\86Ù\88ارون",
        "wlshowhidecategorization": "صفحاتي زمراڪاري",
        "watchlist-options": "نظر ۾ فھرست جا چارا",
        "watching": "نظر ۾ رکندي...",
        "delete-edit-reasonlist": "ڊاٺ جا سبب سنواريو",
        "rollback": "ترميمن کي واپس ورايو",
        "rollbacklink": "واپس ورايو",
-       "rollbacklinkcount": "اڻڪريو $1 {{PLURAL:$1|ترميم|ترميمون}}",
-       "revertpage": "ترميمون واپس ڪيون [[Special:Contributions/$2|$2]] ([[User talk:$2|talk]]) [[User:$1|$1]] جي آخري ترميمن ڏانهن",
+       "rollbacklinkcount": "$1 {{PLURAL:$1|سنوار|سنوارون}} واپس-ورايو",
+       "revertpage": "[[Special:Contributions/$2|$2]] ([[User talk:$2|بحث]]) پاران سنوارون واپس [[User:$1|$1]] جي آخري مسودي ڏانھن ڪيون ويون",
        "changecontentmodel-title-label": "صفحي جو عنوان",
        "changecontentmodel-reason-label": "سبب:",
        "logentry-contentmodel-change-revertlink": "واپس ورايو",
        "restriction-level": "روڪ سطح:",
        "minimum-size": "ننڍي ماپ ۾",
        "maximum-size": "وڌي ماپ ۾:",
-       "pagesize": "(بائيٽس)",
+       "pagesize": "(بائيٽون)",
        "restriction-edit": "سنواريو",
        "restriction-move": "چوريو",
        "restriction-create": "سرجيو",
        "anoncontribs": "ڀاڱيداريون",
        "contribsub2": "{{GENDER:$3|$1}} ($2) لاءِ",
        "contributions-userdoesnotexist": "واپرائيندڙ کاتو \"$1\" درج ٿيل نہ آهي.",
-       "nocontribs": "Ù\87Ù\86 Ù\85عÙ\8aار ØªÙ\8a ÚªÙ\8a Ø¨Ù\87 ØªØ±Ù\85Ù\8aÙ\85Ù\88Ù\86 Ù\86Ù\87 Ù\84Ú\8cيون.",
+       "nocontribs": "Ù\87Ù\86 Ù\85عÙ\8aار Ø³Ø§Ù\86 Ù\85Ù\84Ù\86دÚ\99 ÚªÙ\8a Ø¨Û\81 ØªØ¨Ø¯Ù\8aÙ\84Ù\8aÙ\88Ù\86 Ù\86Û\81 Ù\84Ú\8cÙ\8aÙ\88Ù\86 Ù\88يون.",
        "uctop": "هاڻوڪو",
        "month": "مھيني کان (۽ اڳوڻيون):",
        "year": "سال کان (۽ اڳوڻيون):",
        "sp-contributions-userrights": "{{GENDER:$1|واپرائيندڙ}} حقن-جي سنڀال",
        "sp-contributions-search": "ڀاڱيدارين لاءِ ڳولا ڪريو",
        "sp-contributions-username": "آءِپي پتو يا واپرائيندڙ-نانءُ:",
-       "sp-contributions-toponly": "صرÙ\81 Ø§Ú¾Ù\8a ØªØ±Ù\85Ù\8aÙ\85ون ڏيکاريو جيڪي تازا ترين مسودا آھن",
-       "sp-contributions-newonly": "صرÙ\81 Ø§Ú¾Ù\8a ØªØ±Ù\85Ù\8aÙ\85ون ڏيکاريو جيڪي صرف صفحي تخليقون آھن",
-       "sp-contributions-hideminor": "Ù\85عÙ\85Ù\88Ù\84Ù\8a ØªØ±Ù\85Ù\8aÙ\85ون لڪايو",
+       "sp-contributions-toponly": "صرÙ\81 Ø§Ú¾Ù\8a Ø³Ù\86Ù\88ارون ڏيکاريو جيڪي تازا ترين مسودا آھن",
+       "sp-contributions-newonly": "صرÙ\81 Ø§Ú¾Ù\8a Ø³Ù\86Ù\88ارون ڏيکاريو جيڪي صرف صفحي تخليقون آھن",
+       "sp-contributions-hideminor": "Ù\85عÙ\85Ù\88Ù\84Ù\8a Ø³Ù\86Ù\88ارون لڪايو",
        "sp-contributions-submit": "ڳوليو",
        "whatlinkshere": "هتان ڇا ڳنڍيل آهي",
        "whatlinkshere-title": "\"$1\" سان ڳنڍيندڙ صفحا",
        "unlockbtn": "اعدادخاني کي کوليو",
        "move-page": "$1 چوريو",
        "move-page-legend": "صفحو چوريو",
-       "movepagetext": "هيٺيون فارم استعمال ڪندي ڪنهن صفحي کي نئون عنوان ڏئي سگھجي ٿو، جنهن سان سمورو صفحو نئين عنوان ڏانهن هليو ويندو. \nاڳوڻو عنوان نئين عنوان ڏانهن چورڻو بنجي ويندو. \nتوهان  چورڻن کي سنواري سگھو ٿا جيڪي اصل عنوان ڏانهن خودبخود اشارو ڪن ٿا.\nانهي ڳالهه جي پڪ ڪري وٺو ته [[Special:BrokenRedirects|ٽٽل چورڻا]] يا [[Special:DoubleRedirects|ٻٽا چورڻا]] نه هجن.\nان ڳالهه جي پڪ ڪرڻ ذميواري توهان تي آهي ته ڳنڍڻا اتي ئي وٺي وڃن ٿا جتي انهن کي وٺي وڃڻ گھرجي.\n\nياد رکندا ته جيڪڏهن نئين عنوان سان اڳي ئي ڪو مضمون موجود آهي ته پوءِ صفحو '''نه''' چوريو ويندو، سوا ان جي ته موجوده صفحو محظ خالي آهي يا ڪا به سوانح نه رکندڙ ڪو چورڻو آهي.\n\n<strong>نوٽ!</strong>\nاها هڪ مقبول صفحي لاءِ ڪا غير متوقه ۽ انتهائي اڻوڻندڙ تبديلي ثابت ٿي سگھي ٿي؛ براءِ مهرباني اڳتي وڌڻ کان اڳ پڪ ڪندا ته توهان اها تبديلي آڻڻ جي نتيجن کان چڱيءَ ريت واقف آهيو.",
-       "movepagetext-noredirectfixer": "هيٺيون فارم استعمال ڪندي ڪنهن صفحي کي نئون عنوان ڏئي سگھجي ٿو، جنهن سان سمورو صفحو نئين عنوان ڏانهن هليو ويندو. \nاڳوڻو عنوان نئين عنوان ڏانهن چورڻو بنجي ويندو. \nتوهان  چورڻن کي سنواري سگھو ٿا جيڪي اصل عنوان ڏانهن خودبخود اشارو ڪن ٿا.\nانهي ڳالهه جي پڪ ڪري وٺو ته [[Special:BrokenRedirects|ٽٽل چورڻا]] يا [[Special:DoubleRedirects|ٻٽا چورڻا]] نه هجن.\nان ڳالهه جي پڪ ڪرڻ ذميواري توهان تي آهي ته ڳنڍڻا اتي ئي وٺي وڃن ٿا جتي انهن کي وٺي وڃڻ گھرجي.\n\nياد رکندا ته جيڪڏهن نئين عنوان سان اڳي ئي ڪو مضمون موجود آهي ته پوءِ صفحو '''نه''' چوريو ويندو، سوا ان جي ته موجوده صفحو محظ خالي آهي يا ڪا به سوانح نه رکندڙ ڪو چورڻو آهي.\n\n<strong>نوٽ!</strong>\nاها هڪ مقبول صفحي لاءِ ڪا غير متوقه ۽ انتهائي اڻوڻندڙ تبديلي ثابت ٿي سگھي ٿي؛ مهرباني ڪري اڳتي وڌڻ کان اڳ پڪ ڪندا ته توهان اها تبديلي آڻڻ جي نتيجن کان چڱيءَ ريت واقف آهيو.",
+       "movepagetext": "هيٺيون فارم استعمال ڪندي ڪنھن صفحي کي نئون عنوان ڏئي سگھجي ٿو، جنھن سان سمورو صفحو نئين عنوان ڏانھن هليو ويندو. \nاڳوڻو عنوان نئين عنوان ڏانھن چورڻو بڻجي ويندو. \nتوهان  چورڻن کي سنواري سگھو ٿا جيڪي اصل عنوان ڏانهن خودبخود اشارو ڪن ٿا.\nانهي ڳالھ جي پڪ ڪري وٺو تہ [[Special:BrokenRedirects|ٽٽل چورڻا]] يا [[Special:DoubleRedirects|ٻٽا چورڻا]] نہ هجن.\nان ڳالھ جي پڪ ڪرڻ ذميواري توهان تي آهي تہ ڳنڍڻا اتي ئي وٺي وڃن ٿا جتي انھن کي وٺي وڃڻ گھرجي.\n\nياد رکندا تہ جيڪڏهن نئين عنوان سان اڳي ئي ڪو مضمون موجود آهي ته پوءِ صفحو '''نہ''' چوريو ويندو، سواءِ ان جي تہ موجوده صفحو محظ خالي آهي يا ڪا بہ سوانح نہ رکندڙ ڪو چورڻو آهي.\n\n<strong>نوٽ!</strong>\nاها هڪ مقبول صفحي لاءِ ڪا غير متوقع ۽ انتھائي اڻوڻندڙ تبديلي ثابت ٿي سگھي ٿي؛ براءِ مھرباني اڳتي وڌڻ کان اڳ پڪ ڪندا تہ توهان اها تبديلي آڻڻ جي نتيجن کان چڱيءَ ريت واقف آهيو.",
+       "movepagetext-noredirectfixer": "هيٺيون فارم استعمال ڪندي ڪنھن صفحي کي نئون عنوان ڏئي سگھجي ٿو، جنھن سان سمورو صفحو نئين عنوان ڏانھن هليو ويندو. \nاڳوڻو عنوان نئين عنوان ڏانھن چورڻو بڻجي ويندو. \nتوهان  چورڻن کي سنواري سگھو ٿا جيڪي اصل عنوان ڏانھن خودبخود اشارو ڪن ٿا.\nانهي ڳالھ جي پڪ ڪري وٺو تہ [[Special:BrokenRedirects|ٽٽل چورڻا]] يا [[Special:DoubleRedirects|ٻٽا چورڻا]] نہ هجن.\nان ڳالھ جي پڪ ڪرڻ ذميواري توهان تي آهي تہ ڳنڍڻا اتي ئي وٺي وڃن ٿا جتي انھن کي وٺي وڃڻ گھرجي.\n\nياد رکندا تہ جيڪڏهن نئين عنوان سان اڳي ئي ڪو مضمون موجود آهي ته پوءِ صفحو '''نہ''' چوريو ويندو، سوا ان جي تہ موجوده صفحو محظ خالي آهي يا ڪا بہ سوانح نہ رکندڙ ڪو چورڻو آهي.\n\n<strong>نوٽ!</strong>\nاها هڪ مقبول صفحي لاءِ ڪا غير متوقع ۽ انتھائي اڻوڻندڙ تبديلي ثابت ٿي سگھي ٿي؛ مھرباني ڪري اڳتي وڌڻ کان اڳ پڪ ڪندا تہ توهان اها تبديلي آڻڻ جي نتيجن کان چڱيءَ ريت واقف آهيو.",
        "movepagetalktext": "جيڪڏهن توهان هن خاني کي نشان لڳائيندئو، واسطيدار مباحثي صفحو پاڻ ئي چوريو ويندو ماسواءِ اتي ڪو اڳ ئي ڪو غيرخالي مباحثي صفحو موجود هجي.\n\nان صورت ۾، جيڪڏهن توهان چاهيو ته صفحي کي پاڻ چوري يا ضم ڪري سگھو ٿا.",
        "movecategorypage-warning": "<strong>چتاءُ:</strong> اوهان زمري واري صفحي کي چورڻ وڃي رهيا آهيو. ياد رکو صرف صفحو چورندو، جيڪڏهن ڪي به صفحا پراڻي زمري ۾ شامل آهن، انهن جي نئين زمري ۾ درجابندي <em>نه</em> ٿيندي.",
        "movenotallowed": "توهان کي صفحا چورڻ جي اجازت حاصل ڪانهي.",
        "allmessagescurrent": "موجوده تحرير",
        "allmessages-filter-legend": "ڇاڻي",
        "allmessages-filter-all": "سڀ",
-       "allmessages-filter-modified": "ترÙ\85Ù\8aÙ\85 Ø´Ø¯Ù\87",
+       "allmessages-filter-modified": "بدÙ\84اÙ\8aÙ\84",
        "allmessages-language": "ٻولي:",
        "allmessages-filter-submit": "ھلو",
        "allmessages-filter-translate": "ترجمو ڪريو",
        "tooltip-ca-nstab-main": "مواد جي صفحي کي ڏسو",
        "tooltip-ca-nstab-user": "واپرائيندڙ جو صفحو ڏسو",
        "tooltip-ca-nstab-media": "ميڊيا جو صفحو ڏسو",
-       "tooltip-ca-nstab-special": "هيءُ خاص صفحو آهي، ان ۾ ترميم نٿي ڪري سگھجي",
+       "tooltip-ca-nstab-special": "هي ھڪ خاص صفحو آهي، ان کي سنوار نٿو سگھجي",
        "tooltip-ca-nstab-project": "رٿائي صفحو ڏسو",
        "tooltip-ca-nstab-image": "ھن فائيل جو صفحو ڏسو",
        "tooltip-ca-nstab-mediawiki": "سرشتي جو پيغام ڏيکاريو",
        "tooltip-ca-nstab-template": "سانچو ڏسو",
        "tooltip-ca-nstab-help": "امدادي صفحو ڏسو",
        "tooltip-ca-nstab-category": "هن زمري جو صفحو ڏسو",
-       "tooltip-minoredit": "ان کي هڪ معمولي ترميم ڄاڻايو",
+       "tooltip-minoredit": "ھن کي هڪ معمولي سنوار طور نشان لڳايو",
        "tooltip-save": "پنھنجون تبديليون سانڍيو",
        "tooltip-publish": "پنهنجيون تبديليون ڇاپيو",
        "tooltip-preview": "پنھنجي تبديلين تي نگاھ وجھو. براءِ مھرباني اھو سانڍڻ کان اڳ ڪندا.",
        "tooltip-watchlistedit-normal-submit": "فائيل ھٽايو",
        "tooltip-watchlistedit-raw-submit": "واچ لسٽ کي اَپڊيٽ ڪيو",
        "tooltip-upload": "چاڙهڻ شروع ڪيو",
-       "tooltip-rollback": "\"واپس ورايو\" ھن صفحي ۾ پوئين ڀاڱيدار جي ڪيل ترميم(ن) کي ھڪ ٽڙڪ سان اڻڪري ٿو",
-       "tooltip-undo": "\"اڻڪرÙ\8aÙ\88\" Ù\87Ù\86 ØªØ±Ù\85Ù\8aÙ\85 Ú©Ù\8a Ù\88اپس ÚªÙ\8aÙ\88 Û½ Ù¾Ù\8aØ´ Ù\86گاÙ\87 Û¾ ØªØ±Ù\85Ù\8aÙ\85 Ú©Ù\8a Ú©Ù\88Ù\84Ù\8aÙ\88. Ø§Ù\87Ù\88 Ø§Ù\88Ù\87اÙ\86 Ú©Ù\8a ØªØª Ø´Ø§Ù\85Ù\84 ÚªØ±Ú» Ø¬Ù\88 Ù\85Ù\88Ù\82عÙ\88 Ú\8fÙ\8aÙ\86دو.",
+       "tooltip-rollback": "\"واپس-ورايو\" ھن صفحي ۾ آخري ڀاڱيدار جي سنوارن(ن) کي ھڪ ٽڙڪ سان اڻڪري ٿو",
+       "tooltip-undo": "\"اڻڪرÙ\8aÙ\88\" Ù\87Ù\86 Ø³Ù\86Ù\88ار Ú©Ù\8a Ù\88اپس ÚªØ±Ù\8a Ù¿Ù\88 Û½ Ù¾Ù\8aØ´-Ù\86گاھ Û¾ Ø³Ù\86Ù\88ار Ú©Ù\8a Ú©Ù\88Ù\84Ù\8a Ù¿Ù\88. Ø§Ù\87Ù\88 ØªÙ\8eتÙ\8e Û¾ ÚªÙ\88 Ø³Ø¨Ø¨ Ù\88جھڻ Ø¬Ù\8a Ø§Ø¬Ø§Ø²Øª Ú\8fÙ\8a Ù¿و.",
        "tooltip-preferences-save": "ترجيحون سانڍيو",
        "tooltip-summary": "ننڍو تَتُ ڏيو",
        "anonymous": "{{SITENAME}} جا گمنام {{PLURAL:$1|واپرائيندڙ|واپرائيندڙَ}}",
        "simpleantispam-label": "اينٽي-اسپام روڪ.\nھن کي <strong>نہ</strong> ڀريو!",
        "pageinfo-title": "\"$1\" لاءِ معلومات",
        "pageinfo-header-basic": "بنيادي معلومات",
-       "pageinfo-header-edits": "ترÙ\85Ù\8aÙ\85Ù\8a سوانح",
+       "pageinfo-header-edits": "سÙ\86Ù\88ار سوانح",
        "pageinfo-header-restrictions": "صفحي جو تحفظ",
        "pageinfo-header-properties": "صفحي جون خصوصيتون",
        "pageinfo-display-title": "ڏيکارجندڙ عنوان",
-       "pageinfo-length": "(بائيٽس ۾) صفحي جي ڊيگھ",
+       "pageinfo-length": "(بائيٽن ۾) صفحي جي ڊيگھ",
        "pageinfo-article-id": "صفحي جي آءِڊي",
        "pageinfo-language": "صفحي جي مواد جي ٻولي",
        "pageinfo-content-model": "صفحي جي مواد جو ماڊل",
        "pageinfo-redirects-name": "ھن صفحي ڏانھن ڇوريل صفحن جو انگ",
        "pageinfo-subpages-name": "ھن صفحي جي ذيلي صفحن جو انگ",
        "pageinfo-firstuser": "صفحو تخليقڪار",
-       "pageinfo-firsttime": "صÙ\81Ø­Ù\8a ØªØ®Ù\84Ù\8aÙ\82 ÚªØ±ڻ جي تاريخ",
+       "pageinfo-firsttime": "صÙ\81Ø­Ù\8a Ø³Ø±Ø¬ڻ جي تاريخ",
        "pageinfo-lastuser": "تازو ترين سنواريندڙ",
-       "pageinfo-lasttime": "تازÙ\8a ØªØ±Ù\8aÙ\86 ØªØ±Ù\85Ù\8aÙ\85 جي تاريخ",
+       "pageinfo-lasttime": "تازÙ\8a ØªØ±Ù\8aÙ\86 Ø³Ù\86Ù\88ار جي تاريخ",
        "pageinfo-edits": "سڀني ترميمن جو انگ",
        "pageinfo-authors": "چٽن ليکڪن جو مڪمل انگ",
        "pageinfo-recent-edits": "(گذريل $1 ۾) تازين ترميمن جو انگ",
        "markedaspatrollednotify": "$1 کي گشت ڪيل طور ڄاڻيو ويو آهي.",
        "patrol-log-page": "گشت لاگ",
        "confirm-markpatrolled-button": "ٺيڪ (او ڪي) آهي",
-       "previousdiff": "â\86\92 Ø§Ú³Ù\88Ú»Ù\8a ØªØ±Ù\85Ù\8aÙ\85",
-       "nextdiff": "Ù\86ئÙ\8aÙ\86 ØªØ± ØªØ±Ù\85Ù\8aÙ\85 ←",
+       "previousdiff": "â\86\92 Ø§Ú³Ù\88Ú»Ù\8a Ø³Ù\86Ù\88ار",
+       "nextdiff": "Ù\86ئÙ\8aÙ\86 ØªØ± Ø³Ù\86Ù\88ار ←",
        "widthheightpage": "$1 × $2, $3 {{PLURAL:$3|صفحو|صفحا}}",
        "file-info-size": "$1 × $2 عڪسلون، فائيل سائيز: $3، MIME ٽائيپ: $4",
        "file-nohires": "اڃان سنھو تحلل ميسر ناھي.",
        "sunday-at": "$1 بجي آچر",
        "yesterday-at": "$1 بجي ڪالهہ",
        "metadata": "اعدادِ اعداد",
-       "metadata-help": "هن فائل ۾ وڌيڪ ڄاڻ موجود آهي، جيڪو ڪنهن ڊجيٽل ڪيمرا يا اسڪينر متعلق آهي،  جنهن سان هي عڪس چٽيو ويو آهي. \nجيڪڏهن فائل ترميم ڪيل هجي ته پوءِ ممڪن آهي ته ڪجه ڄاڻ ضايع ٿي چڪو هجي.",
+       "metadata-help": "هن فائل ۾ وڌيڪ ڄاڻ موجود آهي، جيڪا ڪنھن ڊجيٽل ڪيمرا يا اسڪينر کان وڌي وئي آهي، جنھن سان هي عڪس سرجيو يا چٽيو ويو آهي. \nجيڪڏهن فائل بدلايو ويو آھي تہ پوءِ ڪجھ ڄاڻ تفصيل سڄي تبديل ٿيل فائيل کي متان ظاھر نہ ڪري.",
        "metadata-expand": "توسيعي تفصيل ڏيکاريو",
        "metadata-collapse": "توسيعي تفصيل لڪايو",
        "metadata-fields": "عڪس جي ميٽا ڊيٽا فيلڊ لسٽ ڪيل آهي هن پيغام ۾ جيڪو شامل ڪيو ويندو عڪس جي صفحي جي ڊسپلي تي، جڏهن ميٽا ڊيٽا جي ٽيبل ختم ٿيندي، ٻيا طئي ٿيل طريقي سان لڪل هوندا. \n* make\n* model\n* datetimeoriginal\n* exposuretime\n* fnumber\n* isospeedratings\n* focallength\n* artist\n* copyright\n* imagedescription\n* gpslatitude\n* gpslongitude\n* gpsaltitude",
        "confirmemail_success": "توهان جي برقٽپال پتي جي تصديق ڪئي وئي آهي.\nهاڻ توهان [[Special:UserLogin|داخل ٿي]] ۽ وڪي جو مزو وٺي سگھو ٿا.",
        "confirmemail_loggedin": "توهان جي برقٽپال پتي جي تصديق هاڻي ٿي چڪي آهي.",
        "confirmemail_subject": "{{SITENAME}} برقٽپال پتي جي تصديق",
-       "confirmemail_body": "ڪنهن، شايد اوهان جي، هن آئي پي پتي $1 کان {{SITENAME}} ۾ «$2» جي نالي سان کاتو کوليو ۽ هن برق ٽپال کي استعمال ڪيو آهي.\n\nهن ڳالهه جي تصديق ڪري وٺو ته هي کاتو اوهان جو ئي آهي ۽ {{SITENAME}} تي پڻ هي برق ٽپال ڏنل آهي، ته پوءِ هيٺين ڳنڍڻي تي ڪلڪ ڪيو. يا ڪاپي ڪري برائوزر تي کوليو.\n\n$3\n\nجيڪڏهن اوهان کاتي ۾ برق ٽپال رجسٽر *نه* ڪيو آهي ته هيٺين ڳنڍڻي کي ڪلڪ ڪيو يا برائوزر ۾ کوليو:\n\n$5\n\nهن ڪوڊ جي تصديقي جي مدت $4 تي ختم ٿي ويندي.",
+       "confirmemail_body": "ڪنهن، شايد اوهان جي، هن آئي پي پتي $1 کان {{SITENAME}} ۾ «$2» جي نالي سان کاتو کوليو ۽ هن برق ٽپال کي استعمال ڪيو آهي.\n\nهن ڳالھ جي تصديق ڪري وٺو تہ هي کاتو اوهان جو ئي آهي ۽ {{SITENAME}} تي پڻ هي برق ٽپال ڏنل آهي، تہ پوءِ هيٺين ڳنڍڻي تي ڪلڪ ڪيو. يا ڪاپي ڪري برائوزر تي کوليو.\n\n$3\n\nجيڪڏهن اوهان کاتي ۾ برق ٽپال رجسٽر *نہ* ڪيو آهي تہ هيٺين ڳنڍڻي کي ڪلڪ ڪيو يا برائوزر ۾ کوليو:\n\n$5\n\nهن ڪوڊ جي تصديقي جي مدت $4 تي ختم ٿي ويندي.",
        "recreate": "ورسرجيو",
        "confirm-watch-button": "ٺيڪ",
        "confirm-watch-top": "هيءُ صفحو پنهنجي نظر ۾ فھرست ۾ شامل ڪندا؟",
        "confirm-unwatch-button": "ٺيڪ",
        "confirm-unwatch-top": "هيءُ صفحو پنهنجي نظر ۾ فهرست مان هٽائيندا؟",
-       "confirm-rollback-top": "Ú¾Ù\86 ØµÙ\81Ø­Ù\8a Û¾ ÚªÙ\8aÙ\84 ØªØ±Ù\85Ù\8aÙ\85ون واپس ورايون؟",
+       "confirm-rollback-top": "Ú¾Ù\86 ØµÙ\81Ø­Ù\8a Û¾ ÚªÙ\8aÙ\84 Ø³Ù\86Ù\88ارون واپس ورايون؟",
        "quotation-marks": "\"$1\"",
        "imgmultipageprev": "← اڳوڻو صفحو",
        "imgmultipagenext": "ايندڙ صفحو →",
        "table_pager_limit_submit": "ھلو",
        "table_pager_empty": "ڪو بہ نتيجو نہ مليو",
        "autoredircomment": "صفحي کي [[$1]] ڏانھن چوريو",
-       "autosumm-newblank": "خاÙ\84Ù\8a ØµÙ\81Ø­Ù\88 ØªØ®Ù\84Ù\8aÙ\82 Úªيو",
+       "autosumm-newblank": "خاÙ\84Ù\8a ØµÙ\81Ø­Ù\88 Ø³Ø±Ø¬Ù\8aÙ\88 Ù\88يو",
        "watchlistedit-normal-title": "نظر ۾ فھرست کي سنواريو",
        "watchlistedit-raw-titles": "عنوانَ:",
        "watchlistedit-clear-titles": "عنوانَ:",
        "tag-list-wrapper": "[[Special:Tags|{{PLURAL:$1|ٽيگ|ٽيگز}}]]: $2",
        "tag-mw-new-redirect": "نئون چوريل",
        "tag-mw-blank": "خالي",
-       "tag-mw-rollback-description": "Ù\88اپس-Ù\88راÙ\8aÙ\88 Ú³Ù\86Ú\8dÚ»Ù\8a Ú©Ù\8a Ø§Ø³ØªØ¹Ù\85اÙ\84 ÚªÙ\86دÙ\8a Ù¾Ù\88Ù\8aÙ\86 ØªØ±Ù\85Ù\8aÙ\85Ù\86 Ú©Ù\8a Ù\88اپس Ù\88رائÙ\8aÙ\86دÚ\99 ØªØ±Ù\85Ù\8aÙ\85ون",
+       "tag-mw-rollback-description": "Ù\88اپس-Ù\88راÙ\8aÙ\88 Ú³Ù\86Ú\8dÚ»Ù\8a Ú©Ù\8a Ø§Ø³ØªØ¹Ù\85اÙ\84 ÚªÙ\86دÙ\8a Ù¾Ù\88Ù\8aÙ\86 Ø³Ù\86Ù\88ارÙ\86 Ú©Ù\8a Ù\88اپس Ù\88رائÙ\8aÙ\86دÚ\99 Ø³Ù\86Ù\88ارون",
        "tags-title": "ٽيگس",
        "tags-tag": "ٽيگ نانءُ",
        "tags-source-header": "ذريعو",
        "tags-active-header": "سرگرم؟",
-       "tags-hitcount-header": "ٽيگ ٿيل ترميمون",
+       "tags-hitcount-header": "ٽيگ-ٿيل تبديليون",
        "tags-active-yes": "ها",
        "tags-active-no": "نہ",
        "tags-edit": "سنواريو",
        "logentry-move-move_redir": "$1 $3 کي $4 ڏانھن ريڊائريڪٽ مٿان {{GENDER:$2|چوريو}}",
        "logentry-patrol-patrol-auto": "$1 پاڻمرادو صفحي $3 جي $4 مسودي تي گشت ڪيل طور {{GENDER:$2|نشان لڳايو}}",
        "logentry-newusers-create": "واپرائيندڙ کاتو $1 {{GENDER:$2|سرجيو ويو}}",
-       "logentry-newusers-autocreate": "واپرائيندڙ کاتو $1 پاڻمرادو {{GENDER:$2|تخليق ڪيو}} ويو",
+       "logentry-newusers-autocreate": "واپرائيندڙ کاتو $1 پاڻمرادو {{GENDER:$2|کوليو ويو}}",
        "logentry-protect-protect": "$1 {{GENDER:$2|محفوظ ڪيو}} $3 $4",
        "logentry-upload-upload": "$1 {{GENDER:$2|چاڙهيو}} $3",
        "logentry-upload-overwrite": "$1 $3 جو ھڪ نئون ورزن {{GENDER:$2|چاڙھيو}}",
        "special-characters-group-devanagari": "ديوناگري",
        "special-characters-group-thai": "ٿائي",
        "special-characters-group-lao": "لائو",
-       "mw-widgets-abandonedit-discard": "ترÙ\85Ù\8aÙ\85Ù\88Ù\86 Ø®ØªÙ\85 ڪريو",
+       "mw-widgets-abandonedit-discard": "سÙ\86Ù\88ارÙ\88Ù\86 Ú¦Ù½Ù\88 ڪريو",
        "mw-widgets-abandonedit-keep": "سنوارڻ جاري رکو",
        "mw-widgets-abandonedit-title": "ڇا توهان کي پڪ آهي؟",
        "mw-widgets-dateinput-no-date": "ڪا بہ تاريخ نہ چونڊيل",
index edc79a2..c834ecc 100644 (file)
        "toc": "Indizi",
        "showtoc": "musthra",
        "hidetoc": "cua",
+       "confirmable-yes": "Emmu",
+       "confirmable-no": "No",
        "thisisdeleted": "Vedi o turra che primma $1?",
        "viewdeleted": "Vedi $1?",
        "restorelink": "{{PLURAL:$1|una mudìfigga canzilladda|$1 mudìfigghi canzilladdi}}",
        "nospecialpagetext": "La pàgina ippiziari dumandadda no è isthadda ricuniscidda da lu software MediaWiki; l'erencu di li pàgini ippiziari vàriddi s'acciappa i'li [[Special:SpecialPages|{{int:specialpages}}]].",
        "error": "Errori",
        "databaseerror": "Errori di la bancadati",
+       "databaseerror-error": "Errori: $1",
        "laggedslavemode": "Attinzioni: la pàgina pudia no cuntinì l'aggiornamenti più rizzenti.",
        "readonly": "Bancadati broccadda",
        "enterlockreason": "Indica lu mutibu di lu broccu, ippizzifichendi candu po’ assé chi venghia buggaddu.",
        "namespaceprotected": "Nò si diponi di li pimmissi nezzessàri pa mudifiggà li pagini di lu tipu di pagina '''$1'''.",
        "ns-specialprotected": "No è pussìbiri mudifiggà li pàgini ippiziari.",
        "titleprotected": "Chisthu tìturu è isthaddu prutiggiddu da la criazioni da [[User:$1|$1]].\nLa rasgioni frunidda è <em>$2</em>.",
+       "exception-nologin": "Intradda no effettuadda",
        "logouttext": "'''Iscidda effettuadda.'''\n\nSi pò sighì a usà {{SITENAME}} cumenti utenti anònimu oppuru eseguì una noba intradda, cu' lu matessi innòmu utenti o un'innòmu dibessu.\nZerthuni pàgini pudìani continuà a apparì cumenti si la iscidda nò fùssia avvinudda finaghì nò vèni puridda la mimória cache di lu propriu nabiggadori.",
+       "welcomeuser": "Binvinuddi, $1!",
        "yourname": "Innòmu utenti",
        "userlogin-yourname": "Innòmu utenti",
        "userlogin-yourname-ph": "Ischribi l'innommu d'utenti tóiu.",
        "createacct-yourpasswordagain": "Cunfeimmà paràura d'órdini",
        "createacct-yourpasswordagain-ph": "Turrà a ischribì lu còditzi",
        "userlogin-remembermypassword": "Sighiddi a lassammi intraddu/a.",
+       "cannotlogin-title": "L'intradda non è pussìbiri.",
+       "cannotlogin-text": "L'intradda non è pussìbiri.",
+       "cannotloginnow-title": "L'intradda non è pussìbiri abali.",
+       "cannotloginnow-text": "Impriendi $1 l'intradda non è pussìbiri.",
+       "cannotcreateaccount-title": "Impussìbiri crià un profiru.",
        "yourdomainname": "Ippizzificà lu dumìniu",
        "externaldberror": "S'è verifiggaddu un errori cu lu server di autentificazioni esthernu, oppuru nò si diponi di l'autorizazioni nezzessàri pa aggiornà la propria registhrazioni estherna.",
        "login": "Intra",
+       "login-security": "Verifigga l'identiddai toia.",
        "nav-login-createaccount": "Intra / registhrazioni",
        "logout": "Esci",
        "userlogout": "Esci",
        "createaccount": "Crea una noba registhrazioni",
        "userlogin-resetpassword-link": "Hai immintigaddu lu còdizi tóiu?",
        "userlogin-helplink2": "Aggiuddu cun l'intradda.",
+       "userlogin-createanother": "Crea un'althra registhrazioni",
+       "createacct-emailrequired": "Indirizzu di postha erettrònica",
        "createacct-emailoptional": "Indirizzu di postha erettrònica (opzionari)",
        "createacct-email-ph": "Ischribi lu tóiu indirizzu di postha erettrònica",
+       "createacct-another-email-ph": "Ischribi lu tóiu indirizzu di postha erettrònica.",
        "createaccountmail": "via postha erettrònica",
        "createacct-reason": "Mutibu",
        "createacct-submit": "Registhrazioni",
+       "createacct-another-submit": "Crea una noba registhrazioni",
+       "createacct-continue-submit": "Cuntinuà la criazioni di la registhrazioni",
+       "createacct-another-continue-submit": "Cuntinuà la criazioni di la registhrazioni",
        "createacct-benefit-heading": "{{SITENAME}} è criaddu pa jenti cumenti sei tu.",
        "createacct-benefit-body1": "{{PLURAL:$1|Mudìfiggu|Mudìfigghi}}",
        "createacct-benefit-body2": "{{PLURAL:$1|pàgina|pàgini}}",
        "badretype": "Li paràuri d'órdhini insiriddi nò cuinzidhini tra èddi.",
        "userexists": "L'innòmu utenti insiriddu è già utirizaddu. Pa pazieri chirria un'innòmu utenti dibessu.",
        "loginerror": "Errori i' l'intradda",
+       "createacct-error": "Errori cu' la criazioni di lu profiru",
        "nocookiesnew": "La registhrazioni di l'utenti pa l'intradda è isthadda criadda, ma nò è isthaddu pussìbiri intrà a {{SITENAME}} parchí li cookie so disattibaddi. Riprubà l'intradda cu l'innòmu utenti e la paràura d'órdhini appèna criaddi daboi abé attibaddu li cookie i lu propriu nabiggadori.",
        "nocookieslogin": "L'intradda a {{SITENAME}} dumanda l'usu di li cookie, chi risulthani disattibaddi. Riprubà l'intradda daboi abé attibaddu li cookie i' lu propriu nabiggadori.",
        "noname": "L'innòmu utenti indicaddu nò è vàriddu.",
        "loginlanguagelabel": "Linga: $1",
        "pt-login": "Intra",
        "pt-login-button": "Intra",
+       "pt-login-continue-button": "Cuntinuà l'intradda",
        "pt-createaccount": "Crea una noba registhrazioni",
        "pt-userlogout": "Iscidda",
        "changepassword": "Ciamba paràura d'órdhini",
        "retypenew": "Turra a ischribì la noba paràura d'órdhini:",
        "resetpass_submit": "Impustha la paràura d'órdhini e intra",
        "changepassword-success": "La paràura d'órdhini tóia è isthadda mudìfiggadda. Abà sei intrendi...",
+       "botpasswords": "Còdizi di bot",
+       "botpasswords-label-appid": "Nommu di lu bot:",
+       "botpasswords-label-create": "Crià",
+       "botpasswords-label-update": "Attuarizà",
+       "botpasswords-label-cancel": "Annullà",
+       "botpasswords-label-delete": "Canzillà",
+       "botpasswords-label-resetpassword": "Invià una noba paràura d'órdhini pa postha erettrònica",
        "resetpass_forbidden": "No è pussìbiri mudifiggà li paràuri d'órdhini in {{SITENAME}}.",
+       "resetpass-submit-loggedin": "Ciambà paràura d'órdhini",
+       "resetpass-submit-cancel": "Cancillà",
        "passwordreset": "Invia una noba paràura d'órdhini pa postha erettrònica",
        "passwordreset-username": "Innòmu utenti:",
+       "passwordreset-email": "Indirizzu di postha erettrònica:",
+       "passwordreset-emailtitle": "Dettàglii di lu profiru sobbra {{SITENAME}}",
+       "changeemail-oldemail": "Indirizzu di postha erettrònica d'abà:",
+       "changeemail-newemail": "Nobu indirizzu di postha erettrònica:",
        "changeemail-none": "(nisciunu)",
        "resettokens-tokens": "Token:",
        "bold_sample": "Grassetu",
        "sig_tip": "Fimma cun data e ora",
        "hr_tip": "Lìnia orizontari (usà cun moderazioni)",
        "summary": "Oggettu:",
-       "subject": "Tìturu:",
+       "subject": "Oggettu:",
        "minoredit": "Chistha è una mudìfigga minori",
        "watchthis": "Aggiungi a l'abbaidaddi ippiziari",
        "savearticle": "Saivvà la pagina",
+       "savechanges": "Saivvà li ciambamenti",
+       "publishpage": "Prubbiggà la pàgina",
+       "publishchanges": "Prubbiggà li mudìfigghi",
+       "savearticle-start": "Saivvà la pàgina ...",
+       "savechanges-start": "Saivvà li ciambamenti ...",
+       "publishpage-start": "Prubbiggà la pàgina ...",
+       "publishchanges-start": "Prubbiggà li mudìfigghi ...",
        "preview": "Antiprimma",
        "showpreview": "Visuarizza antiprimma",
        "showdiff": "Musthra ciambamenti",
index f951e8a..06a683a 100644 (file)
        "defaultmessagetext": "Uobičajeni tekst poruke",
        "content-failed-to-parse": "Ne mogu da raščlanim sadržaj tipa $2 za model $1: $3",
        "invalid-content-data": "Neispravni podaci sadržaja",
-       "content-not-allowed-here": "Sadržaj modela „$1“ nije dozvoljen na stranici [[:$2]]",
+       "content-not-allowed-here": "Sadržaj modela „$1“ nije dozvoljen na stranici [[:$2]] u slotu „$3”",
        "editwarning-warning": "Ako napustite ovu stranicu, izgubićete sve izmene koje ste napravili.\nAko ste prijavljeni, možete onemogućiti ovo upozorenje u svojim podešavanjima, u odeljku \"{{int:prefs-editing}}\".",
        "editpage-invalidcontentmodel-title": "Sadržajni model nije podržan",
        "editpage-invalidcontentmodel-text": "Sadržajni model \"$1\" nije podržan.",
        "converter-manual-rule-error": "Pronađena je greška u pravilu za ručno pretvaranje jezika",
        "undo-success": "Izmjena se može vratiti.\nMolimo da provjerite usporedbu ispod da budete sigurni da to želite učiniti, a zatim spremite promjene da bi ste završili vraćanje izmjene.",
        "undo-failure": "Izmjene se ne mogu vratiti zbog konflikta sa izmjenama u međuvremenu.",
+       "undo-main-slot-only": "Izmjena ne može se opozvati jer ima sadržaj izvan glavnoga slota.",
        "undo-norev": "Izmjena se ne može vratiti jer ne postoji ranija ili je obrisana.",
        "undo-nochange": "Ovo je uređivanje izgleda već bilo poništeno.",
        "undo-summary": "Poništena izmjena $1 [[Special:Contribs/$2|korisnika $2]] ([[User talk:$2|razgovor]])",
        "search-interwiki-more": "(više)",
        "search-interwiki-more-results": "više stavka",
        "search-relatedarticle": "Povezano",
+       "search-invalid-sort-order": "Ne prepoznajem redoslijed sortiranja u $1; primijenit ću osnovno naručivanje. Važeći redoslijedi su: $2",
+       "search-unknown-profile": "Ne prepoznajem pretraživački profil $1; primijenit ćemo podrazumijevani.",
        "searchrelated": "povezano",
        "searchall": "sve",
        "showingresults": "Dole {{PLURAL:$1|je prikazana <strong>1</strong> stavka|su prikazane <strong>$1</strong> stavke|je prikazano <strong>$1</strong> stavki}} počev od br. <strong>$2</strong>.",
        "rcfilters-clear-all-filters": "Očisti sve filtre",
        "rcfilters-show-new-changes": "Pogl. nove izmjene nakon $1",
        "rcfilters-search-placeholder": "Filtriranje promene (koristite meni ili potražite za naziv filtera)",
+       "rcfilters-search-placeholder-mobile": "Filteri",
        "rcfilters-invalid-filter": "Nevažeći filter",
        "rcfilters-empty-filter": "Nema aktivnih filtera. Prikazani su svi doprinosi.",
        "rcfilters-filterlist-title": "Filteri",
        "undeleteviewlink": "pogledaj",
        "undeleteinvert": "Sve osim odabranog",
        "undeletecomment": "Razlog:",
-       "cannotundelete": "Vraćanje nije uspjelo:\n$1",
+       "cannotundelete": "Vraćanje nije uspjelo djelimično ili potpuno:\n$1",
        "undeletedpage": "'''$1 je vraćena'''\n\nProvjerite [[Special:Log/delete|evidenciju brisanja]] za zapise najskorijih brisanja i vraćanja.",
        "undelete-header": "Pogledajte [[Special:Log/delete|evidenciju brisanja]] za nedavno obrisane stranice.",
        "undelete-search-title": "Pretraga obrisanih stranica",
        "lockedbyandtime": "(od $1 dana $2 u $3)",
        "move-page": "Premještanje $1",
        "move-page-legend": "Premjesti stranicu",
-       "movepagetext": "Korištenjem ovog formulara možete preimenovati stranicu, premještajući cijelu historiju na novo ime.\nČlanak pod starim imenom će postati stranica koja preusmjerava na članak pod novim imenom. \nMožete automatski izmjeniti preusmjerenje do izvornog naslova.\nAko se ne odlučite na to, provjerite [[Special:DoubleRedirects|dvostruka]] ili [[Special:BrokenRedirects|neispravna preusmjeravanja]].\nDužni ste provjeriti da svi linkovi i dalje nastave voditi na prave stranice.\n\nImajte na umu da članak '''neće''' biti preusmjeren ukoliko već postoji članak pod imenom na koje namjeravate da preusmjerite osim u slučaju stranice za preusmjeravanje koja nema nikakvih starih izmjena.\nTo znači da možete vratiti stranicu na prethodno mjesto ako pogriješite, ali ne možete zamijeniti postojeću stranicu.\n\n'''Pažnja!'''\nOvo može biti drastična i neočekivana promjena kad su u pitanju popularne stranice;\nMolimo dobro razmislite prije nego što preimenujete stranicu.",
+       "movepagetext": "Korištenjem ovog obrasca možete preimenovati stranicu, premještajući cijelu historiju na novo ime.\nČlanak pod starim imenom će postati stranica koja preusmjerava na članak pod novim imenom. \nMožete automatski podnoviti preusmjerenja do izvornog naslova.\nAko se ne odlučite na to, provjerite [[Special:DoubleRedirects|dvostruka]] ili [[Special:BrokenRedirects|neispravna preusmjeravanja]].\nDužni ste provjeriti da svi linkovi i dalje nastave voditi na prave stranice.\n\nImajte na umu da stranica <strong>neće</strong> biti preusmjeren ukoliko već postoji stranica pod imenom na koje namjeravate da preusmjerite osim u slučaju stranice za preusmjeravanje koja nema nikakvih starih izmjena.\nTo znači da možete vratiti stranicu na prethodno mjesto ako pogriješite, ali ne možete zamijeniti postojeću stranicu.\n\n<strong>Napomena:</strong>\nOvo može biti drastična i neočekivana promjena kad su u pitanju popularne stranice;\nmolimo dobro razmislite prije nego što preimenujete stranicu.",
        "movepagetext-noredirectfixer": "Koristeći obrazac ispod ćete preimenovati stranicu i premjestiti cijelu njenu historiju na novi naziv.\nStari naziv će postati preusmjerenje na novi naziv.\nMolimo provjerite da li postoje [[Special:DoubleRedirects|dvostruka]] ili [[Special:BrokenRedirects|nedovršena preusmjerenja]].\nVi ste za to odgovorni te morate provjeriti da li su linkovi ispravni i da li vode tamo gdje bi trebali.\n\nImajte na umu da stranica <strong>neće</strong> biti premještena ako već postoji stranica s tim imenom, osim ako je prazna ili je preusmjerenje ili nema ranije historije.\nOvo znali da možete preimenovati stranicu nazad gdje je ranije bila preimenovana ako ste pogriješili a ne možete ponovo preimenovati postojeću stranicu.\n\n<strong>Napomena:</strong>\nImajte na umu da premještanje popularnog članka može biti\ndrastična i neočekivana promjena za korisnike; molimo budite sigurni da ste shvatili posljedice prije nego što nastavite.",
        "movepagetalktext": "Ako označite ovu kutijucu, pridružena stranica za razgovor će se automatski premjestiti na novi naslov, ukoliko ne-prazna stranica razgovor sa istim imenom već postoji. U tom slučaju ćete morati, ako želite, ručno premjestiti ili spojiti stranicu.\n\nU tom slučaju, morat ćete ručno premjestiti ili spojiti stranicu ako to želite.",
        "moveuserpage-warning": "'''Upozorenje:''' Premještate korisničku stranicu. Molimo da zapamtite da će se samo stranica premjestiti a korisnik se ''neće'' preimenovati.",
        "linkaccounts": "Spajanje računa",
        "linkaccounts-success-text": "Račun je spojen.",
        "linkaccounts-submit": "Spoji račune",
+       "cannotunlink-no-provider-title": "Nema povezanih računa za odvezivanje",
+       "cannotunlink-no-provider": "Nemate povezanih računa za odvezivanje.",
        "unlinkaccounts": "Razdvajanje računa",
        "unlinkaccounts-success": "Račun je razdvojen.",
        "authenticationdatachange-ignored": "Promjena podataka u autenifikaciji nije obrađena. Možda nije postavljen provajder?",
index 0dc68bc..e1605fa 100644 (file)
        "ns-specialprotected": "Stránky v mennom priestore {{ns:special}} nie je možné upravovať.",
        "titleprotected": "Používateľ [[User:$1|$1]] zabránil vytváraniu stránky s týmto názvom.\nUdaný dôvod: <em>$2</em>.",
        "filereadonlyerror": "Nebolo možné modifikovať súbor „$1“, pretože úložisko „$2“ je momentálne v režime len na čítanie.\n\nSprávca, ktorý ho zamkol ponúkol toto vysvetlenie: „$3“.",
+       "invalidtitle": "Neplatný názov",
        "invalidtitle-knownnamespace": "Neplatný názov s menným priestorom „$2“ a textom „$3“",
        "invalidtitle-unknownnamespace": "Neplatný názov s neznámym číslom menného priestoru „$1“ a textom „$2“",
        "exception-nologin": "Nie ste prihlásený",
        "virus-scanfailed": "kontrola zlyhala (kód $1)",
        "virus-unknownscanner": "neznámy antivírus:",
        "logouttext": "<strong>Práve ste sa odhlásili.</strong>\n\nUvedomte si, že niektoré stránky sa môžu naďalej zobrazovať ako keby ste boli prihlásený, až kým nevymažete vyrovnávaciu pamäť vášho prehliadača.",
+       "logging-out-notify": "Prebieha vaše odhlásenie, prosím čekejte.",
+       "logout-failed": "Teraz nie je možné odhlásiť sa:$1",
        "cannotlogoutnow-title": "Teraz nie je možné odhlásiť sa",
        "cannotlogoutnow-text": "Nie je možné odhlásiť sa, keď používate $1",
        "welcomeuser": "Vitajte,  $1 !",
        "botpasswords-existing": "Jestvujúce heslá pre botov",
        "botpasswords-createnew": "Vytvoriť nové heslo pre botov",
        "botpasswords-editexisting": "Zmeniť existujúce heslo bota",
+       "botpasswords-label-needsreset": "(heslo sa musí resetovať)",
        "botpasswords-label-appid": "Názov bota:",
        "botpasswords-label-create": "Vytvoriť",
        "botpasswords-label-update": "Aktualizovať",
        "savechanges": "Uložiť zmeny",
        "publishpage": "Publikovať stránku",
        "publishchanges": "Publikovať zmeny",
+       "savearticle-start": "Uložiť stránku...",
+       "savechanges-start": "Uložiť zmeny...",
+       "publishpage-start": "Publikovať stránku...",
        "publishchanges-start": "Zverejniť zmeny…",
        "preview": "Náhľad",
        "showpreview": "Zobraziť náhľad",
        "histfirst": "najstaršie",
        "histlast": "najnovšie",
        "historysize": "(({{PLURAL:$1|jeden bajt|$1 bajty|$1 bajtov}}))",
-       "historyempty": "(prázdne)",
+       "historyempty": "prázdne",
        "history-feed-title": "História úprav",
        "history-feed-description": "História úprav pre túto stránku na wiki",
        "history-feed-item-nocomment": "$1 na $2",
        "revdelete-unsuppress": "Odstrániť obmedzenia obnovených revízií",
        "revdelete-log": "Dôvod:",
        "revdelete-submit": "Použiť na {{PLURAL:$1|zvolenú revíziu|zvolené revízie}}",
-       "revdelete-success": "'''Viditeľnosť revízie bola úspešne aktualizovaná.'''",
+       "revdelete-success": "Viditeľnosť revízie bola aktualizovaná.",
        "revdelete-failure": "'''Viditeľnosť revízie nebolo možné aktualizovať:'''\n$1",
        "logdelete-success": "Viditeľnosť záznamu bola úspešne nastavená.",
        "logdelete-failure": "'''Viditeľnosť záznamu nebolo možné nastaviť:'''\n$1",
        "mergehistory-empty": "Žiadne revízie nie je možné zlúčiť.",
        "mergehistory-done": "$3 {{PLURAL:$3|revízia|revízie|revízií}} stránky $1 {{PLURAL:$3|bola úspešne zlúčená|boli úspešne zlúčené|bolo úspešne zlúčených}} do [[:$2]].",
        "mergehistory-fail": "Nepodarilo sa vykonať zlúčenie histórií. Prosím, skontrolujte parametre stránka a časy.",
+       "mergehistory-fail-bad-timestamp": "Časové razítko je neplatné.",
+       "mergehistory-fail-invalid-source": "Zdrojová stránka je neplatná.",
+       "mergehistory-fail-invalid-dest": "Cieľová stránka je neplatná.",
        "mergehistory-fail-toobig": "Nepodarilo sa zlúčiť histórie, pretože by sa presúvalo viac revízií, než je limit ($1).",
        "mergehistory-no-source": "Zdrojová stránka $1 neexistuje.",
        "mergehistory-no-destination": "Cieľová stránka $1 neexistuje.",
index ea6035c..72d7abb 100644 (file)
        "search-interwiki-more": "(več)",
        "search-interwiki-more-results": "več zadetkov",
        "search-relatedarticle": "Podobno",
+       "search-invalid-sort-order": "Vrstni red razvrščanja $1 nismo prepoznali, zato bomo uporabili privzeto razvrščanje. Veljavni vrstni redi razvrščanja so: $2",
+       "search-unknown-profile": "Iskalnega profila $1 nismo prepoznali, zato bomo uporabili privzeti iskalni profil.",
        "searchrelated": "povezano",
        "searchall": "vse",
        "showingresults": "Prikazujem do '''$1''' {{PLURAL:$1|zadetek|zadetka|zadetke|zadetkov}}, začenši s št. '''$2'''.",
        "rcfilters-clear-all-filters": "Počisti vse filtre",
        "rcfilters-show-new-changes": "Ogled novih sprememb od $1",
        "rcfilters-search-placeholder": "Filtriraj zadnje spremembe (uporabi meni ali vnesi ime filtra)",
+       "rcfilters-search-placeholder-mobile": "Filtri",
        "rcfilters-invalid-filter": "Neveljaven filter",
        "rcfilters-empty-filter": "Ni dejavnih filtrov. Prikazani so vsi prispevki.",
        "rcfilters-filterlist-title": "Filtri",
        "linkaccounts": "Poveži račune",
        "linkaccounts-success-text": "Račun smo povezali.",
        "linkaccounts-submit": "Poveži račune",
+       "cannotunlink-no-provider-title": "Ni povezanih računov za prekinitev povezave",
+       "cannotunlink-no-provider": "Ni povezanih računov, za katere bi lahko prekinili povezavo.",
        "unlinkaccounts": "Razveži račune",
        "unlinkaccounts-success": "Račun smo razvezali.",
        "authenticationdatachange-ignored": "Sprememba overitvenih podatkov ni bila obdelana. Morda ni bil konfiguriran noben ponudnik?",
index 7574f53..236c548 100644 (file)
        "grouppage-autoconfirmed": "{{ns:project}}:Përdorues të vërtetuar automatikisht",
        "grouppage-bot": "{{ns:project}}:Robotë",
        "grouppage-sysop": "{{ns:project}}:Administruesit",
-       "grouppage-interface-admin": "{{ns:project}}:Wikipedia:Administratorët e ndërfaqes",
+       "grouppage-interface-admin": "{{ns:project}}:Administratorët e ndërfaqes",
        "grouppage-bureaucrat": "{{ns:project}}:Burokratë",
        "grouppage-suppress": "{{ns:project}}:Shtypur",
        "right-read": "Lexo faqe",
index 918a8be..ee9fcad 100644 (file)
        "search-interwiki-more": "(mer)",
        "search-interwiki-more-results": "fler resultat",
        "search-relatedarticle": "Relaterad",
+       "search-invalid-sort-order": "Sorteringsföljden av $1 känns inte igen, standardsorteringen kommer används. Giltiga sorteringsföljder är: $2",
+       "search-unknown-profile": "Sökprofilen för $1 känns inte igen, standardsökprofilen kommer används.",
        "searchrelated": "relaterad",
        "searchall": "alla",
        "showingresults": "Nedan visas upp till {{PLURAL:$1|<strong>1</strong> resultat|<strong>$1</strong> resultat}} från och med nummer <strong>$2</strong>.",
        "linkaccounts": "Länka konton",
        "linkaccounts-success-text": "Kontot länkades.",
        "linkaccounts-submit": "Länka konton",
+       "cannotunlink-no-provider-title": "Det finns inga länkade konton att avlänka",
+       "cannotunlink-no-provider": "Det finns inga länkade konton som kan avlänkas.",
        "unlinkaccounts": "Avlänka konton",
        "unlinkaccounts-success": "Kontot avlänkades.",
        "authenticationdatachange-ignored": "Ändringen av autentiseringsdata hanterades inte. Kanske ingen tillhandahållare har konfigurerats?",
index cb1929a..f50c18c 100644 (file)
        "virus-scanfailed": "స్కాన్ విఫలమైంది (సంకేతం $1)",
        "virus-unknownscanner": "అజ్ఞాత యాంటీవైరస్:",
        "logouttext": "<strong>ఇప్పుడు మీరు లాగౌటయ్యారు.</strong>\n\nఅయితే, ఓ గమనిక.. మీ విహారిణిలోని కోశాన్ని ఖాళీ చేసేవరకూ కొన్ని పేజీలు మీరింకా లాగినై ఉన్నట్లుగానే చూపించవచ్చు.",
+       "logout-failed": "ఇప్పుడు లాగౌట్ అవలేరు: $1",
        "cannotlogoutnow-title": "ఇప్పుడు లాగౌట్ అవలేరు",
        "cannotlogoutnow-text": "$1 ను వాడుతూండగా లాగౌట్ అవలేరు.",
        "welcomeuser": "స్వాగతం, $1!",
        "autoblockedtext": "మీ ఐపీ చిరునామా ఆటోమాటిగ్గా నిరోధించబడింది. ఎందుకంటే ఇదే ఐపీ చిరునామాని ఓ నిరోధిత వాడుకరి ఉపయోగించారు. ఆ వాడుకరిని $1 నిరోధించారు.\nఅందుకు ఇచ్చిన కారణం ఇదీ:\n\n:<em>$2</em>\n\n* నిరోధం మొదలైన సమయం: $8\n* నిరోధించిన కాలం: $6\n* ఉద్దేశించిన నిరోధిత వాడుకరి: $7\n\nఈ నిరోధం గురించి చర్చించేందుకు మీరు $1 ను గానీ, లేదా ఇతర [[{{MediaWiki:Grouppage-sysop}}|నిర్వాహకులను]] గానీ సంప్రదించండి.\n\nమీ [[Special:Preferences|అభిరుచులలో]] సరైన ఈమెయిలు ఐడీని ఇచ్చి ఉంటే తప్ప, మీరు \"ఈ వాడుకరికి ఈమెయిలు పంపు\" అనే అంశాన్ని వాడజాలరని గమనించండి. ఆ సౌలభ్యాన్ని వాడటం నుండి మిమ్మల్ని నిరోధించలేదు.\n\nమీ ప్రస్తుత ఐపీ చిరునామా $3, నిరోధపు ఐడీ: $5.\nమీ సంప్రదింపులన్నిటిలోను అన్ని పై వివరాలను ఉదహరించండి.",
        "systemblockedtext": "మీడియావికీ మీ వాడుకరిపేరు లేదా ఐపీ అడ్రసును ఆటోమాటిగ్గా నిరోధించింది.\nఅందుకు ఇచ్చిన కారణం:\n\n:<em>$2</em>\n\n* నిరోధం మొదలైన సమయం: $8\n* నిరోధం ముగిసే సమయం: $6\n* నిరోధానికి గురైనవారు: $7\n\nమీ ప్రస్తుత ఐపీ అడ్రసు $3.\nమీ సంప్రదింపులన్నిటిలోనూ పై వివరాలను పేర్కొనండి.",
        "blockednoreason": "కారణమేమీ ఇవ్వలేదు",
+       "blockedtext-composite-no-ids": "మీ ఐపీ చిరునామా అనేక నిరోధజాబితాల్లో ఉంది",
+       "blockedtext-composite-reason": "మీ ఖాతాపై గాని ఐపీ చిరునామాపై గానీ అనేక నిరోధాలున్నాయి",
        "whitelistedittext": "పుటలలో మార్పులు చెయ్యడానికి $1.",
        "confirmedittext": "పేజీల్లో మార్పులు చేసేముందు మీ ఈమెయిలు చిరునామాను ధృవీకరించాలి. [[Special:Preferences|మీ అభిరుచుల]]లో మీ ఈమెయిలు చిరునామా రాసి, ధృవీకరించండి.",
        "nosuchsectiontitle": "విభాగాన్ని కనగొనలేకపోయాం",
        "userjsonyoucanpreview": "<strong>చిట్కా:</strong> కొత్త JSON ను భద్రపరచే ముందు, \"{{int:showpreview}}\" మీటను నొక్కి దాన్ని పరీక్షించండి.",
        "userjsyoucanpreview": "<strong>చిట్కా:</strong> భద్రపరిచేముందు మీ కొత్త జావాస్క్రిప్టుని పరీక్షించడానికి \"{{int:showpreview}}\" బొత్తాన్ని వాడండి.",
        "usercsspreview": "<strong>మీరు వాడుకరి CSSను కేవలం సరిచూస్తున్నారని గుర్తుంచుకోండి.\nదాన్నింకా భద్రపరచలేదు!</strong>",
+       "userjsonpreview": "<strong>మీరు మీ JSON కాన్ఫిగ్ పరీక్షిస్తున్నారు/మునుజూపును చూస్తున్నారు మాత్రమేనని గుర్తుంచుకోండి.\nదీన్నింకా భద్రపరచలేదు!</strong>",
        "userjspreview": "<strong>గుర్తుంచుకోండి, మీరింకా మీ వాడుకరి జావాస్క్రిప్ట్&zwnj;ను భద్రపరచలేదు, కేవలం పరీక్షిస్తున్నారు/సరిచూస్తున్నారు!</strong>",
        "sitecsspreview": "'''మీరు చూస్తున్నది ఈ CSS మునుజూపును మాత్రమేనని గుర్తుంచుకోండి.'''\n'''దీన్నింకా భద్రపరచలేదు!'''",
+       "sitejsonpreview": "<strong>మీరు మీ JSON కాన్ఫిగ్ మునుజూపును చూస్తున్నారు మాత్రమే అని గుర్తుంచుకోండి.\nదీన్నింకా భద్రపరచలేదు!</strong>",
        "sitejspreview": "'''మీరు చూస్తున్నది ఈ JavaScript మునుజూపును మాత్రమేనని గుర్తుంచుకోండి.''' \n'''దీన్నింకా భద్రపరచలేదు!'''",
        "userinvalidconfigtitle": "<strong>హెచ్చరిక:</strong> \"$1\" అనే రూపు లేదు.\nఅభిమత .css, .js పుటల శీర్షికలు ఇంగ్లీషు చిన్నబడి అక్షరాల లోనే ఉంటాయి. ఉదాహరణ: {{ns:user}}:Foo/vector.css. అంతేగానీ, {{ns:user}}:Foo/Vector.css ఇలా కాదు.",
        "updated": "(నవీకరించబడింది)",
        "yourtext": "మీ పాఠ్యం",
        "storedversion": "భద్రపరచిన కూర్పు",
        "editingold": "<strong>హెచ్చ రిక: ఈ పేజీ యొక్క కాలం చెల్లిన సంచికను మీరు మరుస్తున్నారు.</strong> దీనిని భద్రపరిస్తే, ఆ సంచిక తరువాత జరిగిన మార్పులన్నీ పోతాయి.",
+       "unicode-support-fail": "మీ బ్రౌజరు యూనికోడ్‌కు అనుకూలంగా ఉన్నట్లు లేదు. పేజీలను సరిదిద్దాలంటే అది ఆవశ్యకం. అంచేత మీ దిద్దుబాటును భద్రపరచలేదు.",
        "yourdiff": "తేడాలు",
        "copyrightwarning": "{{SITENAME}}కు సమర్పించే అన్ని రచనలూ $2కు లోబడి ప్రచురింపబడినట్లుగా భావించబడతాయి (వివరాలకు $1 చూడండి). మీ రచనలను ఎవ్వరూ మార్చ రాదనీ లెదా వేరే ఎవ్వరూ వాడుకో రాదని మీరు భావిస్తే, ఇక్కడ ప్రచురించకండి.<br /> మీ స్వీయ రచనను గాని, సార్వజనీనమైన రచననుగాని, ఇతర ఉచిత వనరుల నుండి సేకరించిన రచననుగాని మాత్రమే ప్రచురిస్తున్నానని కూడా మీరు ప్రమాణం చేస్తున్నారు. <strong>కాపీహక్కులుగల రచనను తగిన అనుమతి లేకుండా సమర్పించకండి!</strong>",
        "copyrightwarning2": "{{SITENAME}}లో ప్రచురించే రచనలన్నిటినీ ఇతర రచయితలు సరిదిద్దడం, మార్చడం, తొలగించడం జరగవచ్చు. మీ రచనలను అలా నిర్దాక్షిణ్యంగా దిద్దుబాట్లు చెయ్యడం మీకిష్టం లేకపోతే, వాటిని ఇక్కడ ప్రచురించకండి. <br />\nఅలాగే, ఈ రచనను మీరే చేసారని, లేదా ఏదైనా సార్వజనిక వనరు నుండి గానీ, అలాంటి ఉచిత, స్వేచ్ఛా వనరు నుండి గానీ కాపీ చేసి తెచ్చారని మాకు వాగ్దానం చేస్తున్నారు. (వివరాలకు $1 చూడండి).\n<strong>తగు అనుమతులు లేకుండా కాపీ హక్కులు గల రచనలను సమర్పించకండి!</strong>",
        "permissionserrors": "అనుమతి లోపం",
        "permissionserrorstext": "కింద పేర్కొన్న {{PLURAL:$1|కారణం|కారణాల}} మూలంగా, ఆ పని చెయ్యడానికి మీకు అనుమతిలేదు:",
        "permissionserrorstext-withaction": "ఈ క్రింది {{PLURAL:$1|కారణం|కారణాల}} వల్ల, $2 అనుమతి మీకు లేదు:",
+       "contentmodelediterror": "ఈ కూర్పు కంటెంటు మోడలు <code>$1</code> కావడం చేత మీరు దీనిలో దిద్దుబాటు చెయ్యలేరు. ఈ మోడలు ఈ పేజీ యొక్క ప్రస్తుత కంటెంటు మోడలైన <code>$2</code> కంటే భిన్నంగా ఉంది.",
        "recreate-moveddeleted-warn": "<strong>హెచ్చరిక: ఇంతకు మునుపు ఒకసారి తొలగించిన పేజీని మళ్లీ సృష్టిద్దామని మీరు ప్రయత్నిస్తున్నారు.</strong>\n\nఈ పేజీపై మార్పులు చేసేముందు, అవి ఇక్కడ ఉండతగినవేనా కాదా అని ఒకసారి ఆలోచించండి.\nమీ సౌలభ్యం కొరకు ఈ పేజీ తొలగింపు, తరలింపు చిట్టాలను ఇక్కడ ఇచ్చాం:",
        "moveddeleted-notice": "ఈ పేజీని తొలగించారు.\nసమాచారానికై ఈ పేజీ తొలగింపు, సంరక్షణ, తరలింపుల చిట్టాను క్రింద ఇచ్చాం.",
        "moveddeleted-notice-recent": "సారీ, ఈ పేజీని ఈమధ్యే తొలగించారు (గత 24 గంటల్లో).\nఈ పేజీకి సంబంధించిన తొలగింపు, సంరక్షణ, తరలింపు లాగ్‌లను కింద ఇచ్చాం.",
        "edit-gone-missing": "పేజీని తాజాకరించలేకపోయాం.\nదాన్ని తొలగించినట్టున్నారు.",
        "edit-conflict": "దిద్దుబాటు ఘర్షణ.",
        "edit-no-change": "పాఠ్యంలో మార్పులేమీ చెయ్యలేదు కాబట్టి, మీ మార్పును పట్టించుకోవట్లేదు.",
+       "edit-slots-cannot-add": "కింది {{PLURAL:$1|స్లాటుకు|స్లాట్లకు}} ఇక్కడ మద్దతు లేదు: $2.",
+       "edit-slots-cannot-remove": "కింది {{PLURAL:$1|స్లాటు|స్లాట్లు}} ఆవశ్యకం, కాబట్టి తీసెయ్యలేరు: $2.",
+       "edit-slots-missing": "కింది {{PLURAL:$1|స్లాటు|స్లాట్లు}} కనబడ్డం లేదు: $2.",
        "postedit-confirmation-created": "పేజీ సృష్టించబడినది.",
        "postedit-confirmation-restored": "పేజీని పునస్థాపించాం.",
        "postedit-confirmation-saved": "మీ మార్పు భద్రమయ్యింది.",
        "invalid-content-data": "చెల్లని కంటెంటు డేటా",
        "content-not-allowed-here": "స్లాట్ \"$3\" లో [[:$2]] పేజీలో పాఠ్యం \"$1\" కి అనుమతి లేదు",
        "editwarning-warning": "ఈ పేజీని వదిలివెళ్ళడం వల్ల మీరు చేసిన మార్పులను కోల్పోయే అవకాశం ఉంది.\nమీరు లాగిన్ అయివుంటే, ఈ హెచ్చరికని మీ అభిరుచులలోని \"{{int:prefs-editing}}\"  విభాగంలో అచేతనం చేసుకోవచ్చు.",
+       "editpage-invalidcontentmodel-title": "ఈ కంటెంటు మోడలుకు మద్దతు లేదు",
        "editpage-notsupportedcontentformat-title": "పాఠ్యపు ఆకృతికి మద్దతు లేదు",
        "editpage-notsupportedcontentformat-text": "$2 పాఠ్యపు మోడల్, పాఠ్యపు ఆకృతి $1 కి మద్దతు ఇవ్వదు",
        "slot-name-main": "ప్రధాన",
        "right-hideuser": "బయటికి కనబడకుండా చేసి, వాడుకరిపేరును నిరోధించు",
        "right-ipblock-exempt": "ఐపీ నిరోధాలు, ఆటో నిరోధాలు, శ్రేణి నిరోధాలను తప్పించు",
        "right-unblockself": "స్వీయ అనిరోధం",
-       "right-protect": "à°¸à°\82à°°à°\95à±\8dà°·à°£ à°¸à±\8dథాయిలనà±\81 à°®à°¾à°°à±\8dà°\9aà±\81, à°\95ాసà±\8dà°\95à±\87à°¡à±\8d-à°°à°\95à±\8dà°·à°¿à°¤ à°ªà±\87à°\9cà±\80లలà±\8b à°¦à°¿à°¦à±\8dà°¦à±\81బాà°\9fà±\81 à°\9aà±\86à°¯à±\8dయి",
+       "right-protect": "à°¸à°\82à°°à°\95à±\8dà°·à°£ à°¸à±\8dథాయిలనà±\81 à°®à°¾à°°à±\8dà°\9aà±\81, à°\95ాసà±\8dà°\95à±\87à°¡à±\8d-à°°à°\95à±\8dà°·à°¿à°¤ à°ªà±\87à°\9cà±\80లలà±\8b à°¦à°¿à°¦à±\8dà°¦à±\81బాà°\9fà±\81 à°\9aà±\86à°¯à±\8dయడà°\82",
        "right-editprotected": "\"{{int:protect-level-sysop}}\" గా సంక్షించబడిన పేజీలను సరిదిద్దు",
        "right-editsemiprotected": "\"{{int:protect-level-autoconfirmed}}\" గా సంరక్షించబడ్డ పేజీలను మార్చు",
        "right-editcontentmodel": "పేజీ యొక్క కంటెంటు మోడలును సవరించు",
        "action-applychangetags": "మీ మార్పులతో ట్యాగులను ఆపాదించే",
        "action-deletechangetags": "డేటాబేసు నుండి ట్యాగులను తొలగించే",
        "action-purge": "ఈ పేజీని పర్జ్ చేసే",
+       "action-blockemail": "ఈమెయిలు పంపకుండా వాడుకరిని నిరోధించే",
+       "action-bot": "ఆటోమాటిక్ ప్రాసెస్ లాగా భావించే",
+       "action-editsemiprotected": "\"{{int:protect-level-autoconfirmed}}\" గా సంరక్షించబడ్డ పేజీలను మార్చే",
+       "action-editinterface": "యూజరు ఇంటరుఫేసులో దిద్దుబాటు చేసే",
+       "action-editusercss": "ఇతర వాడుకరుల CSS ఫైళ్ళలో దిద్దుబాటు చేసే",
+       "action-edituserjson": "ఇతర వాడుకరుల JSON ఫైళ్ళలో దిద్దుబాటు చేసే",
+       "action-edituserjs": "ఇతర వాడుకరుల JavaScript ఫైళ్ళలో దిద్దుబాటు చేసే",
+       "action-editsitecss": "సైటువ్యాప్త CSS ను దిద్దుబాటు చేసే",
+       "action-editsitejson": "సైటువ్యాప్త JSON ను దిద్దుబాటు చేసే",
+       "action-editsitejs": "సైటువ్యాప్త JavaScript ను దిద్దుబాటు చేసే",
+       "action-editmyusercss": "మీ స్వంత వాడుకరి CSS ఫైళ్ళలో దిద్దుబాటు చేసే",
+       "action-editmyuserjson": "మీ స్వంత JSON ను దిద్దుబాటు చేసే",
+       "action-editmyuserjs": "మీ స్వంత JavaScript ఫైళ్ళలో దిద్దుబాటు చేసే",
+       "action-editmyuserjsredirect": "దారిమార్పు చేసే మీ స్వంత JavaScript ఫైళ్లలో దిద్దుబాటు చేసే",
+       "action-viewsuppressed": "ఏ వాడుకరి నుండైనా దాచబడిన కూర్పులను చూసే",
+       "action-hideuser": "వాడుకరిపేరును నిరోధించి, దాన్ని బయటికి కనబడకుండా చేసే",
+       "action-ipblock-exempt": "ఐపీ నిరోధాలు, ఆటో నిరోధాలు, శ్రేణి నిరోధాలను తప్పించే",
+       "action-unblockself": "స్వీయ నిరోధాన్ని తొలగించే",
+       "action-noratelimit": "రేటు పరిమితులు ప్రభావం చూపని",
+       "action-reupload-own": "స్వయంగా అప్‌లోడు చేసిన ఫైళ్ళను తిరగరాసే",
+       "action-nominornewtalk": "చర్చా పేజీల్లో చేసే చిన్న మార్పులు కొత్త సందేశపు ప్రాంప్టును పంపే",
+       "action-markbotedits": "వెనక్కి తెచ్చిన దిద్దుబాట్లను బాట్ దిద్దుబాట్లుగా గుర్తించే",
+       "action-patrolmarks": "ఇటీవలి మార్పుల నిఘా గుర్తింపులను చూసే",
+       "action-override-export-depth": "5 లింకుల లోతు వరకు ఉన్న పేజీలతో సహా, పేజీలను ఎగుమతి చేసే",
+       "action-suppressredirect": "పేజీని తరలించేటపుడు పాత పేరు నుండి దారిమార్పును సృష్టించకుండా చేసే",
        "nchanges": "{{PLURAL:$1|ఒక మార్పు|$1 మార్పులు}}",
        "enhancedrc-since-last-visit": "{{PLURAL:$1|చివరి సందర్శన తరువాత}}, $1",
        "enhancedrc-history": "చరిత్ర",
        "rcfilters-highlightmenu-help": "ఈ లక్షణాన్ని హైలైటు చేసేందుకు ఓ రంగును ఎంచుకోండి",
        "rcfilters-filterlist-noresults": "వడపోతకాలేమీ కనబడలేదు",
        "rcfilters-noresults-conflict": "వెతకాల్సిన పదాలు పరస్పరం ఘర్షిస్తున్నందువలన ఫలితాలేమీ రాలేదు.",
+       "rcfilters-state-message-subset": "ఈ వడపోత ప్రభావం ఏమీ లేదు. ఎందుకంటే దీని ఫలితాలు కింది విస్తృత {{PLURAL:$2|వడపోత|వడపోతల}} ఫలితాల్లో ఉండనే ఉన్నాయి (పోనీ, అవి స్ఫుటంగా కనిపించాలంటే వాటిని హైలైటు చెయ్యండి): $1",
        "rcfilters-state-message-fullcoverage": "ఈగుంపులోని వడపోతలన్నిటినీ ఎంచుకోవడమూ, అసలు దేన్నీ ఎంచుకోకపోవడమూ ఒక్కటే. అంచేత ఈ వడపోత ప్రభావమేమీ ఉండదు. గుంపులో ఈ వడపోతలున్నాయి: $1",
        "rcfilters-filtergroup-authorship": "కర్తృత్వం",
        "rcfilters-filter-editsbyself-label": "మీ దిద్దుబాట్లు",
        "rcfilters-filter-reviewstatus-unpatrolled-label": "నిఘాలో లేనివి",
        "rcfilters-filter-reviewstatus-manual-description": "నిఘాలో ఉన్నట్లు గుర్తించని మార్పులు.",
        "rcfilters-filter-reviewstatus-manual-label": "మనవికంగా నిఘాలో ఉన్నవి",
+       "rcfilters-filter-reviewstatus-auto-label": "ఆటోపాట్రోల్లో ఉన్నవి",
        "rcfilters-filtergroup-significance": "ప్రాముఖ్యం",
        "rcfilters-filter-minor-label": "చిన్న మార్పులు",
        "rcfilters-filter-minor-description": "రచయిత చిన్నవిగా గుర్తు పెట్టిన దిద్దుబాట్లు.",
        "rcfilters-watchlist-edit-watchlist-button": "మీ వీక్షణ జాబితాను సవరించండి",
        "rcfilters-watchlist-showupdated": "మీ గత సందర్శన తరువాత మారిన పేజీలు '''బొద్దుగా'''ను, నింపిన గుండ్రని చుక్కల ద్వారానూ చూపించాం.",
        "rcfilters-preference-label": "జావాస్క్రిప్టు అవసరంలేని అంతరవర్తిని వాడు",
+       "rcfilters-preference-help": "వడపోతల వెతుకులాట, హైలైటింగూ లేకుండా ఇటీవలి మార్పులను చూపిస్తుంది.",
+       "rcfilters-watchlist-preference-label": "జావాస్క్రిప్టు లేని ఇంటర్‌ఫేసును వాడు",
+       "rcfilters-watchlist-preference-help": "వడపోతల వెతుకులాట, హైలైటింగూ లేకుండా వీక్షణజాబితాను చూపిస్తుంది.",
+       "rcfilters-filter-showlinkedfrom-label": "ఓ పేజీకి లింకై ఉన్న పేజీల్లో జరిగిన మార్పులను చూపించు",
        "rcfilters-filter-showlinkedfrom-option-label": "ఎంచుకున్న పేజీ నుండి <strong>లింకైన పేజీలు</strong>",
+       "rcfilters-filter-showlinkedto-label": "ఓ పేజీ నుండి లింకై ఉన్న పేజీల్లో జరిగిన మార్పులను చూపించు",
        "rcfilters-filter-showlinkedto-option-label": "ఎంచుకున్న పేజీకి <strong>లింకైన పేజీలు</strong>",
        "rcfilters-target-page-placeholder": "పేజీ (లేదా వర్గం) పేరు ఇవ్వండి",
        "rcnotefrom": "<strong>$3, $4</strong> తరువాత జరిగిన {{PLURAL:$5|మార్పు|మార్పులు}} కింద ఇచ్చాం (<strong>$1</strong> దాకా చూపించాం).",
        "uploadbtn": "దస్త్రాన్ని ఎక్కించు",
        "reuploaddesc": "మళ్ళీ అప్‌లోడు ఫారంకు వెళ్ళు.",
        "upload-tryagain": "మార్చిన ఫైలు వివరణని దాఖలుచేయండి",
+       "upload-tryagain-nostash": "తిరిగి ఎక్కించిన చేసిన ఫైలును, మార్చిన వివరణనూ సమర్పించండి",
        "uploadnologin": "లాగిన్‌ అయిలేరు",
        "uploadnologintext": "దస్త్రాలను ఎక్కించడానికి మీరు $1 ఉండాలి.",
        "upload_directory_missing": "ఎక్కింపు డైరెక్టరీ ($1) కనబడలేదు. పైగా వెబ్ సర్వర్ దాన్ని సృష్టించలేకపోయింది.",
        "file-deleted-duplicate-notitle": "సరిగ్గా ఈ ఫైలునే పోలిన మరో ఫైలును గతంలో తొలగించాం. దాని పేరును అణచిపెట్టాం.\nదాన్ని తిరిగి ఎక్కించే ముందు, పరిస్థితిని సమీక్షించేందుకు గాను, అణచబడిన ఫైళ్ళ డేటాను చూడగలిగే వారిని అడగండి.",
        "uploadwarning": "ఎక్కింపు హెచ్చరిక",
        "uploadwarning-text": "కింద ఈ ఫైలు వివరణను మార్చి, మళ్ళీ ప్రయత్నించండి.",
+       "uploadwarning-text-nostash": "ఫైలును తిరిగి ఎక్కించి, కింది వివరణను మార్చి మళ్ళీ ప్రయత్నించండి.",
        "savefile": "దస్త్రాన్ని భద్రపరచు",
        "uploaddisabled": "ఎక్కింపులు అచేతనం చెయ్యబడ్డాయి.",
        "copyuploaddisabled": "URL ద్వారా ఎక్కింపుని అశక్తం చేసారు.",
index b016fc5..41436b8 100644 (file)
@@ -34,6 +34,8 @@
        "tog-watchdefault": "Саҳифаҳо ва парвандаҳое, ки вироиш мекунам ба феҳристи пайгириҳоям афзуда шавад",
        "tog-watchmoves": "Саҳифаҳо ва парвандаҳое, ки мунтақил мекунам ба феҳристи пайгириҳоям афзуда шавад",
        "tog-watchdeletion": "Саҳифаҳо ва парвандаҳое, ки ҳазф мекунам ба феҳристи пайгириҳоям афзуда шавад",
+       "tog-watchuploads": "Файлҳои боршудаи маро ба феҳристӣ пайгириҳо илова намуд",
+       "tog-watchrollback": "Саҳифаҳое, ки ман вогардонидам ба феҳристи пайгириҳоям илова намо",
        "tog-minordefault": "Пешфарзи ҳамаи вироишҳоро ҷузъи ишора кунед",
        "tog-previewontop": "Намоиши пешнамоиши қаблӣ пеш аз қуттии вироиш ва на пас аз он",
        "tog-previewonfirst": "Нишон додани пешнамоиш дар нахустин вироиш",
@@ -50,6 +52,7 @@
        "tog-watchlisthidebots": "Пинҳон намудани вироишҳои бот дар феҳристи назарот",
        "tog-watchlisthideminor": "Пинҳон намудани вироишҳои хурд дар феҳристи назарот",
        "tog-watchlisthideliu": "Пинҳон кардани вироишоти корбарони вурудшуда аз феҳристи пайгириҳо",
+       "tog-watchlistreloadautomatically": "Феҳристи пайгириҳоро ҳар дафъе, ки филтр тағйир меёбад аз нав бор кун (JavaScript лозим аст)",
        "tog-watchlisthideanons": "Пинҳон кардани вироишоти корбарони гумном аз феҳристи пайгириҳо",
        "tog-watchlisthidepatrolled": "Пинҳони вироишҳои гаштхӯрда аз феҳристи пайгириҳо",
        "tog-watchlisthidecategorization": "Пинҳон кардани гурӯҳбандии саҳифаҳо",
        "views": "Назарот",
        "toolbox": "Абзорҳо",
        "tool-link-userrights-readonly": "Дидани гурӯҳҳои корбар",
+       "tool-link-emailuser": "Ба ин {{GENDER:$1|корбар}} паём фиристодан",
        "imagepage": "Намоиши саҳифаи парванда",
        "mediawikipage": "Намоиши саҳифаи акс",
        "templatepage": "Нигаристани саҳифаи шаблон",
        "versionrequired": "Нусхаи $1 аз нармафзори МедиаВики лозим аст",
        "versionrequiredtext": "Барои истифодаи ин саҳифа ба нусхаи $1 аз нармафзори МедиаВики ниёз доред. Барои иттилооъ аз нусхаи нармафзори насбшуда дар ин вики ба [[Special:Version|ин саҳифа]] нигаред.",
        "ok": "Бошад",
+       "pagetitle": "$1 - {{SITENAME}}",
        "pagetitle-view-mainpage": "{{SITENAME}}",
        "retrievedfrom": "Баргирифта аз \"$1\"",
        "youhavenewmessages": "Шумо $1 ($2) доред.",
        "page-atom-feed": "Барои \"$1\" Atom Хабархон",
        "feed-atom": "Атом",
        "red-link-title": "$1 (саҳифа вуҷуд надорад)",
+       "sort-descending": "Аз поён ба боло  ба тартиб овардан",
+       "sort-ascending": "Аз поён ба боло ба тартиб даровардан",
        "nstab-main": "Мақола",
        "nstab-user": "Саҳифаи корбар",
        "nstab-media": "Расона",
        "filerenameerror": "Натавонистам парвандаи \"$1\" ба \"$2\" тағйири ном диҳам.",
        "filedeleteerror": "Парванда \"$1\" ҳазф натавонист шуд.",
        "directorycreateerror": "Имкони эҷоди пӯшаи \"$1\" вуҷуд надорад.",
+       "directoryreadonlyerror": "Каталог \"$1\" танҳо барои хондан дастрас аст.",
        "filenotfound": "Парвандаи \"$1\" ёфт нашуд.",
        "unexpected": "Миқдори ғайри мунтазир: \"$1\"=\"$2\".",
        "formerror": "Хато: наметавон формро фиристод",
        "virus-scanfailed": "пуиш номуваффақ (рамзи $1)",
        "virus-unknownscanner": "антивируси ношинос:",
        "logouttext": "<strong>Акнун аз систем хориҷ шудаед.</strong>\n\nШумо метавонед гумном аз {{SITENAME}} истифодабариро идома диҳед, ё метавонед бо ҳамин номи корбариатон ва ё номи корбарии дигаре <span class='plainlinks'>[$1 боз вуруд кунед]</span>.\nТавваҷӯҳ кунед, ки баъзе аз саҳифаҳо қаблан чи тавре намоиш шуда буданд ҳамин тавр намоиш дода мешаванд, то даме ки шумо ҳофизаи мурургаратонро пок кунед.",
+       "cannotlogoutnow-title": "Ҳозир хуруҷ шудан ғайриимкон аст",
        "welcomeuser": "Хуш омадед, $1!",
        "yourname": "Номи корбар",
        "userlogin-yourname": "Номи корбарӣ",
        "userlogin-remembermypassword": "Вурудшуда манро нигоҳ дор",
        "userlogin-signwithsecure": "Истифодаи пайвастшавии амн",
        "cannotlogin-title": "Наметавонед дохил шавед",
+       "cannotlogin-text": "Дохилшавӣ ғайриимкон аст.",
+       "cannotloginnow-title": "Ҳозир вуруд шудан намешавад",
+       "cannotcreateaccount-title": "Ҳисобе сохта наметавонед",
        "yourdomainname": "Домейни Шумо",
        "password-change-forbidden": "Шумо гузарвожаро дар ин вики тағийр дода наметавонед.",
        "externaldberror": "Хатое дар иртибот бо пойгоҳи дода рух дода аст ё ин ки шумо иҷозат ба рӯзрасонии ҳисоби берунии худро надоред.",
        "createacct-reason-ph": "Барои чӣ ҳисоби дигареро эҷод карда истодаед",
        "createacct-submit": "Ҳисоби худро созед",
        "createacct-another-submit": "Ҳисоби дигаре созед",
+       "createacct-continue-submit": "Идомаи эҷоди ҳисоб",
+       "createacct-another-continue-submit": "Идомаи эҷоди ҳисоб",
        "createacct-benefit-heading": "{{SITENAME}} тавассути одамони мисли шумо сохта шудааст.",
        "createacct-benefit-body1": "{{PLURAL:$1|вироиш|вироишҳо}}",
        "createacct-benefit-body2": "{{PLURAL:$1|саҳифа|саҳифаҳо}}",
        "nocookieslogin": "{{SITENAME}} барои ворид кардани корбарон ба систем аз кукиҳо (cookies) истифода мекунад. Кукиҳо фаъол нестанд. Лутфан кукиҳоро фаъол карда бори дигар бисанҷед.",
        "nocookiesfornew": "Ҳисоби корбарӣ сохта нашуд, чун мо манбаъи онро тасдиқ карда натавонистем.\nМутмаин бошед, ки кукиҳои мурургар фаъоланд, ин саҳифро аз нав кушода бори дигар саъй кунед.",
        "noname": "Номи корбари дурустеро шумо пешниҳод накардед.",
-       "loginsuccesstitle": "Вуруд бо муваффақият",
+       "loginsuccesstitle": "Ворид шудед",
        "loginsuccess": "'''Шумо акнун ба Википедия ҳамчун \"$1\". вуруд кардед'''",
        "nosuchuser": "Корбаре бо номи \"$1\" вуҷуд надорад.\nАмали номро барраси кунед, ё [[Special:CreateAccount|ҳисоби ҷадидеро эҷод кунед]].",
        "nosuchusershort": "Ягон корбаре бо номи \"$1\" вуҷуд надорад. Тарзи навишти номро санҷед.",
        "pt-userlogout": "Хуруҷ",
        "changepassword": "Иваз намудани калимаи убур",
        "resetpass_announce": "Барои анҷоми вурудшавӣ, шумо бояд гузарвожаи навро ворид кунед.",
+       "resetpass_text": "<!-- Матнро дар инҷо ворид намоед -->",
        "resetpass_header": "Тағйири гузарвожаи ҳисоб",
        "oldpassword": "Калимаи кӯҳнаи убур:",
        "newpassword": "Калимаи нави убур:",
        "retypenew": "Калимаи нави убурро такроран нависед:",
        "resetpass_submit": "Калимаи убурро танзим карда ба систем вуруд кунед",
-       "changepassword-success": "Гузарвожаи шумо бо муваффақият тағйир дода шуд!",
+       "changepassword-success": "Гузарвожаи шумо тағйир дода шуд!",
+       "botpasswords": "Гузарвожаҳои бот",
        "botpasswords-label-appid": "Номи бот:",
        "botpasswords-label-create": "Эҷод",
        "botpasswords-label-update": "Азнав бор кардан",
        "content-model-wikitext": "вики-матн",
        "content-model-text": "матни содда",
        "content-model-javascript": "ҶаваСкрипт",
+       "content-model-css": "CSS",
        "content-json-empty-object": "Ашёи холӣ",
+       "content-json-empty-array": "Массиви холӣ",
        "post-expand-template-inclusion-warning": "<strong>Огоҳӣ:</strong> Шаблони дар баргирифта хеле калон аст. Баъзе аз шаблонҳо дар бар гирифта намешавад.",
        "post-expand-template-inclusion-category": "Саҳифаҳое ки дар он ҳаҷми шаблон беш аз ҳад аст",
        "post-expand-template-argument-warning": "<strong>Огоҳӣ:</strong> Ин саҳифа ҳаддиақал як шаблоне дорад, ки хосияти ҳаҷм калони васеъшавӣ дорад.\nИн хосият аз байр гирифта шуд.",
        "post-expand-template-argument-category": "Саҳифаҳои ҳавои шаблонҳои бо параметрҳои нодида гирифташуда",
+       "parser-unstrip-loop-warning": "Unstrip -и пушиданашуда ёфт шуд",
        "undo-success": "Ин вироиш метавонад ботил шавад. Лутфан муқоисаи зеринро барои таъйид кардани амалӣ худ, баррасӣ кунед, ва баъдан барои анҷом додани ботилкунии вироиш тағйироти зеринро захира кунед.",
        "undo-failure": "Ба иллати бархӯрдани вироишҳои дар миён омада, ин вироишро ботил наметавон кард.",
        "undo-summary": "Ботили нусхаи $1 аз тарафи [[Special:Contributions/$2|$2]] ([[User talk:$2|Баҳс]])",
        "group-bot-member": "{{GENDER:$1|бот}}",
        "group-sysop-member": "{{GENDER:$1|мудир}}",
        "group-bureaucrat-member": "{{GENDER:$1|девонсолор}}",
-       "group-suppress-member": "{{GENDER:$1|саҳв}}",
+       "group-suppress-member": "{{GENDER:$1|пинҳон}}",
        "grouppage-user": "{{ns:project}}:Корбарон",
        "grouppage-autoconfirmed": "{{ns:project}}:Корбарони таъйидшуда",
        "grouppage-bot": "{{ns:project}}:Роботҳо",
        "grouppage-sysop": "{{ns:project}}:Мудирон",
+       "grouppage-interface-admin": "{{ns:project}}:Мудирони интерфейс",
        "grouppage-bureaucrat": "{{ns:project}}:Девонсолорҳо",
-       "grouppage-suppress": "{{ns:project}}:Ð\9dазоÑ\80аÑ\82",
+       "grouppage-suppress": "{{ns:project}}:Ð\9fинҳонкÑ\83нанда",
        "right-read": "Хондани саҳифаҳо",
-       "right-edit": "Вироиши саҳифаҳо",
+       "right-edit": "Вироиш намудани саҳифаҳо",
        "right-createpage": "Эҷоди саҳифаҳо (кадоме ки саҳифаҳои баҳс нестанд)",
        "right-createtalk": "Эҷоди саҳифаҳои баҳс",
        "right-createaccount": "Эҷоди ҳисобҳои корбарии ҷадид",
        "grant-group-email": "фиристодани имайл",
        "grant-createaccount": "Эҷоди ҳисобҳо",
        "grant-createeditmovepage": "Эҷод, кучониш ва вироиши саҳифаҳо",
+       "grant-editmywatchlist": "Вироиши феҳристи хоми пайгириҳо",
        "grant-rollback": "Вогардонидани тағйироти саҳифа",
        "grant-basic": "Ҳуқуқҳот асосӣ",
+       "grant-viewmywatchlist": "Дидани феҳристи пайгириҳои шумо",
        "newuserlogpage": "Гузориши эҷоди корбар",
        "newuserlogpagetext": "Ин гузориш аз номҳои корбарии тозасохташуда аст.",
        "rightslog": "Гузориши ихтиёроти корбар",
        "action-undelete": "барқарор кардани ин саҳифа",
        "action-rollback": "вогардонии зуди тағйирот",
        "action-sendemail": "фиристодани пайёмҳо",
+       "action-editmywatchlist": "вироиши феҳристи пайгириҳои шумо",
+       "action-viewmywatchlist": "дидани феҳристи пайгириҳои шумо",
+       "action-editinterface": "вироиши интерфейси корбар",
        "action-override-export-depth": "Нишон додан",
        "nchanges": "$1 {{PLURAL:$1|тағйир|тағйирот}}",
        "enhancedrc-history": "таърих",
        "recentchanges-label-unpatrolled": "Ин вироиш ҳанӯз гаштзанӣ нашудааст",
        "recentchanges-label-plusminus": "Ҳаҷми саҳифа ба андозаи ин миқдор байт тағйир ёфтааст",
        "recentchanges-legend-heading": "Ихтисораҳо:",
+       "recentchanges-legend-newpage": "{{int:recentchanges-label-newpage}} (ҳамчунон [[Special:NewPages|феҳристи саҳифаҳои навро бигед]])",
        "recentchanges-submit": "Нишон додан",
        "rcfilters-tag-remove": "'$1'ро ҳазф намудан",
        "rcfilters-legend-heading": "<strong>Феҳристи ихтисорот:</strong>",
        "rcfilters-savedqueries-remove": "Ҳазф намудан",
        "rcfilters-savedqueries-new-name-label": "Ном",
        "rcfilters-savedqueries-cancel-label": "Лағв",
+       "rcfilters-filterlist-title": "Филтрҳо",
+       "rcfilters-filterlist-whatsthis": "Чӣ тавр кор мекунад?",
        "rcfilters-filter-editsbyself-label": "Тағйироти шумо",
        "rcfilters-filter-editsbyself-description": "Вироишоти шумо.",
        "rcfilters-filter-editsbyother-label": "Тағйироти дигарон",
        "rcfilters-filtergroup-watchlist": "Феҳристи пайгириҳо",
        "rcfilters-filter-watchlist-watched-label": "Дар феҳристи пайгириҳо",
        "rcfilters-filter-watchlist-watched-description": "Тағйири саҳифаҳо дар феҳристи пайгириҳо.",
+       "rcfilters-filter-watchlist-notwatched-label": "Дар феҳристи пайгириҳо не",
+       "rcfilters-filter-watchlistactivity-seen-label": "Дидани тағйирот",
        "rcfilters-filtergroup-changetype": "Намуди тағйирот",
        "rcfilters-filter-pageedits-label": "Вироишоти саҳифа",
+       "rcfilters-filter-newpages-label": "Эҷоди саҳифаҳо",
+       "rcfilters-filter-newpages-description": "Вироише, ки саҳифаи наве сохт.",
+       "rcfilters-filter-categorization-label": "Тағйироти гурӯҳҳо",
+       "rcfilters-filtergroup-lastrevision": "Нусхаҳои охирин",
+       "rcfilters-filter-lastrevision-label": "Нусхаи охирин",
+       "rcfilters-filter-lastrevision-description": "Танҳо тағйироти охирин дар саҳифа.",
+       "rcfilters-filter-previousrevision-label": "Нусхаи охирин не",
        "rcnotefrom": "Дар зер тағйиротҳои охирин аз <b>$2</b> (то <b>$1</b> нишон дода шудааст).",
        "rclistfrom": "Нишон додани тағйиротҳои нав сар карда аз $3 $2",
        "rcshowhideminor": "$1 вироишҳои хурд",
        "recentchangeslinked-summary": "Ин феҳристи тағйироти охир барои саҳифаҳои пайваста аз саҳифаи мушаххасшуда мебошад (ё ба аъзоёни гурӯҳи мушаххасшуда).\nСаҳифаҳои дар [[Special:Watchlist|феҳристи назароти шумо]] буда  '''пурранг''' ҳастанд.",
        "recentchangeslinked-page": "Номи саҳифа:",
        "recentchangeslinked-to": "Тағйироти саҳифаҳое, ки ба саҳифаи мавриди назар пайванддоштаро намоиш диҳед",
+       "recentchanges-page-added-to-category": "[[:$1]] ба гурӯҳ илова шуд",
+       "recentchanges-page-removed-from-category": "[[:$1]] аз гурӯҳ ҳазф шуд",
        "upload": "Фиристодани парванда",
        "uploadbtn": "Фиристодани файл",
        "reuploaddesc": "Боргузориро лағв кунед ва ба форми боргузорӣ баргардед.",
        "listfiles_size": "Андоза(ҳаҷм)",
        "listfiles_description": "Тавсифот",
        "listfiles_count": "Нусхаҳо",
+       "listfiles-latestversion": "Нусхаи феълӣ",
        "listfiles-latestversion-yes": "Бале",
        "listfiles-latestversion-no": "На",
        "file-anchor-link": "файл",
        "filehist-filesize": "Андозаи парванда",
        "filehist-comment": "Тавзеҳ",
        "imagelinks": "Истифодаи парванда",
-       "linkstoimage": "{{PLURAL:$1|СаҳиÑ\84аҳои|$1 Ð¡Ð°Ò³Ð¸Ñ\84аи}} Ð·ÐµÑ\80ин Ð±Ð° Ð¸Ð½ Ð°ÐºÑ\81 Ð¿Ð°Ð¹Ð²Ð°Ð½Ð´анд:",
+       "linkstoimage": "{{PLURAL:$1|СаҳиÑ\84аҳои|$1 Ð¡Ð°Ò³Ð¸Ñ\84аи}} Ð¸Ð½ Ð°ÐºÑ\81Ñ\80о Ð¸Ñ\81Ñ\82иÑ\84ода Ð¼ÐµÐ±Ð°Ñ\80анд:",
        "nolinkstoimage": "Ҳеҷ саҳифае ба ин акс пайванд надорад.",
        "sharedupload": "Ин парванда аз $1 мебошад ва шояд аз тарафи дигар лоиҳаҳо истифода шавад.",
        "sharedupload-desc-here": "Ин файл аз $1 ва дар дигар лоиҳаҳо метавонад истифода шавад. Тафсилоти ин файл [$2 саҳифаи тафсилоти файл] дар зер нишон дода шудааст.",
        "uploadnewversion-linktext": "Бор кардани нусхаи ҷадидӣ ин парванда",
        "shared-repo-from": "аз $1",
+       "shared-repo-name-wikimediacommons": "Викианбор",
        "filerevert": "Вогардонии $1",
        "filerevert-legend": "Вогардонии парванда",
        "filerevert-intro": "Шумо дар ҳоли вогардонии '''[[Media:$1|$1]]''' ба [$4 нусхаи аз $3, $2] ҳастед.",
        "statistics-edits": "Шумораи вироишҳо аз замони эҷоди ин {{SITENAME}}",
        "statistics-users": "Корбарони сабтиномшуда",
        "statistics-users-active": "Корбарони фаъол",
+       "pageswithprop-prop": "Номи хусусият:",
        "pageswithprop-submit": "Бирав",
        "doubleredirects": "Тағйири масирҳои дутоӣ",
        "doubleredirectstext": "Ҳар сатр дар бар дорандаи пайвандҳое ба тағйири масири аввал ва дувум ва ҳамчунин хати нахуст тагйири масири дувум аст. Маъмулан саҳифаи мақсади воқеъӣ, ки нахустин тағйири масир бояд ба он бошад ба ин гуна мушаххас мешавад.",
        "fewestrevisions": "Саҳифаҳое, ки шумораи ками нусхаҳо доранд",
        "nbytes": "$1 {{PLURAL:$1|байт}}",
        "ncategories": "$1 {{PLURAL:$1|гурӯҳ|гурӯҳҳо}}",
+       "ninterwikis": "$1 {{PLURAL:$1|миёнавики}}",
        "nlinks": "$1 {{PLURAL:$1|пайванд|пайвандҳо}}",
        "nmembers": "$1 {{PLURAL:$1|узв}}",
        "nrevisions": "{{PLURAL:$1|вироиш|вироиши}} $1",
+       "nimagelinks": "Дар $1 {{PLURAL:$1|саҳифа}} истифода мешавад",
        "specialpage-empty": "Барои ин ҳисобот натиҷае вуҷуд надорад.",
        "lonelypages": "Саҳифаҳои ятим",
        "lonelypagestext": "Ба саҳифаҳои зерин дар дигар саҳифаи {{SITENAME}} пайванд дода нашудааст.",
        "mostimages": "Аксҳое ки бештар аз ҳама бо онҳо пайванд шудааст",
        "mostrevisions": "Саҳифахое, ки аз ҳама бештар вироиш шудаанд",
        "prefixindex": "Ҳамаи саҳифаҳо бо пешванд",
+       "prefixindex-namespace": "Ҳамаи саҳифаҳо бо пешванди (фазои $1)",
        "prefixindex-submit": "Намоиш",
        "shortpages": "Саҳифаҳои кӯтоҳ",
        "longpages": "Саҳифаҳои калон",
        "pager-older-n": "{{PLURAL:$1|1-тои кӯҳнатар|$1-тои кӯҳнатар}}",
        "suppress": "Назорат",
        "apisandbox": "Регдони API",
+       "apisandbox-reset": "Тоза кардан",
+       "apisandbox-retry": "Дубора саъй кунед",
+       "apisandbox-examples": "Намунаҳо",
+       "apisandbox-dynamic-parameters-add-label": "Иловаи тарҷиҳ:",
+       "apisandbox-dynamic-parameters-add-placeholder": "Номи тарҷиҳ",
        "apisandbox-add-multi": "Илова кардан",
        "apisandbox-results": "Натиҷаҳо",
        "booksources": "Манбаҳои китобҳо",
        "listgrouprights-rights": "Дастрасиҳо",
        "listgrouprights-helppage": "Help:Дастрасиҳои гурӯҳӣ",
        "listgrouprights-members": "(феҳристи аъзоён)",
+       "listgrouprights-removegroup-all": "Тоза намудани тамоми гурӯҳҳо",
        "mailnologin": "Нишонае аз фиристанда вуҷуд надорад",
        "mailnologintext": "Барои фиристодани почтаи электронӣ барои корбарони дигар бояд [[Special:UserLogin|ба систем ворид шавед]] ва нишонаи почтаи электронии мӯътабар дар [[Special:Preferences|тарҷиҳоти]] худ дошта бошед.",
        "emailuser": "Фиристодани email ба ин корбар",
-       "emailuser-title-target": "Ð\9dавиÑ\88Ñ\82ани Ð¼Ð°ÐºÑ\82Ñ\83б Ð±Ð° email-и Ð¸Ð½ ÐºÐ¾Ñ\80баÑ\80",
+       "emailuser-title-target": "Ð\91а Ð¸Ð½ {{GENDER:$1|коÑ\80баÑ\80}} Ð¿Ð°Ñ\91м Ð½Ð°Ð²Ð¸Ñ\88Ñ\82ан",
        "emailuser-title-notarget": "Фиристодани пайём ба корбар",
        "emailpagetext": "Агар ин корбар нишонаи почтаи электронии мӯътабаре дар тарҷиҳоти ворид карда бошад, форми зерин як пайғоме мефиристад.\nНишонаи почтаи электроние, ки шумо дар тарҷиҳоти корбариатон ворид кардаед, дар нишони фиристандаи нома \"From\" хоҳад омад, то ки гиранда тавонад ба шумо посух диҳад.",
        "defemailsubject": "Пайёми {{SITENAME}} аз корбар \"$1\"",
        "emailccsubject": "Нусхаи номаи шумо ба $1: $2",
        "emailsent": "Почтаи электронӣ фиристода шуд",
        "emailsenttext": "Номаи почтаи электронии шумо фиристода шуд.",
+       "usermessage-template": "MediaWiki:UserMessage",
        "watchlist": "Феҳристи пайгирӣ",
        "mywatchlist": "Феҳристи пайгириҳо",
        "watchlistfor2": "Барои $1 $2",
        "wlheader-showupdated": "Саҳифаҳое, ки пас аз охирин сар заданатон ба онҳо тағйир кардаанд '''пурранг''' нишон дода шудаанд",
        "wlnote": "Дар зер {{PLURAL:$1|охирин тағйир|'''$1''' охирин тағйирот}} дар $2 соати охир {{PLURAL:омадааст|омадаанд}}.",
        "wlshowlast": "Намоиши охирин $1 соат $2 рӯзҳо",
+       "watchlist-hide": "Пинҳон",
+       "watchlist-submit": "Намоиш",
+       "wlshowhidebots": "ботҳо",
+       "wlshowhideliu": "корбарони сабтиномшуда",
+       "wlshowhideanons": "корбари вориднашуда",
+       "wlshowhidepatr": "вироишҳои гаштӣ",
+       "wlshowhidemine": "вироишоти ман",
+       "wlshowhidecategorization": "гурӯҳбандии саҳифаҳо",
        "watchlist-options": "Ихтиёроти феҳристи пайгириҳо",
        "watching": "Пайгири...",
        "unwatching": "Тавқифи пайгири...",
        "delete-toobig": "Ин саҳифа таърихчаи бузурге дорад, ки шомили беш аз $1 вироиш аст. Ҳазфи ин гуна саҳифаҳо барои пешгири аз шикастани тасодуфӣ дар {{SITENAME}} маҳдуд шудааст.",
        "delete-warning-toobig": "Ин саҳифа таърихи бузурге дорад, ки шомили беш аз $1 вироиш аст. Ҳазфи ин саҳифа метавонад ихтилол ба амалгари пойгоҳи додаи {{SITENAME}} бишавад; лутфан бо эҳтиёт иқдом кунед.",
        "rollback": "Вогардонидани вироишот",
+       "rollback-confirmation-no": "Тамом кардан",
        "rollbacklink": "вогардонидан",
        "rollbacklinkcount": "вогардонидани $1 {{PLURAL:$1|вироиш}}",
        "rollbacklinkcount-morethan": "вогардонидани беш аз $1 {{PLURAL:$1|вироиш}}",
        "revertpage": "Вироиши [[Special:Contributions/$2|$2]] ([[User talk:$2|Баҳс]]) вогардонида шуд ба охирин тағйире, ки [[User:$1|$1]] анҷом дода буд",
        "rollback-success": "Вироишҳои $1 вогардонӣ шуд; саҳифа ба вироиши $2 баргардонида шуд.",
        "sessionfailure": "Ба назар мерасад, мушкилие дар мавриди нишасти корбарии шумо вуҷуд дорад; амали дархостшуда ба унвони иқдоми пешгирона дар баробари рабуда шудани иттилооти нишасти корбарӣ, лағв шуд. Лутфан тугмаи \"бозгашт\"-ро дар мурургари худ пахш кунед ва саҳифае, ки аз он инҷо расидаед муҷаддадан фарохонӣ кунед, сипас муҷаддадан боз саъй кунед.",
+       "changecontentmodel-title-label": "Унвони саҳифа",
+       "changecontentmodel-reason-label": "Сабаб:",
+       "changecontentmodel-submit": "Тағйир",
+       "logentry-contentmodel-change-revertlink": "вогардонӣ",
+       "logentry-contentmodel-change-revert": "вогардонӣ",
        "protectlogpage": "Гузориши муҳофизат",
        "protectlogtext": "Дар зер феҳристи қуфл карданҳо ва аз қуфл озод шуданҳо омада аст. Барои иттилооти бештар ба [[Special:ProtectedPages|феҳристи саҳифаҳои муҳофизатшуда]] нигаред.",
        "protectedarticle": "\"[[$1]]\" муҳофизат шуд",
        "modifiedarticleprotection": "сатҳи муҳофизати саҳифаи \"[[$1]]\" тағйир дода шуд",
        "unprotectedarticle": "аз муҳофизат озод шуда \"[[$1]]\"",
+       "protectedarticle-comment": "{{GENDER:$2|Ҳизф кард}}: \"[[$1]]\"",
        "protect-title": "Дар ҳоли гузоштани сатҳи муҳофизат барои \"$1\"",
        "prot_1movedto2": "[[$1]] ба [[$2]] кӯчонида шудааст",
        "protect-legend": "Тасдиқи муҳофизат",
        "whatlinkshere-hidelinks": "$1 пайвандҳо",
        "whatlinkshere-hideimages": "$1 пайвандҳои парванда",
        "whatlinkshere-filters": "Филтрҳо",
+       "whatlinkshere-submit": "Рав",
        "block": "Бастани корбар",
        "unblock": "Боз кардани корбар",
        "blockip": "Бастани корбар",
        "autoblocklist-submit": "Ҷустуҷӯ",
        "ipblocklist": "Корбарони басташуда",
        "ipblocklist-legend": "Ҷустуҷӯи корбари баста шуда",
+       "blocklist-type-opt-all": "Ҳама",
+       "blocklist-type-opt-partial": "Ҷузъӣ",
        "blocklist-target": "Ҳадаф",
        "blocklist-expiry": "Замони саромадан",
        "blocklist-reason": "Сабаб",
        "noautoblockblock": "бастани худкор ғайрифаъол аст",
        "createaccountblock": "имкони эҷоди ҳисоб баста шудааст",
        "emailblock": "почтаи электронӣ баста шудааст",
+       "blocklist-editing-page": "саҳифаҳо",
        "ipblocklist-empty": "Феҳристи басташуданҳо холӣ аст.",
        "ipblocklist-no-results": "Дастрасии ҳисоби корбарӣ ё нишонаи интернетии мавриди назар қатъ нест.",
        "blocklink": "бастан",
index 423c7a8..f091736 100644 (file)
        "tog-numberheadings": "Атамалар автомат рәвештә номерлансын",
        "tog-editondblclick": "Битләргә ике чирттерү белән үзгәртү бите ачылсын",
        "tog-editsectiononrightclick": "Бүлек исеменә тычканның уң чирттермәсе белән төрткәч үзгәртү",
-       "tog-watchcreations": "Мин төзегән битләр һәм йөкләгән файллар күзәтү исемлегемә өстәлсен",
+       "tog-watchcreations": "Мин төзегән битләр һәм төягән файллар күзәтү исемлегемә өстәлсен",
        "tog-watchdefault": "Мин үзгәрткән битләр һәм файллар күзәтү исемлегемә өстәлсен",
        "tog-watchmoves": "Мин күчергән битләр һәм файллар күзәтү исемлегемә өстәлсен",
        "tog-watchdeletion": "Мин бетергән битләр һәм файлларны күзәтү исемлегемгә өстәлсен",
-       "tog-watchuploads": "Минем тарафтан йөкләнелгән файлларны күзәтү исемлегемә кертелсен",
+       "tog-watchuploads": "Мин төягән файлларны күзәтү исемлегемә кертелсен",
        "tog-watchrollback": "Мин үткәрмәгән битләрне күзәтү исемлегемә өстәргә",
        "tog-minordefault": "Барлык төзәтмәләрне килешү буенча кече дип билгеләнсен",
        "tog-previewontop": "Үзгәртү тәрәзәсеннән өстәрәк битне алдан карау өлкәсен күрсәтелсен",
        "and": "&#32;һәм",
        "faq": "ЕБС",
        "actions": "Хәрәкәтләр",
-       "namespaces": "Ð\98Ñ\81емнÓ\99Ñ\80 Ð¼Ó\99йданÑ\8b",
+       "namespaces": "Ð\98Ñ\81емнÓ\99Ñ\80 ÐºÐ¸Ò£Ð»ÐµÐºÐ»Ó\99Ñ\80е",
        "variants": "Вариантлар",
        "navigation-heading": "Навигация",
        "errorpagetitle": "Хата",
        "returnto": "$1 битенә кайту.",
        "tagline": "{{SITENAME}} проектыннан",
-       "help": "ЯÑ\80дÓ\99м",
+       "help": "Ð\91елеÑ\88мÓ\99",
        "help-mediawiki": "MediaWiki турында белешмә",
        "search": "Эзләү",
        "searchbutton": "Эзләү",
        "tool-link-emailuser": "{{GENDER:$1|Кулланучыга}} хат язу",
        "imagepage": "Файл битен карау",
        "mediawikipage": "Хәбәр битен карау",
-       "templatepage": "Үрнәк битен карау",
+       "templatepage": "Калып битен карау",
        "viewhelppage": "Ярдәм битен карау",
        "categorypage": "Төркем битен карау",
        "viewtalkpage": "Бәхәс битен карау",
        "badaccess-groups": "Соралган гамәлне $1 {{PLURAL:$2|1=төркеме|төркемнәренең}} кулланучылары гына башкара ала.",
        "versionrequired": "MediaWikiның $1 версиясе таләп ителә",
        "versionrequiredtext": "Бу бит белән эшләү өчен MediaWikiның $1 версиясе кирәк. [[Special:Version|Кулланылучы программа версиясе турында мәгълүмат битен]] кара.",
-       "ok": "OK",
+       "ok": "Ярар",
        "pagetitle": "$1 — {{SITENAME}}",
        "pagetitle-view-mainpage": "{{SITENAME}}",
        "retrievedfrom": "Чыганагы — \"$1\"",
-       "youhavenewmessages": "Сездә $1 бар ($2).",
+       "youhavenewmessages": "{{PLURAL:$3|Сездә}} $1 бар ($2).",
        "youhavenewmessagesfromusers": "{{PLURAL:$4|Сезгә}} {{PLURAL:$3|$3 кулланучыдан}} $1 килде ($2).",
        "youhavenewmessagesmanyusers": "Сез бик күп кулланучыдан $1 алдыгыз ($2).",
-       "newmessageslinkplural": "{{PLURAL:$1|яңа хәбәр|999=яңа хәбәрләр}}",
+       "newmessageslinkplural": "яңа {{PLURAL:$1|хәбәр|999=хәбәрләр}}",
        "newmessagesdifflinkplural": "соңгы {{PLURAL:$1|үзгәреш|999=үзгәрешләр}}",
        "youhavenewmessagesmulti": "$1 эчендә яңа хат бар",
        "editsection": "үзгәртү",
        "nstab-special": "Махсус бит",
        "nstab-project": "Проект бите",
        "nstab-image": "Файл",
-       "nstab-mediawiki": "Хат",
+       "nstab-mediawiki": "Хәбәр",
        "nstab-template": "Калып",
        "nstab-help": "Ярдәм",
        "nstab-category": "Төркем",
        "mycustomjsprotected": "Сезнең биттә JavaScript үзгәртергә хокукларыгыз юк.",
        "ns-specialprotected": "Махсус битләрне үзгәртеп булмый.",
        "titleprotected": "Бу исем белән бит ясау [[User:$1|$1]] тарафыннан тыелган.\nУл күрсәткән сәбәп: <em>$2</em>.",
+       "invalidtitle": "Ярамаган атама",
+       "invalidtitle-knownnamespace": "«$2» исемнәр киңлеге һәм «$3» тексты белән ярамаган атама",
+       "invalidtitle-unknownnamespace": "Билгесез $1 санлы исемнәр киңлеге һәм «$2» тексты белән ярамаган атама",
        "exception-nologin": "Сез хисап язмагызга кермәгәнсез",
        "virus-badscanner": "Көйләү хатасы. Билгесез вируслар сканеры: ''$1''",
        "virus-scanfailed": "сканерлау хатасы ($1 коды)",
        "virus-unknownscanner": "билгесез антивирус:",
        "logouttext": "<strong>Сез хисап язмагыздан чыктыгыз.</strong>\n\nКайбер битләр Сез кергән кебек күрсәтелергә мөмкин. Моны бетерү өчен браузер кэшын чистартыгыз.",
+       "logout-failed": "Хәзер үк чыгып булмый: $1",
        "cannotlogoutnow-title": "Хәзер үк чыгып булмый",
+       "cannotlogoutnow-text": "$1 куллану вакытында чыгып булмый.",
        "welcomeuser": "Рәхим итегез, $1!",
        "yourname": "Кулланучы исеме:",
        "userlogin-yourname": "Кулланучы исеме",
        "wrongpassword": "Серсүз яисә кулланучы исеме дөрес түгел. Яңадан җыеп карагыз.",
        "wrongpasswordempty": "Серсүз юлы буш булырга тиеш түгел.",
        "passwordtooshort": "Сезсүз кимендә $1 {{PLURAL:$1|символдан}} торырга тиеш.",
+       "passwordtoolong": "Серсүз {{PLURAL:$1|$1 билгедән}} озынрак була алмый.",
        "password-name-match": "Кертелгән серсүз кулланучы исеменнән аерылырга тиеш.",
        "password-login-forbidden": "Бу кулланучы исемен һәм серсүзне куллану тыелган",
        "mailmypassword": "Серсүзне бетерү",
        "watchthis": "Бу битне күзәтү",
        "savearticle": "Битне саклау",
        "savechanges": "Үзгәрешләрне саклау",
-       "publishpage": "Ð\91иÑ\82 Ñ\8fÑ\81аÑ\83",
-       "publishchanges": "Битне бастыру",
+       "publishpage": "Ð\91иÑ\82 Ñ\82өзү",
+       "publishchanges": "Үзгәрешләр саклау",
        "savearticle-start": "Битне саклау...",
        "savechanges-start": "Үзгәрешләрне саклау…",
        "publishpage-start": "Битне бастыру...",
        "publishchanges-start": "Төзәтмәләрне бастыру...",
        "preview": "Алдан карау",
        "showpreview": "Алдан карау",
-       "showdiff": "Кертелгән үзгәрешләр",
+       "showdiff": "Үзгәрешләр күрсәтү",
        "anoneditwarning": "<strong>Игътибар!</strong> Сез сайтта теркәлмәдегез. Әгәрдә сез нинди дә булсә төзәтмәләр яисә үзгәртүләр кертсәгез, сезнең IP-адрес башка кулланучыларга да билгеле булачак. Сайтка <strong>[$1 керсәгез]</strong> яисә <strong>[$2 кулланучы язмасын төзесәгез]</strong>, сез керткән үзгәртүләр сезнең кулланучы язмагызга бәйләнгән булачак, шулай ук башка мөмкинлекләр дә туачак.",
        "anonpreviewwarning": "''Сез системада теркәлмәдегез.Сезнең тарафтан эшләнгән барлык үзгәртүләр дә сезнең IP-юлламагызны саклауга китерә.''",
        "missingsummary": "<strong>Искәртмә.</strong> Сез төзәтмәнең кыскача аңлатмасы бирмәдегез. \nӘгәр «$1» төймәсенә кабат бассагыз, төзәтмәгез аңлатмасыз сакланачак.",
        "yourtext": "Сезнең текст",
        "storedversion": "Сакланган юрама",
        "editingold": "'''Кисәтү: Сез битнең искергән юрамасын үзгәртәсез.'''\nСаклау төймәсенә баскан очракта яңа юрамалардагы үзгәртүләр югалачак.",
-       "yourdiff": "Аермалар",
+       "yourdiff": "Аермалыклар",
        "copyrightwarning": "Бөтен өстәмәләр һәм үзгәртүләр $2 (карагыз: $1) лицензиясе шартларында башкарыла дип санала.\nӘгәр аларның ирекле таратылуын һәм үзгәртелүен теләмәсәгез, монда өстәмәвегез сорала.<br />\nСез өстәмәләрнең авторы булырга яисә мәгълүматның ирекле чыганаклардан алынуын күрсәтергә тиеш.<br />\n'''МАХСУС РӨХСӘТТӘН БАШКА АВТОРЛЫК ХОКУКЫ БУЕНЧА САКЛАНУЧЫ МӘГЪЛҮМАТЛАР УРНАШТЫРМАГЫЗ!'''",
        "copyrightwarning2": "Сезнең үзгәртүләр башка кулланучылар тарафыннан үзгәртелә яисә бетерелә ала.\nӘгәр аларның үзгәртелүен теләмәсәгез, монда өстәмәвегез сорала.<br />\nСез өстәмәләрнең авторы булырга яисә мәгълүматның ирекле чыганаклардан алынуын күрсәтергә тиеш (карагыз: $1).\n'''МАХСУС РӨХСӘТТӘН БАШКА АВТОРЛЫК ХОКУКЫ БУЕНЧА САКЛАНУЧЫ МӘГЪЛҮМАТЛАР УРНАШТЫРМАГЫЗ!'''",
        "longpageerror": "<strong>ХАТА: сакланучы текст зурлыгы - $1 {{PLURAL:$1|килобайт}}, бу $2 {{PLURAL:$2|килобайт}} чигеннән күбрәк. Бит саклана алмый.</strong>",
        "mergelog": "Берләштерүләр көндәлеге",
        "revertmerge": "Бүлү",
        "history-title": "$1 битенең үзгәртү тарихы",
-       "difference-title": "«$1» битенең юрамалары арасындагы аермалар",
+       "difference-title": "«$1» битенең юрамалары арасында аерма",
+       "difference-title-multipage": "«$1» һәм «$2» битләре арасында аерма",
+       "difference-multipage": "(Битләр арасында аерма)",
        "lineno": "$1 юл:",
        "compareselectedversions": "Сайланган юрамаларны чагыштыру",
        "showhideselectedversions": "Сайланган юрамаларны күрсәтү/яшерү",
        "timezonelegend": "Сәгать поясы:",
        "localtime": "Җирле вакыт",
        "timezoneuseserverdefault": "Сервернең көйләнмәләре кулланылсын ($1)",
-       "timezoneuseoffset": "Башка (күчерелүне күрсәтегез)",
+       "timezoneuseoffset": "Башка (түбәндә күчерелү күрсәтегез)",
+       "timezone-useoffset-placeholder": "Мәсәлән: «-07:00» яки «01:00»",
        "servertime": "Серверның вакыты:",
        "guesstimezone": "Браузердан тутыру",
        "timezoneregion-africa": "Африка",
        "email-allow-new-users-label": "Яңа кулланучылардан да электрон хатлар алуны рөхсәт итү",
        "email-blacklist-label": "Әлеге кулланучыларга минем электрон почтага хат җибәрүне тыярга:",
        "prefs-searchoptions": "Эзләү",
-       "prefs-namespaces": "Ð\98Ñ\81емнÓ\99Ñ\80 Ð¼Ó\99йданÑ\8b",
+       "prefs-namespaces": "Ð\98Ñ\81емнÓ\99Ñ\80 ÐºÐ¸Ò£Ð»ÐµÐºÐ»Ó\99Ñ\80е",
        "default": "килешү буенча",
        "prefs-files": "Файллар",
        "prefs-custom-css": "Шәхси CSS",
        "prefs-advancedwatchlist": "Киңәйтелгән көйләүләр",
        "prefs-displayrc": "Күрсәтү көйләнмәләре",
        "prefs-displaywatchlist": "Күрсәтү көйләнмәләре",
+       "prefs-pageswatchlist": "Күзәтелгән битләр",
        "prefs-tokenwatchlist": "Токен",
        "prefs-diffs": "Юрамалар аермасы",
        "userrights": "Кулланучы хокуклары",
        "right-move-categorypages": "төркемдәге битләрне күчерү",
        "right-movefile": "файлларның исемен алмаштыру",
        "right-suppressredirect": "Элекке исемнән юнәлтү ясамыйча исемне алмаштыру",
-       "right-upload": "файлларны йөкләү",
+       "right-upload": "Файллар төяү",
        "right-reupload": "Булган файллар өстеннән язу",
        "right-writeapi": "Язма өчен API куллану",
        "right-delete": "битләрне бетерү",
        "right-suppressionlog": "Шәхси журналларны карау",
        "right-unblockself": "Үзеңне блоктан алу",
        "right-editinterface": "Кулланучы интерфейсын үзгәртү",
+       "right-userrights": "Барлык кулланучы хокукларын үзгәртү",
+       "right-userrights-interwiki": "Башка википроектларда кулланучы хокукларын үзгәртү",
        "grant-group-email": "Хатлар җибәрү",
-       "grant-uploadfile": "Яңа файллар йөкләү",
+       "grant-uploadfile": "Яңа файллар төяү",
        "grant-basic": "Төп хокуклар",
        "newuserlogpage": "Кулланучыларны теркәү көндәлеге",
        "newuserlogpagetext": "Яңа теркәлгән кулланучылар исемлеге",
        "action-createpage": "бу битне төзү",
        "action-createtalk": "бу бәхәс битен төзү",
        "action-createaccount": "бу кулланучы язмасын ясау",
-       "action-move": "бу битне күчерергә",
+       "action-history": "бу битнең үзгәртү тарихын карау",
+       "action-minoredit": "бу төзәтмәне кече дип билгеләү",
+       "action-move": "бу битне күчерү",
+       "action-move-subpages": "бу битне һәм аның асбитләрне күчерү",
+       "action-move-rootuserpages": "кулланучыдагы төп битләрне күчерү",
+       "action-move-categorypages": "төркемдәге битләрне күчерү",
+       "action-movefile": "бу файлны күчерү",
+       "action-upload": "бу файлны төяү",
+       "action-reupload": "булган файл өстеннән язу",
        "action-sendemail": "электрон хат җибәрү",
+       "action-editmyoptions": "көйләнмәләрегезне үзгәртү",
+       "action-editmywatchlist": "күзәтү исемлегегезне үзгәртү",
+       "action-viewmywatchlist": "күзәтү исемлегегезне карау",
+       "action-viewmyprivateinfo": "шәхси мәгълүматыгызны карау",
+       "action-editmyprivateinfo": "шәхси мәгълүматыгызны үзгәртү",
        "nchanges": "$1 {{PLURAL:$1|үзгәртү}}",
        "enhancedrc-history": "тарих",
        "recentchanges": "Соңгы үзгәрешләр",
        "rcfilters-days-show-hours": "$1 {{PLURAL:$1|сәгать}}",
        "rcfilters-quickfilters": "Сакланган фильтрлар",
        "rcfilters-quickfilters-placeholder-title": "Әлегә сакланылган фильтрлар юк",
-       "rcfilters-savedqueries-rename": "Исемен үзгәртергә",
+       "rcfilters-savedqueries-defaultlabel": "Сакланган сөзгечләр",
+       "rcfilters-savedqueries-rename": "Исемен үзгәртү",
        "rcfilters-savedqueries-remove": "Бетерү",
        "rcfilters-savedqueries-new-name-label": "Исем",
        "rcfilters-savedqueries-apply-label": "Фильтр кую",
        "rcfilters-savedqueries-cancel-label": "Баш тарту",
        "rcfilters-savedqueries-add-new-title": "Хәзерге фильтр көйләнмәләрен саклау",
-       "rcfilters-clear-all-filters": "Барлык филтерләрне чистарту",
+       "rcfilters-clear-all-filters": "Барлык сөзгечләрне бушату",
+       "rcfilters-show-new-changes": "$1 башлап яңа үзгәрешләрне карау",
        "rcfilters-search-placeholder": "Үзгәрешләрне сөзү (меню кулланыгыз яки сөзгеч аты буенча эзлигез)",
        "rcfilters-invalid-filter": "Яраксыз фильтр",
-       "rcfilters-filterlist-title": "Фильтрлар",
+       "rcfilters-filterlist-title": "Сөзгечләр",
+       "rcfilters-filterlist-whatsthis": "Бу ничек эшли?",
        "rcfilters-filterlist-feedbacklink": "Әлеге фильтрлау кораллары турында турында фикер калдырыгыз",
        "rcfilters-highlightmenu-title": "Төсен сайлагыз",
        "rcfilters-highlightmenu-help": "Үзлекләрен аеру өчен аның төсен сайлагыз",
+       "rcfilters-filterlist-noresults": "Сөзгечләр табылмаган",
        "rcfilters-filtergroup-authorship": "Үзгәртүләрнең авторлыгы",
-       "rcfilters-filter-editsbyself-label": "Сезнең үзгәртүләр",
+       "rcfilters-filter-editsbyself-label": "Сез ясаган үзгәрешләр",
        "rcfilters-filter-editsbyself-description": "Сезнең кертемегез.",
-       "rcfilters-filter-editsbyother-label": "Башка кулланучыларның үзгәртүләре",
+       "rcfilters-filter-editsbyother-label": "Башка кулланучылар ясаган үзгәрешләр",
+       "rcfilters-filter-editsbyother-description": "Сездән башка ясаган үзгәрешләр.",
        "rcfilters-filter-user-experience-level-registered-label": "Теркәлүчеләр",
        "rcfilters-filter-user-experience-level-registered-description": "Теркәлгән мөхәррирләр.",
        "rcfilters-filter-user-experience-level-unregistered-label": "Теркәлмәгәннәр",
        "rcfilters-filter-user-experience-level-experienced-description": "Төзәтүләре 500 дән күбрәк һәм актив эш көннәре 30 дан артык теркәлгән мөхәррирләр",
        "rcfilters-filtergroup-automated": "Автоматлаштырылган кертем",
        "rcfilters-filter-bots-label": "Бот",
-       "rcfilters-filter-bots-description": "Ð\90вÑ\82омаÑ\82лаÑ\88Ñ\82Ñ\8bÑ\80Ñ\8bлган ÐºÐ¾Ñ\80аллаÑ\80 Ñ\8fÑ\80дÓ\99мендÓ\99 Ñ\8fÑ\81алган Ò¯Ð·Ð³Ó\99Ñ\80Ñ\82Ò¯ләр.",
+       "rcfilters-filter-bots-description": "Ð\90вÑ\82окоÑ\80аллаÑ\80 Ñ\8fÑ\80дÓ\99мендÓ\99 Ñ\8fÑ\81алган Ñ\82өзÓ\99Ñ\82мÓ\99ләр.",
        "rcfilters-filter-humans-label": "Кеше (бот түгел)",
-       "rcfilters-filter-humans-description": "Кешеләр ясаган үзгәртүләр.",
+       "rcfilters-filter-humans-description": "Кешеләр ясаган төзәтмәләр.",
        "rcfilters-filtergroup-reviewstatus": "Тикшерү статусы",
+       "rcfilters-filter-reviewstatus-unpatrolled-label": "Тикшерелмәгән",
+       "rcfilters-filter-reviewstatus-manual-label": "Кулдан тикшерелгән",
        "rcfilters-filter-reviewstatus-auto-label": "Автотикшеренүчеләр",
        "rcfilters-filtergroup-significance": "Мәгънәсе",
-       "rcfilters-filter-minor-label": "Кече үзгәртүләр",
-       "rcfilters-filter-minor-description": "«Кече үзгәртү» дип тамгаланган үзгәртүләр",
-       "rcfilters-filter-major-label": "Ð\93ади Ò¯Ð·Ð³Ó\99Ñ\80Ñ\82Ò¯ләр",
-       "rcfilters-filter-major-description": "«Кече» дип тамгаланмаган үзгәртүләр",
-       "rcfilters-filtergroup-watchlist": "Ð\9aүзÓ\99Ñ\82Ò¯ Ð¸Ñ\81емлегегездәге битләр",
+       "rcfilters-filter-minor-label": "Кече төзәтмәләр",
+       "rcfilters-filter-minor-description": "Автордан «кече» дип билгеләнгән төзәтмәләр.",
+       "rcfilters-filter-major-label": "Ð\9aеÑ\87е Ð±Ñ\83лмаган Ñ\82өзÓ\99Ñ\82мÓ\99ләр",
+       "rcfilters-filter-major-description": "«Кече» дип билгеләнмәгән төзәтмәләр.",
+       "rcfilters-filtergroup-watchlist": "Ð\9aүзÓ\99Ñ\82Ò¯ Ð¸Ñ\81емлегендәге битләр",
        "rcfilters-filter-watchlist-watched-label": "Күзәтү исемлегендә",
        "rcfilters-filtergroup-changetype": "Үзгәртү төре",
-       "rcfilters-filter-pageedits-label": "Бит үзгәртүләре",
+       "rcfilters-filter-pageedits-label": "Бит төзәтмәләре",
        "rcfilters-filter-newpages-label": "Бит төзүләре",
-       "rcfilters-filter-categorization-label": "Төркем үзгәртүләре",
+       "rcfilters-filter-categorization-label": "Төркем үзгәрешләре",
        "rcfilters-filter-categorization-description": "Төркемнәргә кушылган яки төркемнәрдән алып ташланган битләр турында язмалар.",
        "rcfilters-filter-logactions-label": "Беркетмәләнүче гамәлләр",
        "rcfilters-filter-logactions-description": "Административ гамәлләр, хисап язмасын төзүләр, битне бетерүләр, файл йөкләүләр...",
+       "rcfilters-filtergroup-lastrevision": "Соңгы юрамалар",
        "rcfilters-filter-lastrevision-label": "Соңгы юрама",
        "rcfilters-filter-lastrevision-description": "Битнең соңгы гына үзгәртүе.",
        "rcfilters-filter-previousrevision-label": "Соңгы булмаган юрама",
-       "rcfilters-filter-previousrevision-description": "«Соңгы юрама» булмаган барлык үзгәртүләр.",
-       "rcfilters-filter-excluded": "РөÑ\85Ñ\81Ó\99Ñ\82 Ñ\8eк",
+       "rcfilters-filter-previousrevision-description": "«Соңгы юрама» булмаган барлык үзгәрешләр.",
+       "rcfilters-filter-excluded": "ЧÑ\8bгаÑ\80Ñ\8bп Ñ\82аÑ\88ланган",
        "rcfilters-tag-prefix-namespace-inverted": "<strong>:not</strong> $1",
-       "rcfilters-view-tags": "Тәгләнгән үзгәртүләр",
+       "rcfilters-view-tags": "Тамгаланган төзәтмәләр",
        "rcfilters-liveupdates-button": "Автоматик яңарту",
-       "rcfilters-watchlist-markseen-button": "Бар үзгәртүләрне каралган дип билгеләргә",
+       "rcfilters-watchlist-markseen-button": "Барлык үзгәрешләрне каралган дип билгеләргә",
        "rcfilters-watchlist-edit-watchlist-button": "Күзәтү исемлегегезне үзгәртү",
        "rcfilters-watchlist-showupdated": "Сезнең соңгы төзәтмәләрдән соң үзгәргән битләр <strong>калын</strong> һәм тулы маркер белән күрсәтелгән",
        "rcfilters-preference-label": "JavaScript тан башка интерфейсын куллану",
-       "rcnotefrom": "Астарак <strong>$3, $4</strong> өчен {{PLURAL:$5|үзгәртүләр күрсәтелгән}} (<strong>$1</strong> артык түгел).",
-       "rclistfrom": "$3 $2 башлап яңа үзгәртүләрне күрсәт",
-       "rcshowhideminor": "кече төзәтмәләрне $1",
-       "rcshowhideminor-show": "Ð\9aүрсәтү",
-       "rcshowhideminor-hide": "Яшерү",
-       "rcshowhidebots": "ботларны $1",
-       "rcshowhidebots-show": "Ð\9aүрсәтү",
-       "rcshowhidebots-hide": "Яшерү",
-       "rcshowhideliu": "теркәлгән кулланучыларны $1",
-       "rcshowhideliu-show": "Ð\9aÒ¯Ñ\80Ñ\81Ó\99Ñ\82",
-       "rcshowhideliu-hide": "Яшерү",
-       "rcshowhideanons": "кермәгән кулланучыларны $1",
-       "rcshowhideanons-show": "Ð\9aүрсәтү",
-       "rcshowhideanons-hide": "Яшерү",
+       "rcnotefrom": "Астарак <strong>$3, $4</strong> өчен {{PLURAL:$5|үзгәрешләр күрсәтелгән}} (<strong>$1</strong> артык түгел).",
+       "rclistfrom": "$3 $2 башлап яңа үзгәрешләрне күрсәтү",
+       "rcshowhideminor": "Ð\9aече төзәтмәләрне $1",
+       "rcshowhideminor-show": "күрсәтү",
+       "rcshowhideminor-hide": "яшерү",
+       "rcshowhidebots": "Ð\91отларны $1",
+       "rcshowhidebots-show": "күрсәтү",
+       "rcshowhidebots-hide": "яшерү",
+       "rcshowhideliu": "Кергән кулланучыларны $1",
+       "rcshowhideliu-show": "күÑ\80Ñ\81Ó\99Ñ\82Ò¯",
+       "rcshowhideliu-hide": "яшерү",
+       "rcshowhideanons": "Ð\9aермәгән кулланучыларны $1",
+       "rcshowhideanons-show": "күрсәтү",
+       "rcshowhideanons-hide": "яшерү",
        "rcshowhidepatr": "Тикшерелгән төзәтмәләрне $1",
        "rcshowhidepatr-show": "күрсәтү",
        "rcshowhidepatr-hide": "яшерү",
-       "rcshowhidemine": "минем төзәтмәләремне $1",
-       "rcshowhidemine-show": "Ð\9aүрсәтү",
-       "rcshowhidemine-hide": "Яшерү",
-       "rcshowhidecategorization": "биÑ\82лÓ\99Ñ\80не Ñ\82Ó©Ñ\80кемлÓ\99үне $1",
-       "rcshowhidecategorization-show": "Ð\9aÒ¯Ñ\80Ñ\81Ó\99Ñ\82",
-       "rcshowhidecategorization-hide": "Яшер",
-       "rclinks": "Соңгы $2 көндә ясалган $1 үзгәртүне күрсәтергә",
+       "rcshowhidemine": "Ð\9cинем төзәтмәләремне $1",
+       "rcshowhidemine-show": "күрсәтү",
+       "rcshowhidemine-hide": "яшерү",
+       "rcshowhidecategorization": "Ð\91иÑ\82лÓ\99Ñ\80нең Ñ\82Ó©Ñ\80кемлÓ\99вен $1",
+       "rcshowhidecategorization-show": "күÑ\80Ñ\81Ó\99Ñ\82Ò¯",
+       "rcshowhidecategorization-hide": "яшерү",
+       "rclinks": "Соңгы $2 көндә ясалган $1 үзгәрешне күрсәтергә",
        "diff": "аерма",
        "hist": "тарих",
-       "hide": "Яшер",
-       "show": "Күрсәт",
+       "hide": "Яшерү",
+       "show": "Күрсәтү",
        "minoreditletter": "к",
        "newpageletter": "Я",
        "boteditletter": "б",
        "recentchangeslinked-toolbox": "Бәйләнешле үзгәрешләр",
        "recentchangeslinked-title": "\"$1\" битенә бәйләнешле үзгәртүләр",
        "recentchangeslinked-summary": "Бу битттән яисә бу биткә сылтаган битләрдәге үзгәртмәле карау өчен битнең исемен кертегез. (Билгеле бер төркемгә караган битләрне карау өчен {{ns:category}}:Төркем исемен языгыз).[[Special:Watchlist|Күзәтү исемлегегезгә]] керә торган битләр '''калын''' итеп күрсәтелгән.",
-       "recentchangeslinked-page": "Битнең исеме:",
+       "recentchangeslinked-page": "Битң исеме:",
        "recentchangeslinked-to": "Моның урынына бу биткә бәйле булган битләрдәге үзгәртүләрне күрсәтү",
        "upload": "Файл төяү",
-       "uploadbtn": "Файлны йөкләү",
-       "reuploaddesc": "Файлны йөкләүгә кире кату",
+       "uploadbtn": "Файл төяү",
+       "reuploaddesc": "Файл төявен кире кагу",
        "upload-tryagain": "Яңартылган файлны җибәрү",
        "uploadnologin": "Сез хисап язмагызга кермәгәнсез",
        "uploadnologintext": "Файлларны серверга йөкләү өчен Сез сәхифәгә $1 тиеш.",
        "upload_directory_missing": "$1 Йөкләнү директориясе юк",
        "upload_directory_read_only": "Моңа Сезнең хокукларыгыз юк һәм веб-сервер $1 папкасыны йөкли алмый.",
-       "uploaderror": "Файлны йөкләүдә хата",
+       "uploaderror": "Файл төяү хатасы",
        "upload-recreate-warning": "'''Игътибар: Мондый исемле файл бетерелгән яки исеме алмаштырылган '''",
        "uploadtext": "Бу форманы кулланып серверга файллар йөкли аласыз. \nЭлегрәк йөкләнелгән файлларны карау өчен [[Special:FileList|йөкләнелгән файллар исемлегенә]] мәрәҗәгать итегез. Шулай ук ул [[Special:Log/upload|йөкләнмәләр исемлегенә]] һәм [[Special:Log/delete|бетерелгән файллар]] исемлегенә дә языла.\n\nФайлны мәкаләгә йөкләү өчен Сез менә бу үрнәкләрне куллана аласыз:\n* <strong><code><nowiki>[[</nowiki>{{ns:file}}<nowiki>:Рәсем.jpg]]</nowiki></code></strong> — файлның тулы юрамасын кую өчен;\n* <strong><code><nowiki>[[</nowiki>{{ns:file}}<nowiki>:Рәсем.png|200px|thumb|left|тасвирламасы]]</nowiki></code></strong> — 200 пиксельга кадәр киңлектәге  һәм текстның сул ягында, тасвирламасы белән;\n* <strong><code><nowiki>[[</nowiki>{{ns:media}}<nowiki>:Файл.ogg]]</nowiki></code></strong> — биттә файлны сүрәтләмичә, бары тик сылтамасын гына кую.",
        "upload-permitted": "{{PLURAL:$2|Рөхсәт ителгән файл төрләре}}: $1.",
        "upload-preferred": "{{PLURAL:$2|Мөмкин булган файл төрләре}}: $1.",
        "upload-prohibited": "{{PLURAL:$2|Тыелган файл төрләре}}: $1.",
-       "uploadlogpage": "Ð\99өклÓ\99ү көндәлеге",
+       "uploadlogpage": "ТөÑ\8fү көндәлеге",
        "uploadlogpagetext": "Аста яңа йөкләнелгән файллар исемлеге бирелә.\nШулай ук [[Special:NewFiles|яңа файллар галлереясын]] карагыз",
        "filename": "Файл исеме",
-       "filedesc": "Кыска тасвирлама",
-       "fileuploadsummary": "Үзгәртүләр тасвирламасы:",
-       "filereuploadsummary": "Файлдагы үзгәртүләр:",
+       "filedesc": "Кыскача аңлатма",
+       "fileuploadsummary": "Кыскача аңлатма:",
+       "filereuploadsummary": "Файлда үзгәрешләр:",
        "filestatus": "Тарату хокуклары:",
        "filesource": "Чыганагы:",
        "ignorewarning": "Белдерүне кире кагу һәм файлны саклау",
        "illegal-filename": "Мондый файл исеменә рөхсәт юк",
        "uploadwarning": "Кисәтү",
        "savefile": "Файлны саклау",
-       "uploaddisabled": "Ð\99өклÓ\99Ò¯ Ñ\82Ñ\8bелган",
-       "copyuploaddisabled": "URL Ð°Ð´Ñ\80еÑ\81Ñ\8b Ð±Ñ\83енÑ\87а Ð¹Ó©ÐºÐ»Ó\99Ò¯ Ñ\8fбÑ\8bлган.",
+       "uploaddisabled": "ТөÑ\8fÒ¯ Ñ\82Ñ\8bелган.",
+       "copyuploaddisabled": "URL Ð±Ñ\83енÑ\87а Ñ\82Ó©Ñ\8fÒ¯ Ñ\81үндеÑ\80елгÓ\99н.",
        "uploaddisabledtext": "Файлларны йөкләү ябылган.",
        "upload-source": "Файлның чыганагы",
        "sourcefilename": "Файлның чыганагы:",
        "destfilename": "Файлның яңа исеме:",
        "upload-maxfilesize": "Файлның максималь зурлыгы: $1",
        "upload-description": "Файлның тасвирламасы",
-       "upload-options": "Ð\99өклÓ\99ү параметрлары",
+       "upload-options": "ТөÑ\8fү параметрлары",
        "watchthisupload": "Бу файлны күзәтү",
        "filewasdeleted": "Мондый исемле файл бетерелгән булган инде. Зинһар,яңадан йөкләү алдыннан $1 карагыз",
        "filename-bad-prefix": "Файлның исеме '''«$1»''' дип башлана. Зинһар, файлны тасвирлаучы исем бирегез.",
        "filename-prefix-blacklist": " #<!-- ничек бар шулай калдырыгыз --> <pre>\n# Синтаксис төбәндәгечә:\n#   *  «#» дип башланган барлык нәрсә дә комментарий дип аталачак\n#   * Һәрбер буш рәт — файлның исеменең префиксы, цифрлы камера бирүче исем\nCIMG # Casio\nDSC_ # Nikon\nDSCF # Fuji\nDSCN # Nikon\nDUW # кайсыбер кәрәзле телефоннар\nIMG # барлык\nJD # Jenoptik\nMGP # Pentax\nPICT # төрле\n #</pre> <!-- ничек бар шулай калдырыгыз -->",
        "upload-proto-error": "Протокол дөрес түгел",
        "upload-file-error": "Эчке хата",
-       "upload-misc-error": "Билгесез йөкләү хатасы",
-       "upload-dialog-title": "Файл йөкләү",
-       "upload-dialog-button-cancel": "Ð\91аÑ\88 Ñ\82аÑ\80Ñ\82у",
-       "upload-dialog-button-back": "Ð\90Ñ\80Ñ\82ка",
-       "upload-dialog-button-done": "Әзер",
+       "upload-misc-error": "Билгесез төяү хатасы",
+       "upload-dialog-title": "Файл төяү",
+       "upload-dialog-button-cancel": "Ð\9aиÑ\80е Ð°Ð»у",
+       "upload-dialog-button-back": "Ð\9aиÑ\80егÓ\99",
+       "upload-dialog-button-done": "Тәмам",
        "upload-dialog-button-save": "Саклау",
-       "upload-dialog-button-upload": "Ð\99өклÓ\99ү",
+       "upload-dialog-button-upload": "ТөÑ\8fү",
        "upload-form-label-infoform-title": "Тулырак",
        "upload-form-label-infoform-name": "Исем",
        "upload-form-label-infoform-description": "Тасвир",
        "upload-form-label-own-work": "Бу минем үз эшем",
        "upload-form-label-infoform-categories": "Төркемнәр",
        "upload-form-label-infoform-date": "Дата",
-       "uploadstash": "Яшерен йөкләү",
+       "uploadstash": "Яшерен төяү",
        "uploadstash-summary": "Әлеге бит йөкләнгән (яисә йукләү барышында булган), әмма викида әлегә күрсәтелмәгән файлларны карау мөмкинлеген бирә. Бу файлларны йөкләгән кулланучыдан башка беркемдә күрә алмый.",
        "uploadstash-clear": "Яшерен файлларны бетерү",
        "uploadstash-nofiles": "Сезнең яшерен файллар юк.",
        "listfiles_date": "Вакыт",
        "listfiles_name": "Файл исеме",
        "listfiles_user": "Кулланучы",
-       "listfiles_size": "Үлчәм",
+       "listfiles_size": "Зурлык",
        "listfiles_description": "Тасвир",
        "listfiles_count": "Юрамалар",
        "listfiles-latestversion": "Агымдагы юрама",
        "randomincategory-category": "Төркем:",
        "randomincategory-submit": "Күчү",
        "randomredirect": "Очраклы биткә күчү",
-       "statistics": "ХиÑ\81апнамÓ\99",
-       "statistics-header-pages": "Ð\91иÑ\82лÓ\99Ñ\80 Ñ\85иÑ\81апнамÓ\99Ñ\81е",
-       "statistics-header-edits": "Үзгәртүләр хисапнамәсе",
-       "statistics-header-users": "Кулланучылар буенча хисапнамә",
-       "statistics-header-hooks": "Ð\91аÑ\88ка Ñ\85иÑ\81апнамÓ\99лÓ\99Ñ\80",
+       "statistics": "СÑ\82аÑ\82иÑ\81Ñ\82ика",
+       "statistics-header-pages": "Ð\91иÑ\82лÓ\99Ñ\80 Ñ\81Ñ\82аÑ\82иÑ\81Ñ\82икаÑ\81Ñ\8b",
+       "statistics-header-edits": "Төзәтмәләр статистикасы",
+       "statistics-header-users": "Кулланучылар статистикасы",
+       "statistics-header-hooks": "Ð\91аÑ\88ка Ñ\81Ñ\82аÑ\82иÑ\81Ñ\82ика",
        "statistics-articles": "Мәкаләләр саны",
        "statistics-pages": "Битләр саны",
        "statistics-pages-desc": "Барлык вики, бәхәс, күчерү һәм башка битләрне дә истә тотып.",
        "mostimages": "Иң кулланган сүрәтләр",
        "mostrevisions": "Күп үзгәртүләр белән битләр",
        "prefixindex": "Барлык алкушымча белән битләр",
-       "prefixindex-submit": "Күрсәт",
+       "prefixindex-submit": "Күрсәтү",
        "shortpages": "Кыска битләр",
        "longpages": "Озын битләр",
        "deadendpages": "Тупик битләре",
        "protectedpages": "Якланган битләр",
+       "protectedpages-filters": "Сөзгечләр:",
        "protectedpages-timestamp": "Дата/вакыт",
        "protectedpages-page": "Бит",
        "protectedpages-expiry": "Тәмамлана",
        "listusers": "Кулланучылар исемлеге",
        "usercreated": "$3 $1 көнне $2 вакытта {{GENDER:$3|теркәлде}}",
        "newpages": "Яңа битләр",
-       "newpages-submit": "Күрсәт",
+       "newpages-submit": "Күрсәтү",
        "newpages-username": "Кулланучы:",
        "ancientpages": "Иң иске битләр",
        "move": "Күчерү",
        "specialloguserlabel": "Башкаручы:",
        "speciallogtitlelabel": "Максат (атама яки {{ns:user}}:кулланучы исеме):",
        "log": "Көндәлекләр",
-       "logeventslist-submit": "Күрсәт",
+       "logeventslist-submit": "Күрсәтү",
        "all-logs-page": "Барлык көндәлекләр",
        "alllogstext": "{{SITENAME}} сәхифәсенең гомуми көндәлекләре исемлеге.\nСез нәтиҗәләрне көндәлек төре, кулланучы исеме (хәреф зурлыгын истә тотыгыз) яки куззаллаган бит (шулай ук хәреф зурлыгын истә тотыгыз) буенча тәртипкә салырга мөмкин.",
        "logempty": "Кирәкле язмалар көндәлектә юк.",
        "allpages-hide-redirects": "Юнәлтүләрне яшерү",
        "cachedspecial-refresh-now": "Соңгы юраманы карау.",
        "categories": "Төркемнәр",
-       "categories-submit": "Күрсәт",
+       "categories-submit": "Күрсәтү",
        "categoriespagetext": "{{PLURAL:$1|1=Әлеге төркем үз өченә|Әлеге төркемнәр  үз өченә}}   битләрне һәм медиа-файлларны ала.\nАста [[Special:UnusedCategories|кулланылмаган төркемнәр]] кәрсәтелгән.\nШулай ук  [[Special:WantedCategories|кирәкле төркемнәр исемлегендә]] карагыз.",
        "deletedcontributions": "Кулланучының бетерелгән кертеме",
        "deletedcontributions-title": "Бетерелгән кертем",
        "sp-deletedcontributions-contribs": "кертем",
        "linksearch": "Тышкы сылтамаларны эзләү",
        "linksearch-pat": "Эзләү өчен үрнәк:",
-       "linksearch-ns": "Ð\98Ñ\81емнÓ\99Ñ\80 Ð¼Ó\99йданÑ\8b:",
+       "linksearch-ns": "Ð\98Ñ\81емнÓ\99Ñ\80 ÐºÐ¸Ò£Ð»ÐµÐ³Ðµ:",
        "linksearch-ok": "Эзләү",
        "linksearch-line": "$2 мәкаләсеннән $1 мәкаләгә сылтама",
        "listusers-submit": "Күрсәтү",
        "listgrouprights-rights": "Хокуклар",
        "listgrouprights-helppage": "Help:Төркемнәрнең хокуклары",
        "listgrouprights-members": "(төркем исемлеге)",
-       "listgrouprights-namespaceprotection-namespace": "Ð\98Ñ\81емнÓ\99Ñ\80 Ð¼Ó\99йданÑ\8b",
+       "listgrouprights-namespaceprotection-namespace": "Ð\98Ñ\81емнÓ\99Ñ\80 ÐºÐ¸Ò£Ð»ÐµÐ³Ðµ",
        "listgrants": "Рөхсәтләр",
        "listgrants-grant": "Рөхсәт",
        "listgrants-rights": "Хокуклар",
        "trackingcategories": "Күзәтелүче төркемнәр",
        "trackingcategories-msg": "Күзәтүче төркем",
-       "trackingcategories-name": "Хат исеме",
+       "trackingcategories-name": "Хәбәр исеме",
        "emailuser": "Кулланучыга хат",
        "emailuser-title-target": "{{GENDER:$1|Кулланучыга}} электрон хат язу",
        "emailuser-title-notarget": "Кулланучыга хат җибәрү",
        "watchlistfor2": "$1 өчен $2",
        "nowatchlist": "Күзәтү исемлегегездә битләр юк.",
        "watchnologin": "Кермәдегез",
+       "addwatch": "Күзәтү исемлегенә өстәү",
        "addedwatchtext": "\"[[:$1]]\" бите [[Special:Watchlist|күзәтү исемлегегезгә]] өстәлде.",
        "removedwatchtext": "«[[:$1]]» мәкаләсе 1әм аның бәхәс бите [[Special:Watchlist|сезнең күзәтү исемлегегездән]] бетерелде.",
        "watch": "Күзәтү",
        "watchthispage": "Бу битне күзәтү",
        "unwatch": "Күзәтмәү",
-       "unwatchthispage": "Күзәтүне туктат",
+       "unwatchthispage": "Күзәтүне туктату",
        "notanarticle": "Мәкалә түгел",
-       "watchlist-details": "Күзәтү исемлегегездә (бәхәс битләре белән бергә) {{PLURAL:$1|$1 бит}}.",
+       "watchlist-details": "Күзәтү исемлегегездә {{PLURAL:$1|$1 бит}} (бәхәс битләреннән тыш).",
        "wlheader-enotif": "Электрон почта аша белдерүләр ачык.",
-       "wlheader-showupdated": "Сезнең соңгы төзәтмәләрдән соң үзгәргән битләр <strong>калын</strong> шрифт белән күрсәтелгән.",
-       "wlnote": "Түбәндә $3 $4 вакыт аралыгының {{PLURAL:$2|соңгы сәгатендә|соңгы <strong>$2</strong> сәгатендә}} ясалган {{PLURAL:$1|ахыргы төзәтмә|ахыргы <strong>$1</strong> төзәтмә}} күрсәтелгән.",
-       "wlshowlast": "$1 сәгать $2 көн өчендә күрсәтү",
-       "watchlist-hide": "Яшер",
-       "watchlist-submit": "Күрсәт",
-       "wlshowtime": "Ð\9aÒ¯Ñ\80Ñ\81Ó\99Ñ\82елүÑ\87е Ð²Ð°ÐºÑ\8bÑ\82 Ð°Ñ\80алÑ\8bгÑ\8b:",
-       "wlshowhideminor": "кече үзгәртүләр",
-       "wlshowhidebots": "бот",
-       "wlshowhideliu": "теркәлгән кулланучы",
-       "wlshowhideanons": "аноним ÐºÑ\83лланÑ\83Ñ\87Ñ\8bлаÑ\80нÑ\8bкÑ\8bн",
-       "wlshowhidepatr": "тикшерелгән үзгәртүләр",
-       "wlshowhidemine": "үзгәртүләрем",
-       "wlshowhidecategorization": "битләрне төркемләүне",
+       "wlheader-showupdated": "Сез соңгы карап чыгудан соң үзгәртелгән битләр <strong>калын</strong> хәрефләр белән күрсәтелгән.",
+       "wlnote": "Түбәндә $3 көнгә $4 вакытка соңгы {{PLURAL:$2|бер сәгать|<strong>$2</strong> сәгать}} эчендә ясалган соңгы {{PLURAL:$1|үзгәреш|<strong>$1</strong> үзгәреш}} күрсәтелгән.",
+       "wlshowlast": "Соңгы $1 сәгать $2 көн эчендә күрсәтү",
+       "watchlist-hide": "Яшерү",
+       "watchlist-submit": "Күрсәтү",
+       "wlshowtime": "Ð\92акÑ\8bÑ\82 Ñ\8dÑ\87ендÓ\99 ÐºÒ¯Ñ\80Ñ\81Ó\99Ñ\82еÑ\80гÓ\99:",
+       "wlshowhideminor": "кече төзәтмәләр",
+       "wlshowhidebots": "ботныкы",
+       "wlshowhideliu": "кергән кулланучыныкы",
+       "wlshowhideanons": "кеÑ\80мÓ\99гÓ\99н ÐºÑ\83лланÑ\83Ñ\87Ñ\8bнÑ\8bкÑ\8b",
+       "wlshowhidepatr": "тикшерелгән төзәтмәләр",
+       "wlshowhidemine": "төзәтмәләрем",
+       "wlshowhidecategorization": "битләр төркемләү",
        "watchlist-options": "Күзәтү исемлеге көйләүләре",
        "watching": "Күзәтү исемлегемә өстәүе…",
        "unwatching": "Күзәтү исемлегемнән чыгаруы…",
        "enotif_impersonal_salutation": "{{SITENAME}} кулланучы",
        "enotif_lastvisited": "Соңгы керүегездән соң булган барлык үзгәртүләрне күрер өчен, бу сылтама аша узыгыз: $1",
        "enotif_body": "Хөрмәтле $WATCHINGUSERNAME,\n\n\n$PAGEINTRO $NEWPAGE\n\nҮзгәртүнең кыска эчтәлеге: $PAGESUMMARY $PAGEMINOREDIT\n\nҮзгәртүчегә язу:\nэл. почта $PAGEEDITOR_EMAIL\nвики $PAGEEDITOR_WIKI\n\nБу биткә кермәсәгез, аның башка үзгәртүләре турында хат җибәрелмәячәк. Шулай ук сез күзәтү исемлегегездә булган битләр өчен хәбәр бирү флагын алып куя аласыз.\n\n             {{grammar:genitive|{{SITENAME}}}} хәбәр бирү системасы\n\n--\nХәбәр итүләр көйләүләрен үзгәртү:\n{{canonicalurl:{{#special:Preferences}}}}\n\nКүзәтү исемлеге көйләүләрен үзгәртү:\n$HELPPAGE\n\nБитне сезнең күзәтү исемлегездән бетерү:\n$UNWATCHURL\n\nЭлемтә һәм ярдәм:\n$HELPPAGE",
+       "enotif_minoredit": "Бу кече төзәтмә",
        "created": "төзелгән",
        "changed": "үзгәртелде",
        "deletepage": "Битне бетерү",
        "deletedtext": "«$1» бетерелгән инде.<br />\nСоңгы бетерелгән битләрне күрер өчен, $2 карагыз.",
        "dellogpage": "Бетерү көндәлеге",
        "deletionlog": "бетерү көндәлеге",
+       "log-name-create": "Битләр төзү көндәлеге",
        "deletecomment": "Сәбәп:",
        "deleteotherreason": "Башка/өстәмә сәбәп:",
        "deletereasonotherlist": "Башка сәбәп",
        "deletereason-dropdown": "* Бетерүнең сәбәпләре\n** спам\n** вандаллык\n** автор хокукларын бозу\n** автор соравы буенча\n** эшсез күчермә",
        "delete-edit-reasonlist": "Сәбәпләр исемлеген үзгәртү",
+       "rollback-confirmation-yes": "Кире кайтару",
        "rollback-confirmation-no": "Кире алу",
        "rollbacklink": "кире кайтару",
        "rollbacklinkcount": "$1 {{PLURAL:$1|төзәтмәне}} кире кагу",
        "restriction-edit": "Үзгәртү",
        "restriction-move": "Күчерү",
        "restriction-create": "Төзү",
-       "restriction-upload": "Ð\99өклÓ\99ү",
+       "restriction-upload": "ТөÑ\8fү",
        "restriction-level-sysop": "тулы яклау",
        "restriction-level-autoconfirmed": "өлешчә яклау",
        "restriction-level-all": "барлык дәрәҗәләр",
        "undelete-search-submit": "Эзләү",
        "undelete-error-long": "Файлны торгызу вакытында хаталар чыкты:\n\n$1",
        "undelete-show-file-submit": "Әйе",
-       "namespace": "Ð\98Ñ\81емнÓ\99Ñ\80 Ð¼Ó\99йданÑ\8b:",
+       "namespace": "Ð\98Ñ\81емнÓ\99Ñ\80 ÐºÐ¸Ò£Ð»ÐµÐ³Ðµ:",
        "invert": "Киресен сайлау",
        "tooltip-invert": "Сайланган исемлектәге үзгәргәртүләр күрсәтелмәсен өчен монда тамга куегыз",
        "namespace_association": "Бәйле тирәлек",
        "uctop": "хәзерге",
        "month": "Айдан башлап (һәм элегрәк):",
        "year": "Елдан башлап (һәм элегрәк):",
+       "date": "Датадан башлап (һәм элегрәк):",
        "sp-contributions-newbies": "Яңа хисап язмаларыннан ясалган кертемне генә карау",
        "sp-contributions-newbies-sub": "Яңа хисап язмалары өчен",
        "sp-contributions-blocklog": "тыю көндәлеге",
-       "sp-contributions-uploads": "йөкләүләр",
+       "sp-contributions-uploads": "төяүләр",
        "sp-contributions-logs": "көндәлекләр",
        "sp-contributions-talk": "бәхәс",
        "sp-contributions-search": "Кертемне эзләү",
        "whatlinkshere-hidetrans": "Кертүләрне $1",
        "whatlinkshere-hidelinks": "Сылтамаларны $1",
        "whatlinkshere-hideimages": "$1 файл сылтамалары",
-       "whatlinkshere-filters": "ФилÑ\8cÑ\82Ñ\80лар",
+       "whatlinkshere-filters": "СөзгеÑ\87лÓ\99р",
        "whatlinkshere-submit": "Башкару",
        "autoblockid": "Автотыю #$1",
        "block": "Кулланучыны тыю",
        "ipbother": "Башка вакыт:",
        "ipboptions": "2 сәгать:2 hours,1 көн:1 day,3 көн:3 days,1 атна:1 week,2 атна:2 weeks,1 ай:1 month,3ай:3 months,6 ай:6 months,1 ел:1 year,чикләүсез:infinite",
        "ipb-confirm": "Тыюны раслау",
+       "ipb-partial": "Өлешчә",
        "ipb-pages-label": "Битләр",
+       "ipb-namespaces-label": "Исемнәр киңлекләре",
        "badipaddress": "Ялгыш IP адресы",
        "blockipsuccesssub": "Тыю башкарылган",
        "ipb-unblock-addr": "$1 кулланучысын тыюдан азат итү",
        "autoblocklist-submit": "Эзләү",
        "autoblocklist-legend": "Автотыелу исемлеге",
        "ipblocklist": "Тыелган кулланучылар",
+       "blocklist-type-opt-all": "Барлык",
        "blocklist-timestamp": "Дата/вакыт",
        "blocklist-target": "Максат",
        "blocklist-expiry": "Тәмамлана",
        "export-submit": "Экспортлау",
        "export-addcattext": "Бу төркемнән битләр өстәү:",
        "export-addcat": "Өстәү",
+       "export-addnstext": "Исемнәр киңлегеннән битләр өстәү:",
        "export-addns": "Өстәү",
        "export-download": "Файл буларак саклау",
        "allmessages": "Система хәбәрләре",
        "allmessagesname": "Исем",
        "allmessagesdefault": "Баштан ук куелган текс",
        "allmessagestext": "Бу исемлек MediaWiki исемнәр мәйданында булган система хәбәрләренең исемлеге.\nГомуми MediaWiki локализациясендә катнашырга теләсәгез, зинһар [https://www.mediawiki.org/wiki/Special:MyLanguage/Localisation MediaWiki Локализациясе] һәм [https://translatewiki.net translatewiki.net] сәхифәләрне кулланыгыз.",
-       "allmessages-filter-legend": "ФилÑ\8cÑ\82Ñ\80",
+       "allmessages-filter-legend": "СөзгеÑ\87",
        "allmessages-filter-unmodified": "Үзгәртелмәгән",
        "allmessages-filter-all": "Барысы",
        "allmessages-filter-modified": "Үзгәртелгән",
        "allmessages-language": "Тел:",
-       "allmessages-filter-submit": "Ð\9aÒ¯Ñ\87ү",
+       "allmessages-filter-submit": "Сөзү",
        "allmessages-filter-translate": "Тәрҗемә итү",
        "thumbnail-more": "Зурайту",
        "filemissing": "Файл табылмады",
        "import-interwiki-sourcewiki": "Чыганак вики:",
        "import-interwiki-sourcepage": "Чыганак бит:",
        "import-interwiki-history": "Бу битнең барлык үзгәртү тарихын күчермәләү",
-       "import-interwiki-templates": "Барлык үрнәкләрне кертү",
-       "import-interwiki-submit": "Ð\98мпоÑ\80Ñ\82лаÑ\83",
+       "import-interwiki-templates": "Барлык калыпларны өстәү",
+       "import-interwiki-submit": "Ð\9aеÑ\80Ñ\82Ò¯",
        "import-upload-filename": "Файл исеме:",
        "import-comment": "Искәрмә:",
        "importtext": "Зинһар өчен, битне күчерү өчен [[Special:Export|махсус корал]] кулланыгыз. Файлны дискка саклагыз, аннан соң монда йөкләгез.",
        "tooltip-ca-nstab-special": "Бу махсус бит, аны үзгәртү мөмкин түгел",
        "tooltip-ca-nstab-project": "Проектның бите",
        "tooltip-ca-nstab-image": "Сүрәтнең бите",
-       "tooltip-ca-nstab-mediawiki": "MediaWiki - хат бите",
-       "tooltip-ca-nstab-template": "Үрнәк бите",
-       "tooltip-ca-nstab-help": "ЯÑ\80дÓ\99м Ð±Ð¸Ñ\82ен ÐºÐ°Ñ\80аÑ\83",
+       "tooltip-ca-nstab-mediawiki": "MediaWiki хәбәре бите",
+       "tooltip-ca-nstab-template": "Калып бите",
+       "tooltip-ca-nstab-help": "Ð\91елеÑ\88мÓ\99 Ð±Ð¸Ñ\82е",
        "tooltip-ca-nstab-category": "Төркем битен карау",
        "tooltip-minoredit": "Бу төзәтмәне кече дип билгеләү",
        "tooltip-save": "Үзгәртүләрегезне саклау",
-       "tooltip-preview": "Алдан карау, саклау алдыннан үзгәртүләрегезнең карап чыгыгыз!",
+       "tooltip-publish": "Үзгәрешләрегезне сакларга",
+       "tooltip-preview": "Үзгәртүләрегезне алдан карарга. Моны саклау алдыннан кулланыгыз әле.",
        "tooltip-diff": "Сезнең үзгәртүләрегезне күрсәтү.",
        "tooltip-compareselectedversions": "Бу битнең сайланган ике юрамасы арасында аерманы карау",
        "tooltip-watch": "Бу битне күзәтү исемлегемә өстәү",
+       "tooltip-watchlistedit-raw-submit": "Күзәтү исемлеген яңарту",
        "tooltip-recreate": "Бу битне кире кайтару",
-       "tooltip-upload": "Ð\99өклÓ\99үне Ð±Ð°Ñ\88лаÑ\83",
+       "tooltip-upload": "ТөÑ\8fүне Ð±Ð°Ñ\88лаÑ\80га",
        "tooltip-rollback": "\"Кире кайтару\" соңгы кулланучының бу биттә ясаган '''барлык''' үзгәртүләрен бетерә.",
        "tooltip-undo": "Бу үзгәртүне алдан карап үткәрмәү. Шулай ук үткәрмәүнең сәбәбен язып була.",
        "tooltip-preferences-save": "Көйләнмәләрегезне саклау",
        "markedaspatrolledtext": "Сайланган [[:$1]] мәкаләсенең әлеге юрамасы тикшерелгән дип тамгаланды.",
        "patrol-log-page": "Тикшерү көндәлеге",
        "patrol-log-header": "Бу тикшерелгән битләрнең көндәлеге.",
-       "confirm-markpatrolled-button": "Ярый",
+       "confirm-markpatrolled-button": "Ярар",
        "deletedrevision": "$1 битенең иске юрамасы бетерелде",
        "filedeleteerror-short": "Файлны бетерү хатасы: $1",
        "filedeleteerror-long": "Файлны бетерү вакытында хаталар чыкты:\n\n$1",
        "show-big-image": "Төп файл",
        "show-big-image-preview": "Алдан карауның зурлыгы: $1.",
        "show-big-image-other": "{{PLURAL:$2|1=Башка зурлык|Башка зурлыклар}}: $1.",
-       "show-big-image-size": "$1 Ã\97 $2 Ð¿Ð¸ÐºÑ\81елÑ\8c",
+       "show-big-image-size": "$1 Ã\97 $2 Ð½Ð¾ÐºÑ\82а",
        "file-info-gif-looped": "әйләнешле",
        "file-info-gif-frames": "$1 {{PLURAL:$1|фрейм}}",
        "file-info-png-looped": "әйләнешле",
        "newimages": "Яңа сүрәтләр җыелмасы",
-       "newimages-legend": "ФилÑ\8cÑ\82Ñ\80",
+       "newimages-legend": "СөзгеÑ\87",
        "ilsubmit": "Эзләү",
        "bydate": "дата буенча",
        "seconds": "{{PLURAL:$1|$1 секунд}}",
        "namespacesall": "барлык",
        "monthsall": "барлык",
        "recreate": "Яңадан ясау",
-       "confirm_purge_button": "OK",
+       "confirm_purge_button": "Ярар",
        "confirm-purge-top": "Бу битнең кэшы чистартылсынмы?",
        "confirm-purge-bottom": "Кэшны чистартудан соң аның соңгы юрамасы күрсәтеләчәк.",
-       "confirm-watch-button": "OK",
-       "confirm-unwatch-button": "Ð\9eÐ\9a",
-       "confirm-rollback-button": "Ð\9eÐ\9a",
+       "confirm-watch-button": "Ярар",
+       "confirm-unwatch-button": "ЯÑ\80аÑ\80",
+       "confirm-rollback-button": "ЯÑ\80аÑ\80",
        "pipe-separator": "&#32;|&#32;",
        "quotation-marks": "«$1»",
        "imgmultipageprev": "← алдагы бит",
        "autoredircomment": "[[$1]] битенә юнәлтү",
        "autosumm-new": "Яңа бит: «$1»",
        "watchlistedit-raw-titles": "Язмалар:",
+       "watchlistedit-raw-submit": "Исемлекне саклау",
        "watchlistedit-clear-titles": "Башлык:",
        "watchlisttools-clear": "Күзәтү исемлеген бушату",
        "watchlisttools-view": "Соңгы үзгәртүләрне күрсәтү",
        "redirect-file": "Файл исеме",
        "fileduplicatesearch": "Бер үк файлларны эзләү",
        "fileduplicatesearch-submit": "Эзләү",
+       "fileduplicatesearch-info": "$1 × $2 нокта<br />Файл зурлыгы: $3<br />MIME төре: $4",
        "specialpages": "Махсус битләр",
        "specialpages-note-restricted": "* Гади махсус битләр.\n* <span class=\"mw-specialpagerestricted\">Чикләнелгән махсус битләр.</span>",
        "specialpages-group-maintenance": "Техник карау хисапнамәсе",
        "intentionallyblankpage": "Бу бит махсус буш калдырылган",
        "external_image_whitelist": "#Бу юлны ничек бар, шулаө калдырыгыз<pre>\n#Монда даими фразаларның фрагментларын куегыз (// арасында торган өлешен)\n#алар тышкы сурәтләрнең URL белән бәйләнерләр.\n#Туры килгәннәре сурәт буларак, туры килмәгәннәре сурәткә сылтама буларак күрсәтеләчәкләр.\n# # билгесе белән башланучы юллар шәрехнамә дип саналалар.\n#Юллар регистрга игътибар бирмиләр.\n\n#Даими фразаларның фрагментларын бу кыр өстендә куегыз. Бу кырны ничек бар, шулай калдырыгыз.</pre>",
        "tags": "Гамәлдә булучы үзгәртүләр билгеләре",
-       "tag-filter": "[[Special:Tags|Ð\91илгелÓ\99Ñ\80]] Ñ\84илÑ\8cÑ\82Ñ\80Ñ\8b:",
-       "tag-filter-submit": "ФилÑ\8cÑ\82Ñ\80лаÑ\83",
+       "tag-filter": "[[Special:Tags|Ð\91илгелÓ\99Ñ\80]] Ñ\81өзгеÑ\87е:",
+       "tag-filter-submit": "Сөзү",
        "tag-list-wrapper": "[[Special:Tags|{{PLURAL:$1|1=Билге|Билгеләр}}]]: $2",
        "tags-title": "Теглар",
        "tags-intro": "Әлеге сәхифәдә төзәтүләрне билгеләгән, программа тәэмин итә торган теглар исемлеге һәм шул тегларның аңламнары китерелгән.",
        "tags-create-submit": "Төзү",
        "tags-deactivate-submit": "Өзергә",
        "comparepages": "Битләрне чагыштыру",
-       "compare-page1": "Беренче сәхифә",
-       "compare-page2": "Икенче сәхифә",
+       "compare-page1": "Беренче бит",
+       "compare-page2": "Икенче бит",
        "compare-rev1": "Беренче юрама",
        "compare-rev2": "Икенче юрама",
-       "compare-submit": "Чагыштыр",
+       "compare-submit": "Чагыштыру",
+       "diff-form": "Аермалыклар",
+       "diff-form-submit": "Аермасын күрсәтү",
        "permanentlink": "Даими сылтама",
        "dberr-problems": "Гафу итегез! Сайтта техник кыенлыклар чыкты.",
        "dberr-again": "Сәхифәне берничә минуттан соң яңартып карагыз.",
        "dberr-info": "(Мәгълүматлар базасы серверы белән тоташырга мөмкин түгел: $1)",
        "htmlform-submit": "Җибәрү",
-       "htmlform-reset": "Үзгәртүләрне кире кайтару",
+       "htmlform-reset": "Үзгәрешләрне кире кайтару",
        "htmlform-selectorother-other": "Башка",
        "htmlform-no": "Юк",
        "htmlform-yes": "Әйе",
        "htmlform-date-placeholder": "ЕЕЕЕ-АА-КК",
        "htmlform-time-placeholder": "СС:ММ:СС",
        "htmlform-datetime-placeholder": "ЕЕЕЕ-АА-КК СС:ММ:СС",
+       "htmlform-title-not-exists": "$1 барлыкта юк.",
+       "htmlform-user-not-exists": "<strong>$1</strong> барлыкта юк.",
        "logentry-delete-delete": "$1 $3 битен {{GENDER:$2|бетерә}}",
        "logentry-delete-restore": "$1 $3 ($4) битен {{GENDER:$2|торгызды}}",
        "logentry-delete-revision": "$1 $3 битендә {{PLURAL:$5|$5 юрамасының}} күренешен {{GENDER:$2|үзгәртте}}: $4",
        "logentry-upload-overwrite": "$1 $3 өчен яңа юрама {{GENDER:$2|йөкләде}}",
        "rightsnone": "(юк)",
        "feedback-adding": "Фикерне сәхифәгә өстәү ...",
-       "feedback-back": "Ð\90Ñ\80Ñ\82ка",
+       "feedback-back": "Ð\9aиÑ\80егÓ\99",
        "feedback-bugnew": "Мин тикшердем. Яңа хата турында хәбәр итү",
        "feedback-bugornote": "Әгәр дә сез техник проблеманы җентекләп тасвирларга әзер икәнсез, зинһар өчен, [$1 хата турында хәбәр итегез].\nБашка очракта сез түбәндәге гади форманы куллана аласыз. Сезнең шәрехләмә \"[$3 $2]\" сәхифәсенә сезнең кулланучы исеме һәм сез кулланган браузер исеме белән бергә өстәләчәк.",
-       "feedback-cancel": "Ð\91аÑ\88 Ñ\82аÑ\80Ñ\82у",
-       "feedback-close": "Әзер",
+       "feedback-cancel": "Ð\9aиÑ\80е Ð°Ð»у",
+       "feedback-close": "Тәмам",
        "feedback-error1": "Хата. APIдан билгесез нәтиҗә",
        "feedback-error2": "Хата: төзәтү уңышсыз килеп чыкты",
        "feedback-error3": "Хата: APIдан җавап юк.",
        "limitreport-templateargumentsize-value": "$1/$2 {{PLURAL:$2|байт}}",
        "expandtemplates": "Үрнәкләрне ачу",
        "expand_templates_output": "Нәтиҗә",
-       "expand_templates_ok": "OK",
+       "expand_templates_ok": "Ярар",
        "expand_templates_preview": "Алдан карау",
        "pagelang-name": "Бит",
        "pagelang-language": "Тел",
        "default-skin-not-found-row-enabled": "* <code>$1</code> / $2 (ачык)",
        "default-skin-not-found-row-disabled": "* <code>$1</code> / $2 (<strong>ябык</strong>)",
        "mediastatistics": "Медиа хисабы",
+       "mediastatistics-header-audio": "Аудио",
+       "mediastatistics-header-video": "Видео",
        "special-characters-group-latin": "Латин",
        "special-characters-group-latinextended": "Латин (киңәйтелгән)",
        "special-characters-group-ipa": "ХФӘ (IPA)",
        "special-characters-group-arabic": "Гарәп",
        "special-characters-group-arabicextended": "Гарәп (киңәйтелгән)",
        "special-characters-group-persian": "Фарсы",
-       "special-characters-group-hebrew": "ЯÑ\85үд",
-       "special-characters-group-bangla": "Бенгаль",
+       "special-characters-group-hebrew": "Ð\98вÑ\80иÑ\82",
+       "special-characters-group-bangla": "Бенгал",
        "special-characters-group-tamil": "Тамил",
        "special-characters-group-telugu": "Телугу",
        "special-characters-group-sinhala": "Сингал",
        "special-characters-group-lao": "Лаос",
        "special-characters-group-khmer": "Кһмер",
        "special-characters-group-canadianaboriginal": "Канада иҗек язуы",
+       "special-characters-title-endash": "урта сызык",
+       "special-characters-title-emdash": "озын сызык",
+       "special-characters-title-minus": "минус билгесе",
        "mw-widgets-abandonedit": "Сез чыннан да үзгәртүләрне сакламыйча карау режимына чыгарга телисезме?",
        "mw-widgets-abandonedit-discard": "Үзгәрешне кире кагу",
        "mw-widgets-abandonedit-keep": "Үзгәртүне дәвам итү",
        "mw-widgets-abandonedit-title": "Сез ризамы?",
+       "mw-widgets-copytextlayout-copy": "Күчермә алу",
        "mw-widgets-dateinput-no-date": "Дата сайланмаган",
        "mw-widgets-dateinput-placeholder-day": "ЕЕЕЕ-АА-КК",
        "mw-widgets-dateinput-placeholder-month": "ЕЕЕЕ-АА",
        "authmanager-email-label": "Электрон почта",
        "authmanager-email-help": "Электрон почта адресы",
        "authmanager-realname-label": "Чын исеме",
-       "authprovider-resetpass-skip-label": "Калдыру"
+       "authprovider-resetpass-skip-label": "Калдыру",
+       "edit-error-short": "Хата: $1",
+       "edit-error-long": "Хаталар:\n\n$1",
+       "specialmute": "Белдерүсез",
+       "revid": "юрама $1",
+       "pagedata-title": "Бит мәгълүматлары",
+       "passwordpolicies-group": "Төркем",
+       "passwordpolicies-policies": "Кагыйдәләр"
 }
index 0263425..c3644ee 100644 (file)
@@ -1366,28 +1366,34 @@ abstract class Maintenance {
        }
 
        /**
-        * Returns a database to be used by current maintenance script. It can be set by setDB().
-        * If not set, wfGetDB() will be used.
-        * This function has the same parameters as wfGetDB()
+        * Returns a database to be used by current maintenance script.
+        *
+        * This uses the main LBFactory instance by default unless overriden via setDB().
+        *
+        * This function has the same parameters as LoadBalancer::getConnection().
         *
         * @param int $db DB index (DB_REPLICA/DB_MASTER)
         * @param string|string[] $groups default: empty array
-        * @param string|bool $wiki default: current wiki
+        * @param string|bool $dbDomain default: current wiki
         * @return IMaintainableDatabase
         */
-       protected function getDB( $db, $groups = [], $wiki = false ) {
+       protected function getDB( $db, $groups = [], $dbDomain = false ) {
                if ( $this->mDb === null ) {
-                       return wfGetDB( $db, $groups, $wiki );
+                       return MediaWikiServices::getInstance()
+                               ->getDBLoadBalancerFactory()
+                               ->getMainLB( $dbDomain )
+                               ->getMaintenanceConnectionRef( $db, $groups, $dbDomain );
                }
+
                return $this->mDb;
        }
 
        /**
         * Sets database object to be returned by getDB().
         *
-        * @param IDatabase $db
+        * @param IMaintainableDatabase $db
         */
-       public function setDB( IDatabase $db ) {
+       public function setDB( IMaintainableDatabase $db ) {
                $this->mDb = $db;
        }
 
@@ -1529,7 +1535,7 @@ abstract class Maintenance {
                        $title = $titleObj->getPrefixedDBkey();
                        $this->output( "$title..." );
                        # Update searchindex
-                       $u = new SearchUpdate( $pageId, $titleObj->getText(), $rev->getContent() );
+                       $u = new SearchUpdate( $pageId, $titleObj, $rev->getContent() );
                        $u->doUpdate();
                        $this->output( "\n" );
                }
index df3b4a1..08eade9 100644 (file)
@@ -30,7 +30,7 @@ require_once __DIR__ . '/../../includes/export/WikiExporter.php';
 
 use MediaWiki\MediaWikiServices;
 use Wikimedia\Rdbms\LoadBalancer;
-use Wikimedia\Rdbms\IDatabase;
+use Wikimedia\Rdbms\IMaintainableDatabase;
 
 /**
  * @ingroup Dump
@@ -68,7 +68,7 @@ abstract class BackupDumper extends Maintenance {
        /**
         * The dependency-injected database to use.
         *
-        * @var IDatabase|null
+        * @var IMaintainableDatabase|null
         *
         * @see self::setDB
         */
@@ -328,7 +328,7 @@ abstract class BackupDumper extends Maintenance {
         * @todo Fixme: the --server parameter is currently not respected, as it
         * doesn't seem terribly easy to ask the load balancer for a particular
         * connection by name.
-        * @return IDatabase
+        * @return IMaintainableDatabase
         */
        function backupDb() {
                if ( $this->forcedDb !== null ) {
@@ -337,7 +337,7 @@ abstract class BackupDumper extends Maintenance {
 
                $lbFactory = MediaWikiServices::getInstance()->getDBLoadBalancerFactory();
                $this->lb = $lbFactory->newMainLB();
-               $db = $this->lb->getConnection( DB_REPLICA, 'dump' );
+               $db = $this->lb->getMaintenanceConnectionRef( DB_REPLICA, 'dump' );
 
                // Discourage the server from disconnecting us if it takes a long time
                // to read out the big ol' batch query.
@@ -350,10 +350,9 @@ abstract class BackupDumper extends Maintenance {
         * Force the dump to use the provided database connection for database
         * operations, wherever possible.
         *
-        * @param IDatabase|null $db (Optional) the database connection to use. If null, resort to
-        *   use the globally provided ways to get database connections.
+        * @param IMaintainableDatabase $db The database connection to use
         */
-       function setDB( IDatabase $db = null ) {
+       function setDB( IMaintainableDatabase $db ) {
                parent::setDB( $db );
                $this->forcedDb = $db;
        }
index f9b3951..21b92c5 100644 (file)
@@ -240,7 +240,7 @@ TEXT
                }
 
                try {
-                       $this->db = $this->lb->getConnection( DB_REPLICA, 'dump' );
+                       $this->db = $this->lb->getMaintenanceConnectionRef( DB_REPLICA, 'dump' );
                } catch ( Exception $e ) {
                        throw new MWException( __METHOD__
                                . " rotating DB failed to obtain new database (" . $e->getMessage() . ")" );
index 1dd1909..a71bb74 100644 (file)
@@ -115,7 +115,12 @@ class CommandLineInstaller extends Maintenance {
                        $this->setPassOption();
                }
 
-               $installer = InstallerOverrides::getCliInstaller( $siteName, $adminName, $this->mOptions );
+               try {
+                       $installer = InstallerOverrides::getCliInstaller( $siteName, $adminName, $this->mOptions );
+               } catch ( \MediaWiki\Installer\InstallException $e ) {
+                       $this->output( $e->getStatus()->getMessage()->parse() . "\n" );
+                       return false;
+               }
 
                $status = $installer->doEnvironmentChecks();
                if ( $status->isGood() ) {
@@ -123,17 +128,21 @@ class CommandLineInstaller extends Maintenance {
                } else {
                        $installer->showStatusMessage( $status );
 
-                       return;
+                       return false;
                }
                if ( !$envChecksOnly ) {
-                       $installer->execute();
+                       $status = $installer->execute();
+                       if ( !$status->isGood() ) {
+                               return false;
+                       }
                        $installer->writeConfigurationFile( $this->getOption( 'confpath', $IP ) );
+                       $installer->showMessage(
+                               'config-install-success',
+                               $installer->getVar( 'wgServer' ),
+                               $installer->getVar( 'wgScriptPath' )
+                       );
                }
-               $installer->showMessage(
-                       'config-install-success',
-                       $installer->getVar( 'wgServer' ),
-                       $installer->getVar( 'wgScriptPath' )
-               );
+               return true;
        }
 
        private function setDbPassOption() {
index a67417f..f5ebc02 100644 (file)
@@ -93,8 +93,8 @@ class MediaWikiShell extends Maintenance {
                }
                if ( $d > 1 ) {
                        # Set DBO_DEBUG (equivalent of $wgDebugDumpSql)
-                       wfGetDB( DB_MASTER )->setFlag( DBO_DEBUG );
-                       wfGetDB( DB_REPLICA )->setFlag( DBO_DEBUG );
+                       $this->getDB( DB_MASTER )->setFlag( DBO_DEBUG );
+                       $this->getDB( DB_REPLICA )->setFlag( DBO_DEBUG );
                }
        }
 
index 3b0607f..21d8b2d 100644 (file)
@@ -83,8 +83,7 @@ class MwSql extends Maintenance {
                        $index = DB_MASTER;
                }
 
-               /** @var IDatabase $db DB handle for the appropriate cluster/wiki */
-               $db = $lb->getConnection( $index, [], $wiki );
+               $db = $lb->getMaintenanceConnectionRef( $index, [], $wiki );
                if ( $replicaDB != '' && $db->getLBInfo( 'master' ) !== null ) {
                        $this->fatalError( "The server selected ({$db->getServer()}) is not a replica DB." );
                }
index 3866be7..60f88ba 100644 (file)
@@ -42,7 +42,7 @@ class OrphanStats extends Maintenance {
                $lbFactory = MediaWikiServices::getInstance()->getDBLoadBalancerFactory();
                $lb = $lbFactory->getExternalLB( $cluster );
 
-               return $lb->getConnection( DB_REPLICA );
+               return $lb->getMaintenanceConnectionRef( DB_REPLICA );
        }
 
        public function execute() {
index 6fd53cc..92b6679 100644 (file)
  * @ingroup Maintenance ExternalStorage
  */
 
+use Wikimedia\Rdbms\IMaintainableDatabase;
 use MediaWiki\Logger\LegacyLogger;
 use MediaWiki\MediaWikiServices;
 use MediaWiki\Shell\Shell;
-use Wikimedia\Rdbms\IDatabase;
 
 $optionsWithArgs = RecompressTracked::getOptionsWithArgs();
 require __DIR__ . '/../commandLine.inc';
@@ -275,6 +275,7 @@ class RecompressTracked {
        /**
         * Dispatch a command to the next available replica DB.
         * This may block until a replica DB finishes its work and becomes available.
+        * @param array ...$args
         */
        function dispatch( ...$args ) {
                $pipes = $this->replicaPipes;
@@ -647,13 +648,13 @@ class RecompressTracked {
        /**
         * Gets a DB master connection for the given external cluster name
         * @param string $cluster
-        * @return IDatabase
+        * @return IMaintainableDatabase
         */
        function getExtDB( $cluster ) {
                $lbFactory = MediaWikiServices::getInstance()->getDBLoadBalancerFactory();
                $lb = $lbFactory->getExternalLB( $cluster );
 
-               return $lb->getConnection( DB_MASTER );
+               return $lb->getMaintenanceConnectionRef( DB_MASTER );
        }
 
        /**
index 385ae6a..d9793b4 100644 (file)
@@ -232,7 +232,7 @@ class TrackBlobs {
                $pos = $dbw->getMasterPos();
                $dbr->masterPosWait( $pos, 100000 );
 
-               $textClause = $this->getTextClause( $this->clusters );
+               $textClause = $this->getTextClause();
                $startId = 0;
                $endId = $dbr->selectField( 'text', 'MAX(old_id)', '', __METHOD__ );
                $rowsInserted = 0;
@@ -325,9 +325,9 @@ class TrackBlobs {
                        $lbFactory = MediaWikiServices::getInstance()->getDBLoadBalancerFactory();
                        $lb = $lbFactory->getExternalLB( $cluster );
                        try {
-                               $extDB = $lb->getConnection( DB_REPLICA );
+                               $extDB = $lb->getMaintenanceConnectionRef( DB_REPLICA );
                        } catch ( DBConnectionError $e ) {
-                               if ( strpos( $e->error, 'Unknown database' ) !== false ) {
+                               if ( strpos( $e->getMessage(), 'Unknown database' ) !== false ) {
                                        echo "No database on $cluster\n";
                                } else {
                                        echo "Error on $cluster: " . $e->getMessage() . "\n";
@@ -362,8 +362,8 @@ class TrackBlobs {
 
                                foreach ( $res as $row ) {
                                        gmp_setbit( $actualBlobs, $row->blob_id );
+                                       $startId = $row->blob_id;
                                }
-                               $startId = $row->blob_id;
 
                                ++$batchesDone;
                                if ( $batchesDone >= $this->reportingInterval ) {
index 9a7b9e8..133ba0f 100644 (file)
@@ -266,11 +266,6 @@ return [
                'scripts' => 'resources/src/jquery/jquery.mw-jump.js',
                'targets' => [ 'desktop', 'mobile' ],
        ],
-       'jquery.qunit' => [
-               'scripts' => 'resources/lib/qunitjs/qunit.js',
-               'styles' => 'resources/lib/qunitjs/qunit.css',
-               'targets' => [ 'desktop', 'mobile' ],
-       ],
        'jquery.spinner' => [
                'scripts' => 'resources/src/jquery.spinner/spinner.js',
                'styles' => 'resources/src/jquery.spinner/spinner.less',
@@ -768,10 +763,6 @@ return [
        ],
 
        /* MediaWiki */
-       'mediawiki.apihelp' => [
-               'styles' => 'resources/src/mediawiki.apihelp.css',
-               'targets' => [ 'desktop' ],
-       ],
        'mediawiki.template' => [
                'scripts' => 'resources/src/mediawiki.template.js',
                'targets' => [ 'desktop', 'mobile' ],
@@ -790,7 +781,10 @@ return [
                'dependencies' => 'mediawiki.template',
        ],
        'mediawiki.apipretty' => [
-               'styles' => 'resources/src/mediawiki.apipretty.css',
+               'styles' => [
+                       'resources/src/mediawiki.apipretty/apipretty.css',
+                       'resources/src/mediawiki.apipretty/apihelp.css',
+               ],
                'targets' => [ 'desktop', 'mobile' ],
        ],
        'mediawiki.api' => [
@@ -1159,6 +1153,7 @@ return [
                        'upload-form-label-usage-filename',
                        'action-upload',
                        'apierror-mustbeloggedin',
+                       'apierror-permissiondenied',
                        'badaccess-groups',
                        'apierror-timeout',
                        'apierror-offline',
@@ -1856,6 +1851,7 @@ return [
                        'styles/mw.rcfilters.ui.RclToOrFromWidget.less',
                        'styles/mw.rcfilters.ui.RclTargetPageWidget.less',
                        'styles/mw.rcfilters.ui.WatchlistTopSectionWidget.less',
+                       'styles/mw.rcfilters.ui.FilterTagMultiselectWidgetMobile.less'
                ],
                'skinStyles' => [
                        'vector' => [
@@ -1903,6 +1899,7 @@ return [
                        'rcfilters-clear-all-filters',
                        'rcfilters-show-new-changes',
                        'rcfilters-search-placeholder',
+                       'rcfilters-search-placeholder-mobile',
                        'rcfilters-invalid-filter',
                        'rcfilters-empty-filter',
                        'rcfilters-filterlist-title',
@@ -1964,6 +1961,7 @@ return [
                        'oojs-ui.styles.icons-interactions',
                        'oojs-ui.styles.icons-layout',
                        'oojs-ui.styles.icons-media',
+                       'oojs-ui-windows.icons'
                ],
        ],
        'mediawiki.interface.helpers.styles' => [
diff --git a/resources/src/mediawiki.apihelp.css b/resources/src/mediawiki.apihelp.css
deleted file mode 100644 (file)
index d1f32ab..0000000
+++ /dev/null
@@ -1,109 +0,0 @@
-/* stylelint-disable selector-class-pattern */
-
-.apihelp-header {
-       clear: both;
-       margin-bottom: 0.1em;
-}
-
-.apihelp-header.apihelp-module-name {
-       /*
-        * This element is explicitly set to dir="ltr" in HTML.
-        * Set explicit alignment so that CSSJanus will flip it to "right";
-        * otherwise the alignment will be automatically set to "left" according
-        * to the element's direction, and this will have an inconsistent look.
-        */
-       text-align: left;
-}
-
-div.apihelp-linktrail {
-       font-size: smaller;
-}
-
-.apihelp-block {
-       margin-top: 0.5em;
-}
-
-.apihelp-block-head {
-       font-weight: bold;
-}
-
-.apihelp-flags {
-       font-size: smaller;
-       float: right;
-       border: 1px solid #000;
-       padding: 0.25em;
-       width: 20em;
-}
-
-.apihelp-deprecated,
-.apihelp-flag-deprecated,
-.apihelp-flag-internal strong {
-       font-weight: bold;
-       color: #d33;
-}
-
-.apihelp-deprecated-value {
-       text-decoration: line-through;
-}
-
-.apihelp-unknown {
-       color: #72777d;
-}
-
-.apihelp-empty {
-       color: #72777d;
-}
-
-.apihelp-help-urls ul {
-       list-style-image: none;
-       list-style-type: none;
-       margin-left: 0;
-}
-
-.apihelp-parameters dl,
-.apihelp-examples dl,
-.apihelp-permissions dl {
-       margin-left: 2em;
-}
-
-.apihelp-parameters dt {
-       float: left;
-       clear: left;
-       min-width: 10em;
-       white-space: nowrap;
-       line-height: 1.5em;
-}
-
-.apihelp-parameters dt:after {
-       content: ':\A0';
-}
-
-.apihelp-parameters dd {
-       margin: 0 0 0.5em 10em;
-       line-height: 1.5em;
-}
-
-.apihelp-parameters dd p:first-child {
-       margin-top: 0;
-}
-
-.apihelp-parameters dd.info {
-       margin-left: 12em;
-       text-indent: -2em;
-}
-
-.apihelp-examples dt {
-       font-weight: normal;
-}
-
-.api-main-links {
-       text-align: center;
-}
-
-.api-main-links ul:before {
-       content: '[';
-}
-
-.api-main-links ul:after {
-       content: ']';
-}
diff --git a/resources/src/mediawiki.apipretty.css b/resources/src/mediawiki.apipretty.css
deleted file mode 100644 (file)
index 3e921f4..0000000
+++ /dev/null
@@ -1,13 +0,0 @@
-/* stylelint-disable selector-class-pattern */
-
-.mw-special-ApiHelp h1.firstHeading {
-       display: none;
-}
-
-.api-pretty-header {
-       font-size: small;
-}
-
-.api-pretty-content {
-       white-space: pre-wrap;
-}
diff --git a/resources/src/mediawiki.apipretty/apihelp.css b/resources/src/mediawiki.apipretty/apihelp.css
new file mode 100644 (file)
index 0000000..d1f32ab
--- /dev/null
@@ -0,0 +1,109 @@
+/* stylelint-disable selector-class-pattern */
+
+.apihelp-header {
+       clear: both;
+       margin-bottom: 0.1em;
+}
+
+.apihelp-header.apihelp-module-name {
+       /*
+        * This element is explicitly set to dir="ltr" in HTML.
+        * Set explicit alignment so that CSSJanus will flip it to "right";
+        * otherwise the alignment will be automatically set to "left" according
+        * to the element's direction, and this will have an inconsistent look.
+        */
+       text-align: left;
+}
+
+div.apihelp-linktrail {
+       font-size: smaller;
+}
+
+.apihelp-block {
+       margin-top: 0.5em;
+}
+
+.apihelp-block-head {
+       font-weight: bold;
+}
+
+.apihelp-flags {
+       font-size: smaller;
+       float: right;
+       border: 1px solid #000;
+       padding: 0.25em;
+       width: 20em;
+}
+
+.apihelp-deprecated,
+.apihelp-flag-deprecated,
+.apihelp-flag-internal strong {
+       font-weight: bold;
+       color: #d33;
+}
+
+.apihelp-deprecated-value {
+       text-decoration: line-through;
+}
+
+.apihelp-unknown {
+       color: #72777d;
+}
+
+.apihelp-empty {
+       color: #72777d;
+}
+
+.apihelp-help-urls ul {
+       list-style-image: none;
+       list-style-type: none;
+       margin-left: 0;
+}
+
+.apihelp-parameters dl,
+.apihelp-examples dl,
+.apihelp-permissions dl {
+       margin-left: 2em;
+}
+
+.apihelp-parameters dt {
+       float: left;
+       clear: left;
+       min-width: 10em;
+       white-space: nowrap;
+       line-height: 1.5em;
+}
+
+.apihelp-parameters dt:after {
+       content: ':\A0';
+}
+
+.apihelp-parameters dd {
+       margin: 0 0 0.5em 10em;
+       line-height: 1.5em;
+}
+
+.apihelp-parameters dd p:first-child {
+       margin-top: 0;
+}
+
+.apihelp-parameters dd.info {
+       margin-left: 12em;
+       text-indent: -2em;
+}
+
+.apihelp-examples dt {
+       font-weight: normal;
+}
+
+.api-main-links {
+       text-align: center;
+}
+
+.api-main-links ul:before {
+       content: '[';
+}
+
+.api-main-links ul:after {
+       content: ']';
+}
diff --git a/resources/src/mediawiki.apipretty/apipretty.css b/resources/src/mediawiki.apipretty/apipretty.css
new file mode 100644 (file)
index 0000000..3e921f4
--- /dev/null
@@ -0,0 +1,13 @@
+/* stylelint-disable selector-class-pattern */
+
+.mw-special-ApiHelp h1.firstHeading {
+       display: none;
+}
+
+.api-pretty-header {
+       font-size: small;
+}
+
+.api-pretty-content {
+       white-space: pre-wrap;
+}
index a500226..7f4af5b 100644 (file)
                        if ( stats.enabled ) {
                                $.extend( stats, mw.loader.store.stats );
                                try {
-                                       raw = localStorage.getItem( mw.loader.store.getStoreKey() );
+                                       raw = localStorage.getItem( mw.loader.store.key );
                                        stats.totalSizeInBytes = byteLength( raw );
                                        stats.totalSize = humanSize( byteLength( raw ) );
                                } catch ( e ) {}
index 4684ea2..cce3061 100644 (file)
        background-image: e( '/* @embed */' ) url( @url );
 }
 
-.vertical-gradient( @startColor: gray, @endColor: white, @startPos: 0, @endPos: 100% ) {
-       background-color: @endColor;
-       background-image: -webkit-linear-gradient( top, @startColor @startPos, @endColor @endPos ); // Safari 5.1+, Chrome 10+
-       background-image: -moz-linear-gradient( top, @startColor @startPos, @endColor @endPos ); // Firefox 3.6+
-       background-image: linear-gradient( @startColor @startPos, @endColor @endPos ); // Standard
+.horizontal-gradient( @startColor: #808080, @endColor: #fff, @startPos: 0, @endPos: 100% ) {
+       background-color: average( @startColor, @endColor );
+       background-image: -webkit-gradient( linear, left top, right top, color-stop( @startPos, @startColor ), color-stop( @endPos, @endColor ) );
+       background-image: -webkit-linear-gradient( left, @startColor @startPos, @endColor @endPos );
+       background-image: -moz-linear-gradient( left, @startColor @startPos, @endColor @endPos );
+       background-image: linear-gradient( to right, @startColor @startPos, @endColor @endPos );
+}
+
+.vertical-gradient( @startColor: #808080, @endColor: #fff, @startPos: 0, @endPos: 100% ) {
+       background-color: average( @startColor, @endColor );
+       background-image: -webkit-gradient( linear, right top, right bottom, color-stop( @startPos, @startColor ), color-stop( @endPos, @endColor ) );
+       background-image: -webkit-linear-gradient( top, @startColor @startPos, @endColor @endPos );
+       background-image: -moz-linear-gradient( top, @startColor @startPos, @endColor @endPos );
+       background-image: linear-gradient( to bottom, @startColor @startPos, @endColor @endPos );
 }
 
 // SVG support using a transparent gradient to guarantee cross-browser
index 87f257b..4338b0b 100644 (file)
@@ -28,7 +28,6 @@
        }
 
        &-results {
-               width: 35em;
                margin: 5em auto;
 
                &-noresult,
 .mw-rcfilters-highlight-color-c1.mw-rcfilters-highlight-color-c2.mw-rcfilters-highlight-color-c3.mw-rcfilters-highlight-color-c4.mw-rcfilters-highlight-color-c5 {
        .highlight-results( tint( mix( @highlight-c1, mix( @highlight-c2, mix( @highlight-c3, average( @highlight-c4, @highlight-c5 ), 20% ), 20% ), 20% ), 15% ) );
 }
+
+@media screen and ( min-width: @width-breakpoint-tablet ) {
+       // center conflict message
+       // e.g. Special:RecentChanges?goodfaith=maybebad&hidepageedits=1&hidenewpages=1&hidecategorization=1&hideWikibase=1&limit=50&days=0.0833&enhanced=1&urlversion=2
+       // More context in https://phabricator.wikimedia.org/T223363#5374874
+       .mw-rcfilters-ui-changesListWrapperWidget {
+               &-results {
+                       width: 35em;
+               }
+       }
+}
diff --git a/resources/src/mediawiki.rcfilters/styles/mw.rcfilters.ui.FilterTagMultiselectWidgetMobile.less b/resources/src/mediawiki.rcfilters/styles/mw.rcfilters.ui.FilterTagMultiselectWidgetMobile.less
new file mode 100644 (file)
index 0000000..d3b4a2b
--- /dev/null
@@ -0,0 +1,35 @@
+@import 'mediawiki.mixins';
+@import 'mediawiki.ui/variables';
+
+.mw-rcfilters-ui-filterTagMultiselectWidget-mobile {
+
+       // Them mobile version of the search input is meant to function
+       // as a button, so styles are modified to that effect. See T224655 for details.
+       .oo-ui-tagMultiselectWidget-input {
+               & .oo-ui-iconElement-icon {
+                       opacity: 1;
+                       cursor: pointer;
+               }
+
+               &.oo-ui-textInputWidget input[ readonly ] {
+                       background-color: @background-color-base;
+                       font-weight: bold;
+                       cursor: pointer;
+                       .mixin-placeholder( { color: @colorText; } );
+               }
+       }
+
+       .mw-rcfilters-ui-filterTagMultiselectWidget-mobile-view {
+               width: 100%;
+               margin-top: -1px;
+
+               & .oo-ui-buttonOptionWidget {
+                       width: 50%;
+
+                       & .oo-ui-buttonElement-button {
+                               width: 100%;
+                               text-align: initial;
+                       }
+               }
+       }
+}
index 949980d..4e7d02d 100644 (file)
        }
 
        &-bottom {
-               .flex-display;
-               .flex;
+               .flex-display();
+               .flex();
+               flex-wrap: wrap;
                margin-top: 1em;
        }
+
+       &-bottom-mobile {
+               .oo-ui-buttonElement {
+                       margin-bottom: 1em;
+
+                       &-button {
+                               text-align: left;
+                       }
+               }
+
+               .mw-rcfilters-ui-changesLimitAndDateButtonWidget {
+                       order: 1;
+               }
+
+               .mw-rcfilters-ui-liveUpdateButtonWidget {
+                       order: 2;
+               }
+
+               .mw-rcfilters-ui-filterWrapperWidget-showNewChanges {
+                       order: 3;
+                       font-size: 0.85em;
+
+                       & > a {
+                               white-space: normal;
+                               /* stylelint-disable-next-line */
+                               padding-top: 0 !important; //overrides .oo-ui-buttonElement-button
+                       }
+               }
+       }
 }
index a0c0d80..ce869e3 100644 (file)
@@ -42,13 +42,14 @@ ChangesLimitPopupWidget = function MwRcfiltersUiChangesLimitPopupWidget( limitMo
                .addClass( 'mw-rcfilters-ui-changesLimitPopupWidget' )
                .append(
                        this.valuePicker.$element,
-                       new OO.ui.FieldLayout(
-                               this.groupByPageCheckbox,
-                               {
-                                       align: 'inline',
-                                       label: mw.msg( 'rcfilters-group-results-by-page' )
-                               }
-                       ).$element
+                       OO.ui.isMobile() ? undefined :
+                               new OO.ui.FieldLayout(
+                                       this.groupByPageCheckbox,
+                                       {
+                                               align: 'inline',
+                                               label: mw.msg( 'rcfilters-group-results-by-page' )
+                                       }
+                               ).$element
                );
 };
 
index 3429590..a50cd0e 100644 (file)
@@ -40,6 +40,7 @@ FilterTagMultiselectWidget = function MwRcfiltersUiFilterTagMultiselectWidget( c
        this.matchingQuery = null;
        this.currentView = this.model.getCurrentView();
        this.collapsed = false;
+       this.isMobile = config.isMobile;
 
        // Parent
        FilterTagMultiselectWidget.parent.call( this, $.extend( true, {
@@ -55,6 +56,8 @@ FilterTagMultiselectWidget = function MwRcfiltersUiFilterTagMultiselectWidget( c
                        filterFromInput: false,
                        hideWhenOutOfView: false,
                        hideOnChoose: false,
+                       // Only set width and footers for desktop
+                       isMobile: this.isMobile,
                        width: 650,
                        footers: [
                                {
@@ -81,9 +84,17 @@ FilterTagMultiselectWidget = function MwRcfiltersUiFilterTagMultiselectWidget( c
                                }
                        ]
                },
+               /**
+                * In the presence of an onscreen keyboard (i.e. isMobile) the filter input should act as a button
+                * rather than a text input. Mobile screens are too small to accommodate both an
+                * onscreen keyboard and a popup-menu, so readyOnly is set to disable the keyboard.
+                * A different icon and shorter message is used for mobile as well. (See T224655 for details).
+                */
                input: {
-                       icon: 'menu',
-                       placeholder: mw.msg( 'rcfilters-search-placeholder' )
+                       icon: this.isMobile ? 'funnel' : 'menu',
+                       placeholder: this.isMobile ? mw.msg( 'rcfilters-search-placeholder-mobile' ) : mw.msg( 'rcfilters-search-placeholder' ),
+                       readOnly: !!this.isMobile,
+                       classes: [ 'oo-ui-tagMultiselectWidget-input' ]
                }
        }, config ) );
 
@@ -148,11 +159,14 @@ FilterTagMultiselectWidget = function MwRcfiltersUiFilterTagMultiselectWidget( c
        this.model.connect( this, {
                initialize: 'onModelInitialize',
                update: 'onModelUpdate',
-               searchChange: 'onModelSearchChange',
+               searchChange: this.isMobile ? function () {} : 'onModelSearchChange',
                itemUpdate: 'onModelItemUpdate',
                highlightChange: 'onModelHighlightChange'
        } );
-       this.input.connect( this, { change: 'onInputChange' } );
+
+       if ( !this.isMobile ) {
+               this.input.connect( this, { change: 'onInputChange' } );
+       }
 
        // The filter list and button should appear side by side regardless of how
        // wide the button is; the button also changes its width depending
@@ -176,46 +190,10 @@ FilterTagMultiselectWidget = function MwRcfiltersUiFilterTagMultiselectWidget( c
        }
 
        // Add a selector at the right of the input
-       this.viewsSelectWidget = new OO.ui.ButtonSelectWidget( {
-               classes: [ 'mw-rcfilters-ui-filterTagMultiselectWidget-views-select-widget' ],
-               items: [
-                       new OO.ui.ButtonOptionWidget( {
-                               framed: false,
-                               data: 'namespaces',
-                               icon: 'article',
-                               label: mw.msg( 'namespaces' ),
-                               title: mw.msg( 'rcfilters-view-namespaces-tooltip' )
-                       } ),
-                       new OO.ui.ButtonOptionWidget( {
-                               framed: false,
-                               data: 'tags',
-                               icon: 'tag',
-                               label: mw.msg( 'tags-title' ),
-                               title: mw.msg( 'rcfilters-view-tags-tooltip' )
-                       } )
-               ]
-       } );
+       this.viewsSelectWidget = this.createViewsSelectWidget();
 
-       // Rearrange the UI so the select widget is at the right of the input
-       this.$element.append(
-               $( '<div>' )
-                       .addClass( 'mw-rcfilters-ui-table' )
-                       .append(
-                               $( '<div>' )
-                                       .addClass( 'mw-rcfilters-ui-row' )
-                                       .addClass( 'mw-rcfilters-ui-filterTagMultiselectWidget-views' )
-                                       .append(
-                                               $( '<div>' )
-                                                       .addClass( 'mw-rcfilters-ui-cell' )
-                                                       .addClass( 'mw-rcfilters-ui-filterTagMultiselectWidget-views-input' )
-                                                       .append( this.input.$element ),
-                                               $( '<div>' )
-                                                       .addClass( 'mw-rcfilters-ui-cell' )
-                                                       .addClass( 'mw-rcfilters-ui-filterTagMultiselectWidget-views-select' )
-                                                       .append( this.viewsSelectWidget.$element )
-                                       )
-                       )
-       );
+       // change the layout of the viewsSelectWidget
+       this.restructureViewsSelectWidget();
 
        // Event
        this.viewsSelectWidget.connect( this, { choose: 'onViewsSelectWidgetChoose' } );
@@ -258,6 +236,11 @@ FilterTagMultiselectWidget = function MwRcfiltersUiFilterTagMultiselectWidget( c
        this.$element
                .addClass( 'mw-rcfilters-ui-filterTagMultiselectWidget' );
 
+       if ( this.isMobile ) {
+               this.$element
+                       .addClass( 'mw-rcfilters-ui-filterTagMultiselectWidget-mobile' );
+       }
+
        this.reevaluateResetRestoreState();
 };
 
@@ -267,6 +250,78 @@ OO.inheritClass( FilterTagMultiselectWidget, OO.ui.MenuTagMultiselectWidget );
 
 /* Methods */
 
+/**
+ * Create a OOUI ButtonSelectWidget. The buttons are framed and have additional CSS
+ * classes applied on mobile.
+ * @return {OO.ui.ButtonSelectWidget}
+ */
+FilterTagMultiselectWidget.prototype.createViewsSelectWidget = function () {
+       return new OO.ui.ButtonSelectWidget( {
+               classes: this.isMobile ?
+                       [
+                               'mw-rcfilters-ui-table',
+                               'mw-rcfilters-ui-filterTagMultiselectWidget-mobile-view'
+                       ] :
+                       [
+                               'mw-rcfilters-ui-filterTagMultiselectWidget-views-select-widget'
+                       ],
+               items: [
+                       new OO.ui.ButtonOptionWidget( {
+                               framed: !!this.isMobile,
+                               data: 'namespaces',
+                               icon: 'article',
+                               label: mw.msg( 'namespaces' ),
+                               classes: this.isMobile ? [ 'mw-rcfilters-ui-cell' ] : []
+                       } ),
+                       new OO.ui.ButtonOptionWidget( {
+                               framed: !!this.isMobile,
+                               data: 'tags',
+                               icon: 'tag',
+                               label: mw.msg( 'tags-title' ),
+                               title: mw.msg( 'rcfilters-view-tags-tooltip' ),
+                               classes: this.isMobile ? [ 'mw-rcfilters-ui-cell' ] : []
+                       } )
+               ]
+       } );
+};
+
+/**
+ * Rearrange the DOM structure of the viewsSelectWiget so that on the namespace & tags buttons
+ * are at the right of the input on desktop, and below the input on mobile.
+ */
+FilterTagMultiselectWidget.prototype.restructureViewsSelectWidget = function () {
+       if ( this.isMobile ) {
+               // On mobile, append the search input and the extra buttons below the search input.
+               this.$element.append(
+                       $( '<div>' )
+                               .addClass( 'mw-rcfilters-ui-filterTagMultiselectWidget-views-input' )
+                               .append( this.input.$element )
+                               .append( this.viewsSelectWidget.$element )
+               );
+       } else {
+               // On desktop, rearrange the UI so the select widget is at the right of the input
+               this.$element.append(
+                       $( '<div>' )
+                               .addClass( 'mw-rcfilters-ui-table' )
+                               .append(
+                                       $( '<div>' )
+                                               .addClass( 'mw-rcfilters-ui-row' )
+                                               .addClass( 'mw-rcfilters-ui-filterTagMultiselectWidget-views' )
+                                               .append(
+                                                       $( '<div>' )
+                                                               .addClass( 'mw-rcfilters-ui-cell' )
+                                                               .addClass( 'mw-rcfilters-ui-filterTagMultiselectWidget-views-input' )
+                                                               .append( this.input.$element ),
+                                                       $( '<div>' )
+                                                               .addClass( 'mw-rcfilters-ui-cell' )
+                                                               .addClass( 'mw-rcfilters-ui-filterTagMultiselectWidget-views-select' )
+                                                               .append( this.viewsSelectWidget.$element )
+                                               )
+                               )
+               );
+       }
+};
+
 /**
  * Respond to view select widget choose event
  *
@@ -333,7 +388,9 @@ FilterTagMultiselectWidget.prototype.onMenuToggle = function ( isVisible ) {
        FilterTagMultiselectWidget.parent.prototype.onMenuToggle.call( this );
 
        if ( isVisible ) {
-               this.focus();
+               if ( !this.isMobile ) {
+                       this.focus();
+               }
 
                mw.hook( 'RcFilters.popup.open' ).fire();
 
@@ -360,21 +417,33 @@ FilterTagMultiselectWidget.prototype.onMenuToggle = function ( isVisible ) {
                this.blur();
        }
 
-       this.input.setIcon( isVisible ? 'search' : 'menu' );
+       if ( this.isMobile ) {
+               this.input.setIcon( isVisible ? 'close' : 'funnel' );
+       } else {
+               this.input.setIcon( isVisible ? 'search' : 'menu' );
+       }
 };
 
 /**
  * @inheritdoc
  */
 FilterTagMultiselectWidget.prototype.onInputFocus = function () {
-       // Parent
-       FilterTagMultiselectWidget.parent.prototype.onInputFocus.call( this );
+       var scrollToElement = this.isMobile ? this.input.$input : this.$element;
+
+       // treat the input as a menu toggle rather than a text field on mobile
+       if ( this.isMobile ) {
+               this.input.$input.trigger( 'blur' );
+               this.getMenu().toggle();
+       } else {
+               // Parent
+               FilterTagMultiselectWidget.parent.prototype.onInputFocus.call( this );
+       }
 
        // Only scroll to top of the viewport if:
        // - The widget is more than 20px from the top
        // - The widget is not above the top of the viewport (do not scroll downwards)
        //   (This isn't represented because >20 is, anyways and always, bigger than 0)
-       this.scrollToTop( this.$element, 0, { min: 20, max: Infinity } );
+       this.scrollToTop( scrollToElement, 0, { min: 20, max: Infinity } );
 };
 
 /**
@@ -525,7 +594,10 @@ FilterTagMultiselectWidget.prototype.onMenuChoose = function ( item ) {
        // Select the tag if it exists, or reset selection otherwise
        this.selectTag( this.findItemFromData( item.model.getName() ) );
 
-       this.focus();
+       if ( !this.isMobile ) {
+               this.focus();
+       }
+
 };
 
 /**
index a0f098c..5893a6c 100644 (file)
@@ -47,7 +47,8 @@ FilterWrapperWidget = function MwRcfiltersUiFilterWrapperWidget(
                {
                        $overlay: this.$overlay,
                        collapsed: config.collapsed,
-                       $wrapper: this.$wrapper
+                       $wrapper: this.$wrapper,
+                       isMobile: OO.ui.isMobile()
                }
        );
 
@@ -82,7 +83,11 @@ FilterWrapperWidget = function MwRcfiltersUiFilterWrapperWidget(
                .addClass( 'mw-rcfilters-ui-filterWrapperWidget-top' );
 
        $bottom = $( '<div>' )
-               .addClass( 'mw-rcfilters-ui-filterWrapperWidget-bottom' )
+               .addClass( OO.ui.isMobile() ?
+                       'mw-rcfilters-ui-filterWrapperWidget-bottom ' +
+                       'mw-rcfilters-ui-filterWrapperWidget-bottom-mobile' :
+                       'mw-rcfilters-ui-filterWrapperWidget-bottom'
+               )
                .append(
                        this.showNewChangesLink.$element,
                        this.numChangesAndDateWidget.$element
index 1e75020..465d5b9 100644 (file)
@@ -14,6 +14,8 @@ var FilterMenuHeaderWidget = require( './FilterMenuHeaderWidget.js' ),
  * @param {mw.rcfilters.Controller} controller Controller
  * @param {mw.rcfilters.dm.FiltersViewModel} model View model
  * @param {Object} [config] Configuration object
+ * @cfg {boolean} [isMobile] a boolean flag determining whether the menu
+ * should display a header or not (the header is omitted on mobile).
  * @cfg {jQuery} [$overlay] A jQuery object serving as overlay for popups
  * @cfg {Object[]} [footers] An array of objects defining the footers for
  *  this menu, with a definition whether they appear per specific views.
@@ -46,7 +48,7 @@ MenuSelectWidget = function MwRcfiltersUiMenuSelectWidget( controller, model, co
        // Parent
        MenuSelectWidget.parent.call( this, $.extend( config, {
                $autoCloseIgnore: this.$overlay,
-               width: 650,
+               width: config.isMobile ? undefined : 650,
                // Our filtering is done through the model
                filterFromInput: false
        } ) );
@@ -54,16 +56,21 @@ MenuSelectWidget = function MwRcfiltersUiMenuSelectWidget( controller, model, co
                $( '<div>' )
                        .addClass( 'mw-rcfilters-ui-menuSelectWidget-group' )
        );
-       this.setClippableElement( this.$body );
-       this.setClippableContainer( this.$element );
-
-       header = new FilterMenuHeaderWidget(
-               this.controller,
-               this.model,
-               {
-                       $overlay: this.$overlay
-               }
-       );
+
+       if ( !config.isMobile ) {
+               // When hiding the header (i.e. mobile mode) avoid problems
+               // with clippable and the menu's fixed width.
+               this.setClippableElement( this.$body );
+               this.setClippableContainer( this.$element );
+
+               header = new FilterMenuHeaderWidget(
+                       this.controller,
+                       this.model,
+                       {
+                               $overlay: this.$overlay
+                       }
+               );
+       }
 
        this.noResults = new OO.ui.LabelWidget( {
                label: mw.msg( 'rcfilters-filterlist-noresults' ),
@@ -79,7 +86,7 @@ MenuSelectWidget = function MwRcfiltersUiMenuSelectWidget( controller, model, co
        // Initialization
        this.$element
                .addClass( 'mw-rcfilters-ui-menuSelectWidget' )
-               .append( header.$element )
+               .append( config.isMobile ? undefined : header.$element )
                .append(
                        this.$body
                                .append( this.$group, this.noResults.$element )
@@ -87,7 +94,7 @@ MenuSelectWidget = function MwRcfiltersUiMenuSelectWidget( controller, model, co
 
        // Append all footers; we will control their visibility
        // based on view
-       config.footers = config.footers || [];
+       config.footers = config.isMobile ? [] : config.footers || [];
        config.footers.forEach( function ( footerData ) {
                var isSticky = footerData.sticky === undefined ? true : !!footerData.sticky,
                        adjustedData = {
index 395fb8b..c2c5960 100644 (file)
                                // Can't, sorry.
                        },
                        apiCheckValid: function () {
-                               var ok = this.getValue() !== null || suppressErrors;
-                               this.setIcon( ok ? null : 'alert' );
+                               var ok = this.getValue() !== null && this.getValue() !== undefined || suppressErrors;
+                               this.info.setIcon( ok ? null : 'alert' );
                                this.setTitle( ok ? '' : mw.message( 'apisandbox-alert-field' ).plain() );
                                return $.Deferred().resolve( ok ).promise();
                        }
index 65e7eb7..56419ae 100644 (file)
@@ -54,7 +54,7 @@
 
                // Events
                this.button.connect( this, { click: 'onButtonClick' } );
-               this.textInput.$input.on( 'click', this.onInputClick.bind( this ) );
+               this.textInput.$input.on( 'focus', this.onInputFocus.bind( this ) );
 
                this.$element.addClass( 'mw-widget-copyTextLayout' );
        };
        };
 
        /**
-        * Handle button click events
+        * Handle text widget focus events
         */
-       mw.widgets.CopyTextLayout.prototype.onInputClick = function () {
-               this.selectText();
+       mw.widgets.CopyTextLayout.prototype.onInputFocus = function () {
+               if ( !this.selecting ) {
+                       this.selectText();
+               }
        };
 
        /**
                        scrollTop = input.scrollTop,
                        scrollLeft = input.scrollLeft;
 
+               this.selecting = true;
                this.textInput.select();
+               this.selecting = false;
 
                // Restore scroll position
                input.scrollTop = scrollTop;
index f61255a..4bb4d39 100644 (file)
         * @cfg {Object} [textinput] Config for the text input
         * @cfg {boolean} [or=false] Config for whether the widget is dropdown AND input
         *                           or dropdown OR input
+        * @cfg {boolean} [required=false] Config for whether input is required
         */
        mw.widgets.SelectWithInputWidget = function MwWidgetsSelectWithInputWidget( config ) {
                // Config initialization
-               config = $.extend( { or: false }, config );
+               config = $.extend( { or: false, required: false }, config );
 
                // Properties
                this.textinput = new OO.ui.TextInputWidget( config.textinput );
                this.dropdowninput = new OO.ui.DropdownInputWidget( config.dropdowninput );
                this.or = config.or;
+               this.required = config.required;
 
                // Events
                this.dropdowninput.on( 'change', this.onChange.bind( this ) );
                // is required. However, validity is not checked for disabled fields, as these are not
                // submitted with the form. So we should also disable fields when hiding them.
                this.textinput.setDisabled( textinputIsHidden || disabled );
+               // If the widget is required, set the text field as required, but only if the widget is visible.
+               if ( this.required ) {
+                       this.textinput.setRequired( !this.textinput.isDisabled() );
+               }
        };
 
        /**
index 2081bfe..ace92f0 100644 (file)
                 *
                 * @property {mw.Map} config
                 */
-               // Dummy placeholder later assigned in ResourceLoaderStartUpModule
-               config: null,
+               config: new Map( $VARS.wgLegacyJavaScriptGlobals ),
 
                /**
                 * Empty object for third-party libraries, for cases where you don't
                                         * @return {Object} Module store contents.
                                         */
                                        toJSON: function () {
-                                               return { items: mw.loader.store.items, vary: mw.loader.store.getVary() };
+                                               return { items: mw.loader.store.items, vary: mw.loader.store.vary };
                                        },
 
                                        /**
-                                        * Get the localStorage key for the entire module store. The key references
+                                        * The localStorage key for the entire module store. The key references
                                         * $wgDBname to prevent clashes between wikis which share a common host.
                                         *
-                                        * @return {string} localStorage item key
+                                        * @property {string}
                                         */
-                                       getStoreKey: function () {
-                                               return $VARS.storeKey;
-                                       },
+                                       key: $VARS.storeKey,
 
                                        /**
-                                        * Get a key on which to vary the module cache.
+                                        * A string containing various factors on which to the module cache should vary.
                                         *
-                                        * @return {string} String of concatenated vary conditions.
+                                        * @property {string}
                                         */
-                                       getVary: function () {
-                                               return $VARS.storeVary;
-                                       },
+                                       vary: $VARS.storeVary,
 
                                        /**
                                         * Initialize the store.
                                                }
 
                                                if (
+                                                       !$VARS.storeEnabled ||
+
                                                        // Disabled because localStorage quotas are tight and (in Firefox's case)
                                                        // shared by multiple origins.
                                                        // See T66721, and <https://bugzilla.mozilla.org/show_bug.cgi?id=1064466>.
-                                                       /Firefox/.test( navigator.userAgent ) ||
-
-                                                       // Disabled by configuration.
-                                                       !mw.config.get( 'wgResourceLoaderStorageEnabled' ) ||
-
-                                                       // Disable module store in debug mode context
-                                                       mw.config.get( 'debug' )
+                                                       /Firefox/.test( navigator.userAgent )
                                                ) {
                                                        // Clear any previous store to free up space. (T66721)
                                                        this.clear();
 
                                                try {
                                                        // This a string we stored, or `null` if the key does not (yet) exist.
-                                                       raw = localStorage.getItem( this.getStoreKey() );
+                                                       raw = localStorage.getItem( this.key );
                                                        // If we get here, localStorage is available; mark enabled
                                                        this.enabled = true;
                                                        // If null, JSON.parse() will cast to string and re-parse, still null.
                                                        data = JSON.parse( raw );
-                                                       if ( data && typeof data.items === 'object' && data.vary === this.getVary() ) {
+                                                       if ( data && typeof data.items === 'object' && data.vary === this.vary ) {
                                                                this.items = data.items;
                                                                return;
                                                        }
                                                //    The store was enabled, and `items` starts fresh.
                                                //
                                                // 2. localStorage contained parseable data under our store key,
-                                               //    but it's not applicable to our current context (see getVary).
+                                               //    but it's not applicable to our current context (see #vary).
                                                //    The store was enabled, and `items` starts fresh.
                                                //
                                                // 3. JSON.parse threw (localStorage contained corrupt data).
                                        clear: function () {
                                                this.items = {};
                                                try {
-                                                       localStorage.removeItem( this.getStoreKey() );
+                                                       localStorage.removeItem( this.key );
                                                } catch ( e ) {}
                                        },
 
                                                                mw.loader.store.set( mw.loader.store.queue.shift() );
                                                        }
 
-                                                       key = mw.loader.store.getStoreKey();
+                                                       key = mw.loader.store.key;
                                                        try {
                                                                // Replacing the content of the module store might fail if the new
                                                                // contents would exceed the browser's localStorage size limit. To
index da048ff..06c6737 100644 (file)
@@ -113,7 +113,6 @@ if ( !isCompatible( navigator.userAgent ) ) {
         */
        ( function () {
                /* global mw */
-               mw.config = new mw.Map( $VARS.wgLegacyJavaScriptGlobals );
 
                $CODE.registrations();
 
index 8a98217..8108639 100644 (file)
@@ -1161,7 +1161,7 @@ class PermissionManagerTest extends MediaWikiLangTestCase {
                        'auto' => true,
                        'expiry' => 0
                ] );
-               $this->user->mBlock->mTimestamp = 0;
+               $this->user->mBlock->setTimestamp( 0 );
                $this->assertEquals( [ [ 'autoblockedtext',
                        "[[User:Useruser|\u{202A}Useruser\u{202C}]]", 'no reason given', '127.0.0.1',
                        "\u{202A}Useruser\u{202C}", null, 'infinite', '127.0.8.1',
index b29d333..c2917b6 100644 (file)
@@ -265,6 +265,7 @@ class ApiBlockTest extends ApiTestCase {
                        'partial' => true,
                        'pagerestrictions' => $title,
                        'namespacerestrictions' => $namespace,
+                       'allowusertalk' => true,
                ] );
 
                $block = DatabaseBlock::newFromTarget( $this->mUser->getName() );
index 8faaeda..43fbee8 100644 (file)
@@ -12,8 +12,7 @@ class SearchUpdateTest extends MediaWikiTestCase {
 
        protected function setUp() {
                parent::setUp();
-               $this->setMwGlobals( 'wgSearchType', 'MockSearch' );
-               $this->su = new SearchUpdate( 0, "" );
+               $this->su = new SearchUpdate( 0, Title::newMainPage() );
        }
 
        public function updateText( $text ) {
index 9e4a5d7..329c642 100644 (file)
@@ -1070,7 +1070,7 @@ class WANObjectCacheTest extends PHPUnit\Framework\TestCase {
         * @covers WANObjectCache::getWithSetCallback()
         * @covers WANObjectCache::fetchOrRegenerate()
         */
-       public function testBusyValue() {
+       public function testBusyValueBasic() {
                $cache = $this->cache;
                $key = wfRandomString();
                $value = wfRandomString();
@@ -1080,7 +1080,7 @@ class WANObjectCacheTest extends PHPUnit\Framework\TestCase {
                $cache->setMockTime( $mockWallClock );
 
                $calls = 0;
-               $func = function () use ( &$calls, $value, $cache, $key ) {
+               $func = function () use ( &$calls, $value ) {
                        ++$calls;
                        return $value;
                };
@@ -1125,6 +1125,52 @@ class WANObjectCacheTest extends PHPUnit\Framework\TestCase {
                $this->assertEquals( 3, $calls, 'Callback was not used; used interim' );
        }
 
+       public function getBusyValues_Provider() {
+               $hash = new HashBagOStuff( [] );
+
+               return [
+                       [
+                               function () {
+                                       return "Saint Oliver Plunckett";
+                               },
+                               'Saint Oliver Plunckett'
+                       ],
+                       [ 'strlen', 'strlen' ],
+                       [ 'WANObjectCache::newEmpty', 'WANObjectCache::newEmpty' ],
+                       [ [ 'WANObjectCache', 'newEmpty' ], [ 'WANObjectCache', 'newEmpty' ] ],
+                       [ [ $hash, 'getLastError' ], [ $hash, 'getLastError' ] ],
+                       [ [ 1, 2, 3 ], [ 1, 2, 3 ] ]
+               ];
+       }
+
+       /**
+        * @covers WANObjectCache::getWithSetCallback()
+        * @covers WANObjectCache::fetchOrRegenerate()
+        * @dataProvider getBusyValues_Provider
+        * @param mixed $busyValue
+        * @param mixed $expected
+        */
+       public function testBusyValueTypes( $busyValue, $expected ) {
+               $cache = $this->cache;
+               $key = wfRandomString();
+
+               $mockWallClock = 1549343530.2053;
+               $cache->setMockTime( $mockWallClock );
+
+               $calls = 0;
+               $func = function () use ( &$calls ) {
+                       ++$calls;
+                       return 418;
+               };
+
+               // Acquire a lock to verify that getWithSetCallback uses busyValue properly
+               $this->internalCache->add( 'WANCache:m:' . $key, 1, 0 );
+
+               $ret = $cache->getWithSetCallback( $key, 30, $func, [ 'busyValue' => $busyValue ] );
+               $this->assertSame( $expected, $ret, 'busyValue used as expected' );
+               $this->assertSame( 0, $calls, 'busyValue was used' );
+       }
+
        /**
         * @covers WANObjectCache::getMulti()
         */
index 482ab4b..a775dd7 100644 (file)
@@ -568,62 +568,74 @@ class DatabaseTest extends PHPUnit\Framework\TestCase {
        public function testFlagSetting() {
                $db = $this->db;
                $origTrx = $db->getFlag( DBO_TRX );
-               $origSsl = $db->getFlag( DBO_SSL );
+               $origNoBuffer = $db->getFlag( DBO_NOBUFFER );
 
                $origTrx
                        ? $db->clearFlag( DBO_TRX, $db::REMEMBER_PRIOR )
                        : $db->setFlag( DBO_TRX, $db::REMEMBER_PRIOR );
                $this->assertEquals( !$origTrx, $db->getFlag( DBO_TRX ) );
 
-               $origSsl
-                       ? $db->clearFlag( DBO_SSL, $db::REMEMBER_PRIOR )
-                       : $db->setFlag( DBO_SSL, $db::REMEMBER_PRIOR );
-               $this->assertEquals( !$origSsl, $db->getFlag( DBO_SSL ) );
+               $origNoBuffer
+                       ? $db->clearFlag( DBO_NOBUFFER, $db::REMEMBER_PRIOR )
+                       : $db->setFlag( DBO_NOBUFFER, $db::REMEMBER_PRIOR );
+               $this->assertEquals( !$origNoBuffer, $db->getFlag( DBO_NOBUFFER ) );
 
                $db->restoreFlags( $db::RESTORE_INITIAL );
                $this->assertEquals( $origTrx, $db->getFlag( DBO_TRX ) );
-               $this->assertEquals( $origSsl, $db->getFlag( DBO_SSL ) );
+               $this->assertEquals( $origNoBuffer, $db->getFlag( DBO_NOBUFFER ) );
 
                $origTrx
                        ? $db->clearFlag( DBO_TRX, $db::REMEMBER_PRIOR )
                        : $db->setFlag( DBO_TRX, $db::REMEMBER_PRIOR );
-               $origSsl
-                       ? $db->clearFlag( DBO_SSL, $db::REMEMBER_PRIOR )
-                       : $db->setFlag( DBO_SSL, $db::REMEMBER_PRIOR );
+               $origNoBuffer
+                       ? $db->clearFlag( DBO_NOBUFFER, $db::REMEMBER_PRIOR )
+                       : $db->setFlag( DBO_NOBUFFER, $db::REMEMBER_PRIOR );
 
                $db->restoreFlags();
-               $this->assertEquals( $origSsl, $db->getFlag( DBO_SSL ) );
+               $this->assertEquals( $origNoBuffer, $db->getFlag( DBO_NOBUFFER ) );
                $this->assertEquals( !$origTrx, $db->getFlag( DBO_TRX ) );
 
                $db->restoreFlags();
-               $this->assertEquals( $origSsl, $db->getFlag( DBO_SSL ) );
+               $this->assertEquals( $origNoBuffer, $db->getFlag( DBO_NOBUFFER ) );
                $this->assertEquals( $origTrx, $db->getFlag( DBO_TRX ) );
        }
 
+       public function provideImmutableDBOFlags() {
+               return [
+                       [ Database::DBO_IGNORE ],
+                       [ Database::DBO_DEFAULT ],
+                       [ Database::DBO_PERSISTENT ]
+               ];
+       }
+
        /**
-        * @expectedException UnexpectedValueException
+        * @expectedException DBUnexpectedError
         * @covers Wikimedia\Rdbms\Database::setFlag
+        * @dataProvider provideImmutableDBOFlags
+        * @param int $flag
         */
-       public function testDBOIgnoreSet() {
+       public function testDBOCannotSet( $flag ) {
                $db = $this->getMockBuilder( DatabaseMysqli::class )
                        ->disableOriginalConstructor()
                        ->setMethods( null )
                        ->getMock();
 
-               $db->setFlag( Database::DBO_IGNORE );
+               $db->setFlag( $flag );
        }
 
        /**
-        * @expectedException UnexpectedValueException
+        * @expectedException DBUnexpectedError
         * @covers Wikimedia\Rdbms\Database::clearFlag
+        * @dataProvider provideImmutableDBOFlags
+        * @param int $flag
         */
-       public function testDBOIgnoreClear() {
+       public function testDBOCannotClear( $flag ) {
                $db = $this->getMockBuilder( DatabaseMysqli::class )
                        ->disableOriginalConstructor()
                        ->setMethods( null )
                        ->getMock();
 
-               $db->clearFlag( Database::DBO_IGNORE );
+               $db->clearFlag( $flag );
        }
 
        /**
diff --git a/tests/phpunit/includes/media/SVGMetadataExtractorTest.php b/tests/phpunit/includes/media/SVGMetadataExtractorTest.php
deleted file mode 100644 (file)
index c84efa1..0000000
+++ /dev/null
@@ -1,201 +0,0 @@
-<?php
-
-/**
- * @group Media
- * @covers SVGMetadataExtractor
- */
-class SVGMetadataExtractorTest extends \MediaWikiIntegrationTestCase {
-
-       /**
-        * @dataProvider provideSvgFiles
-        */
-       public function testGetMetadata( $infile, $expected ) {
-               $this->assertMetadata( $infile, $expected );
-       }
-
-       /**
-        * @dataProvider provideSvgFilesWithXMLMetadata
-        */
-       public function testGetXMLMetadata( $infile, $expected ) {
-               $r = new XMLReader();
-               $this->assertMetadata( $infile, $expected );
-       }
-
-       /**
-        * @dataProvider provideSvgUnits
-        */
-       public function testScaleSVGUnit( $inUnit, $expected ) {
-               $this->assertEquals(
-                       $expected,
-                       SVGReader::scaleSVGUnit( $inUnit ),
-                       'SVG unit conversion and scaling failure'
-               );
-       }
-
-       function assertMetadata( $infile, $expected ) {
-               try {
-                       $data = SVGMetadataExtractor::getMetadata( $infile );
-                       $this->assertEquals( $expected, $data, 'SVG metadata extraction test' );
-               } catch ( MWException $e ) {
-                       if ( $expected === false ) {
-                               $this->assertTrue( true, 'SVG metadata extracted test (expected failure)' );
-                       } else {
-                               throw $e;
-                       }
-               }
-       }
-
-       public static function provideSvgFiles() {
-               $base = __DIR__ . '/../../data/media';
-
-               return [
-                       [
-                               "$base/Wikimedia-logo.svg",
-                               [
-                                       'width' => 1024,
-                                       'height' => 1024,
-                                       'originalWidth' => '1024',
-                                       'originalHeight' => '1024',
-                                       'translations' => [],
-                               ]
-                       ],
-                       [
-                               "$base/QA_icon.svg",
-                               [
-                                       'width' => 60,
-                                       'height' => 60,
-                                       'originalWidth' => '60',
-                                       'originalHeight' => '60',
-                                       'translations' => [],
-                               ]
-                       ],
-                       [
-                               "$base/Gtk-media-play-ltr.svg",
-                               [
-                                       'width' => 60,
-                                       'height' => 60,
-                                       'originalWidth' => '60.0000000',
-                                       'originalHeight' => '60.0000000',
-                                       'translations' => [],
-                               ]
-                       ],
-                       [
-                               "$base/Toll_Texas_1.svg",
-                               // This file triggered T33719, needs entity expansion in the xmlns checks
-                               [
-                                       'width' => 385,
-                                       'height' => 385,
-                                       'originalWidth' => '385',
-                                       'originalHeight' => '385.0004883',
-                                       'translations' => [],
-                               ]
-                       ],
-                       [
-                               "$base/Tux.svg",
-                               [
-                                       'width' => 512,
-                                       'height' => 594,
-                                       'originalWidth' => '100%',
-                                       'originalHeight' => '100%',
-                                       'title' => 'Tux',
-                                       'translations' => [],
-                                       'description' => 'For more information see: http://commons.wikimedia.org/wiki/Image:Tux.svg',
-                               ]
-                       ],
-                       [
-                               "$base/Speech_bubbles.svg",
-                               [
-                                       'width' => 627,
-                                       'height' => 461,
-                                       'originalWidth' => '17.7cm',
-                                       'originalHeight' => '13cm',
-                                       'translations' => [
-                                               'de' => SVGReader::LANG_FULL_MATCH,
-                                               'fr' => SVGReader::LANG_FULL_MATCH,
-                                               'nl' => SVGReader::LANG_FULL_MATCH,
-                                               'tlh-ca' => SVGReader::LANG_FULL_MATCH,
-                                               'tlh' => SVGReader::LANG_PREFIX_MATCH
-                                       ],
-                               ]
-                       ],
-                       [
-                               "$base/Soccer_ball_animated.svg",
-                               [
-                                       'width' => 150,
-                                       'height' => 150,
-                                       'originalWidth' => '150',
-                                       'originalHeight' => '150',
-                                       'animated' => true,
-                                       'translations' => []
-                               ],
-                       ],
-                       [
-                               "$base/comma_separated_viewbox.svg",
-                               [
-                                       'width' => 512,
-                                       'height' => 594,
-                                       'originalWidth' => '100%',
-                                       'originalHeight' => '100%',
-                                       'translations' => []
-                               ],
-                       ],
-               ];
-       }
-
-       public static function provideSvgFilesWithXMLMetadata() {
-               $base = __DIR__ . '/../../data/media';
-               // phpcs:disable Generic.Files.LineLength
-               $metadata = '<rdf:RDF xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#">
-      <ns4:Work xmlns:ns4="http://creativecommons.org/ns#" rdf:about="">
-        <ns5:format xmlns:ns5="http://purl.org/dc/elements/1.1/">image/svg+xml</ns5:format>
-        <ns5:type xmlns:ns5="http://purl.org/dc/elements/1.1/" rdf:resource="http://purl.org/dc/dcmitype/StillImage"/>
-      </ns4:Work>
-    </rdf:RDF>';
-               // phpcs:enable
-
-               $metadata = str_replace( "\r", '', $metadata ); // Windows compat
-               return [
-                       [
-                               "$base/US_states_by_total_state_tax_revenue.svg",
-                               [
-                                       'height' => 593,
-                                       'metadata' => $metadata,
-                                       'width' => 959,
-                                       'originalWidth' => '958.69',
-                                       'originalHeight' => '592.78998',
-                                       'translations' => [],
-                               ]
-                       ],
-               ];
-       }
-
-       public static function provideSvgUnits() {
-               return [
-                       [ '1' , 1 ],
-                       [ '1.1' , 1.1 ],
-                       [ '0.1' , 0.1 ],
-                       [ '.1' , 0.1 ],
-                       [ '1e2' , 100 ],
-                       [ '1E2' , 100 ],
-                       [ '+1' , 1 ],
-                       [ '-1' , -1 ],
-                       [ '-1.1' , -1.1 ],
-                       [ '1e+2' , 100 ],
-                       [ '1e-2' , 0.01 ],
-                       [ '10px' , 10 ],
-                       [ '10pt' , 10 * 1.25 ],
-                       [ '10pc' , 10 * 15 ],
-                       [ '10mm' , 10 * 3.543307 ],
-                       [ '10cm' , 10 * 35.43307 ],
-                       [ '10in' , 10 * 90 ],
-                       [ '10em' , 10 * 16 ],
-                       [ '10ex' , 10 * 12 ],
-                       [ '10%' , 51.2 ],
-                       [ '10 px' , 10 ],
-                       // Invalid values
-                       [ '1e1.1', 10 ],
-                       [ '10bp', 10 ],
-                       [ 'p10', null ],
-               ];
-       }
-}
diff --git a/tests/phpunit/includes/media/SVGReaderTest.php b/tests/phpunit/includes/media/SVGReaderTest.php
new file mode 100644 (file)
index 0000000..7063a57
--- /dev/null
@@ -0,0 +1,203 @@
+<?php
+
+/**
+ * @group Media
+ * @covers SVGReader
+ */
+class SVGReaderTest extends \MediaWikiIntegrationTestCase {
+
+       /**
+        * @dataProvider provideSvgFiles
+        */
+       public function testGetMetadata( $infile, $expected ) {
+               $this->assertMetadata( $infile, $expected );
+       }
+
+       /**
+        * @dataProvider provideSvgFilesWithXMLMetadata
+        */
+       public function testGetXMLMetadata( $infile, $expected ) {
+               $r = new XMLReader();
+               $this->assertMetadata( $infile, $expected );
+       }
+
+       /**
+        * @dataProvider provideSvgUnits
+        */
+       public function testScaleSVGUnit( $inUnit, $expected ) {
+               $this->assertEquals(
+                       $expected,
+                       SVGReader::scaleSVGUnit( $inUnit ),
+                       'SVG unit conversion and scaling failure'
+               );
+       }
+
+       function assertMetadata( $infile, $expected ) {
+               try {
+                       $svgReader = new SVGReader( $infile );
+                       $data = $svgReader->getMetadata();
+
+                       $this->assertEquals( $expected, $data, 'SVG metadata extraction test' );
+               } catch ( MWException $e ) {
+                       if ( $expected === false ) {
+                               $this->assertTrue( true, 'SVG metadata extracted test (expected failure)' );
+                       } else {
+                               throw $e;
+                       }
+               }
+       }
+
+       public static function provideSvgFiles() {
+               $base = __DIR__ . '/../../data/media';
+
+               return [
+                       [
+                               "$base/Wikimedia-logo.svg",
+                               [
+                                       'width' => 1024,
+                                       'height' => 1024,
+                                       'originalWidth' => '1024',
+                                       'originalHeight' => '1024',
+                                       'translations' => [],
+                               ]
+                       ],
+                       [
+                               "$base/QA_icon.svg",
+                               [
+                                       'width' => 60,
+                                       'height' => 60,
+                                       'originalWidth' => '60',
+                                       'originalHeight' => '60',
+                                       'translations' => [],
+                               ]
+                       ],
+                       [
+                               "$base/Gtk-media-play-ltr.svg",
+                               [
+                                       'width' => 60,
+                                       'height' => 60,
+                                       'originalWidth' => '60.0000000',
+                                       'originalHeight' => '60.0000000',
+                                       'translations' => [],
+                               ]
+                       ],
+                       [
+                               "$base/Toll_Texas_1.svg",
+                               // This file triggered T33719, needs entity expansion in the xmlns checks
+                               [
+                                       'width' => 385,
+                                       'height' => 385,
+                                       'originalWidth' => '385',
+                                       'originalHeight' => '385.0004883',
+                                       'translations' => [],
+                               ]
+                       ],
+                       [
+                               "$base/Tux.svg",
+                               [
+                                       'width' => 512,
+                                       'height' => 594,
+                                       'originalWidth' => '100%',
+                                       'originalHeight' => '100%',
+                                       'title' => 'Tux',
+                                       'translations' => [],
+                                       'description' => 'For more information see: http://commons.wikimedia.org/wiki/Image:Tux.svg',
+                               ]
+                       ],
+                       [
+                               "$base/Speech_bubbles.svg",
+                               [
+                                       'width' => 627,
+                                       'height' => 461,
+                                       'originalWidth' => '17.7cm',
+                                       'originalHeight' => '13cm',
+                                       'translations' => [
+                                               'de' => SVGReader::LANG_FULL_MATCH,
+                                               'fr' => SVGReader::LANG_FULL_MATCH,
+                                               'nl' => SVGReader::LANG_FULL_MATCH,
+                                               'tlh-ca' => SVGReader::LANG_FULL_MATCH,
+                                               'tlh' => SVGReader::LANG_PREFIX_MATCH
+                                       ],
+                               ]
+                       ],
+                       [
+                               "$base/Soccer_ball_animated.svg",
+                               [
+                                       'width' => 150,
+                                       'height' => 150,
+                                       'originalWidth' => '150',
+                                       'originalHeight' => '150',
+                                       'animated' => true,
+                                       'translations' => []
+                               ],
+                       ],
+                       [
+                               "$base/comma_separated_viewbox.svg",
+                               [
+                                       'width' => 512,
+                                       'height' => 594,
+                                       'originalWidth' => '100%',
+                                       'originalHeight' => '100%',
+                                       'translations' => []
+                               ],
+                       ],
+               ];
+       }
+
+       public static function provideSvgFilesWithXMLMetadata() {
+               $base = __DIR__ . '/../../data/media';
+               // phpcs:disable Generic.Files.LineLength
+               $metadata = '<rdf:RDF xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#">
+      <ns4:Work xmlns:ns4="http://creativecommons.org/ns#" rdf:about="">
+        <ns5:format xmlns:ns5="http://purl.org/dc/elements/1.1/">image/svg+xml</ns5:format>
+        <ns5:type xmlns:ns5="http://purl.org/dc/elements/1.1/" rdf:resource="http://purl.org/dc/dcmitype/StillImage"/>
+      </ns4:Work>
+    </rdf:RDF>';
+               // phpcs:enable
+
+               $metadata = str_replace( "\r", '', $metadata ); // Windows compat
+               return [
+                       [
+                               "$base/US_states_by_total_state_tax_revenue.svg",
+                               [
+                                       'height' => 593,
+                                       'metadata' => $metadata,
+                                       'width' => 959,
+                                       'originalWidth' => '958.69',
+                                       'originalHeight' => '592.78998',
+                                       'translations' => [],
+                               ]
+                       ],
+               ];
+       }
+
+       public static function provideSvgUnits() {
+               return [
+                       [ '1' , 1 ],
+                       [ '1.1' , 1.1 ],
+                       [ '0.1' , 0.1 ],
+                       [ '.1' , 0.1 ],
+                       [ '1e2' , 100 ],
+                       [ '1E2' , 100 ],
+                       [ '+1' , 1 ],
+                       [ '-1' , -1 ],
+                       [ '-1.1' , -1.1 ],
+                       [ '1e+2' , 100 ],
+                       [ '1e-2' , 0.01 ],
+                       [ '10px' , 10 ],
+                       [ '10pt' , 10 * 1.25 ],
+                       [ '10pc' , 10 * 15 ],
+                       [ '10mm' , 10 * 3.543307 ],
+                       [ '10cm' , 10 * 35.43307 ],
+                       [ '10in' , 10 * 90 ],
+                       [ '10em' , 10 * 16 ],
+                       [ '10ex' , 10 * 12 ],
+                       [ '10%' , 51.2 ],
+                       [ '10 px' , 10 ],
+                       // Invalid values
+                       [ '1e1.1', 10 ],
+                       [ '10bp', 10 ],
+                       [ 'p10', null ],
+               ];
+       }
+}
index 34ddb1f..ec60383 100644 (file)
@@ -939,4 +939,24 @@ EOF
                $this->assertSame( $time, $po->getCacheTime() );
        }
 
+       public static function provideOldSerialized() {
+               return [
+                       // phpcs:ignore Generic.Files.LineLength
+                       '1.34.0-wmf.15' => [ 'O:12:"ParserOutput":43:{s:5:"mText";s:0:"";s:14:"mLanguageLinks";a:0:{}s:11:"mCategories";a:0:{}s:11:"mIndicators";a:0:{}s:10:"mTitleText";s:0:"";s:6:"mLinks";a:0:{}s:10:"mTemplates";a:0:{}s:12:"mTemplateIds";a:0:{}s:7:"mImages";a:0:{}s:18:"mFileSearchOptions";a:0:{}s:14:"mExternalLinks";a:0:{}s:15:"mInterwikiLinks";a:0:{}s:11:"mNewSection";b:0;s:15:"mHideNewSection";b:0;s:10:"mNoGallery";b:0;s:10:"mHeadItems";a:0:{}s:8:"mModules";a:0:{}s:13:"mModuleStyles";a:0:{}s:13:"mJsConfigVars";a:0:{}s:12:"mOutputHooks";a:0:{}s:9:"mWarnings";a:0:{}s:9:"mSections";a:0:{}s:11:"mProperties";a:0:{}s:8:"mTOCHTML";s:0:"";s:10:"mTimestamp";N;s:11:"mEnableOOUI";b:0;s:26:"\\000ParserOutput\\000mIndexPolicy";s:0:"";s:30:"\\000ParserOutput\\000mAccessedOptions";a:0:{}s:28:"\\000ParserOutput\\000mExtensionData";a:0:{}s:30:"\\000ParserOutput\\000mLimitReportData";a:0:{}s:32:"\\000ParserOutput\\000mLimitReportJSData";a:0:{}s:34:"\\000ParserOutput\\000mPreventClickjacking";b:0;s:20:"\\000ParserOutput\\000mFlags";a:0:{}s:31:"\\000ParserOutput\\000mSpeculativeRevId";N;s:35:"\\000ParserOutput\\000revisionTimestampUsed";N;s:36:"\\000ParserOutput\\000revisionUsedSha1Base36";N;s:32:"\\000ParserOutput\\000mWrapperDivClasses";a:0:{}s:32:"\\000ParserOutput\\000mMaxAdaptiveExpiry";d:INF;s:12:"mUsedOptions";N;s:8:"mVersion";s:5:"1.6.4";s:10:"mCacheTime";s:0:"";s:12:"mCacheExpiry";N;s:16:"mCacheRevisionId";N;}' ]
+               ];
+       }
+
+       /**
+        * Ensure that old ParserOutput objects can be unserialized and reserialized without an error
+        * (T229366).
+        *
+        * @dataProvider provideOldSerialized
+        * @covers ParserOutput::__sleep()
+        */
+       public function testOldSerialized( $serialized ) {
+               $po = unserialize( stripcslashes( $serialized ) );
+               $reserialized = serialize( $po );
+               $this->assertStringStartsWith( 'O:', $reserialized );
+       }
+
 }
index c3d5ec1..f6fd824 100644 (file)
@@ -48,7 +48,6 @@ class ResourceLoaderContextTest extends PHPUnit\Framework\TestCase {
        public function testAccessors() {
                $ctx = new ResourceLoaderContext( $this->getResourceLoader(), new FauxRequest( [] ) );
                $this->assertInstanceOf( ResourceLoader::class, $ctx->getResourceLoader() );
-               $this->assertInstanceOf( Config::class, $ctx->getConfig() );
                $this->assertInstanceOf( WebRequest::class, $ctx->getRequest() );
                $this->assertInstanceOf( Psr\Log\LoggerInterface::class, $ctx->getLogger() );
        }
index 5964915..089431e 100644 (file)
@@ -311,37 +311,41 @@ class ResourceLoaderWikiModuleTest extends ResourceLoaderTestCase {
 
        public static function provideGetContent() {
                yield 'Bad title' => [ null, '[x]' ];
-               yield 'Dead redirect' => [ null, [
-                       'text' => 'Dead redirect',
-                       'title' => 'Dead_redirect',
-                       'redirect' => 1,
-               ] ];
-               yield 'Bad content model' => [ null, [
-                       'text' => 'MediaWiki:Wikitext',
-                       'ns' => NS_MEDIAWIKI,
-                       'title' => 'Wikitext',
-               ] ];
+
                yield 'No JS content found' => [ null, [
-                       'text' => 'MediaWiki:Script.js',
+                       'text' => 'MediaWiki:Foo.js',
                        'ns' => NS_MEDIAWIKI,
-                       'title' => 'Script.js',
+                       'title' => 'Foo.js',
                ] ];
-               yield 'No CSS content found' => [ null, [
-                       'text' => 'MediaWiki:Styles.css',
+
+               yield 'JS content' => [ 'code;', [
+                       'text' => 'MediaWiki:Foo.js',
                        'ns' => NS_MEDIAWIKI,
-                       'title' => 'Script.css',
-               ] ];
+                       'title' => 'Foo.js',
+               ], new JavaScriptContent( 'code;' ) ];
+
+               yield 'CSS content' => [ 'code {}', [
+                       'text' => 'MediaWiki:Foo.css',
+                       'ns' => NS_MEDIAWIKI,
+                       'title' => 'Foo.css',
+               ], new CssContent( 'code {}' ) ];
+
+               yield 'Wikitext content' => [ null, [
+                       'text' => 'MediaWiki:Foo',
+                       'ns' => NS_MEDIAWIKI,
+                       'title' => 'Foo',
+               ], new WikitextContent( 'code;' ) ];
        }
 
        /**
         * @dataProvider provideGetContent
         */
-       public function testGetContent( $expected, $title ) {
+       public function testGetContent( $expected, $title, Content $contentObj = null ) {
                $context = $this->getResourceLoaderContext( [], new EmptyResourceLoader );
                $module = $this->getMockBuilder( ResourceLoaderWikiModule::class )
                        ->setMethods( [ 'getContentObj' ] )->getMock();
                $module->method( 'getContentObj' )
-                       ->willReturn( null );
+                       ->willReturn( $contentObj );
 
                if ( is_array( $title ) ) {
                        $title += [ 'ns' => NS_MAIN, 'id' => 1, 'len' => 1, 'redirect' => 0 ];
index 774e023..e9efb22 100644 (file)
@@ -3,8 +3,8 @@
 class SearchResultSetTest extends MediaWikiTestCase {
        /**
         * @covers SearchResultSet::getIterator
-        * @covers SearchResultSet::next
-        * @covers SearchResultSet::rewind
+        * @covers BaseSearchResultSet::next
+        * @covers BaseSearchResultSet::rewind
         */
        public function testIterate() {
                $result = SearchResult::newFromTitle( Title::newMainPage() );
@@ -17,8 +17,8 @@ class SearchResultSetTest extends MediaWikiTestCase {
                }
                $this->assertEquals( 1, $count );
 
-               $this->hideDeprecated( 'SearchResultSet::rewind' );
-               $this->hideDeprecated( 'SearchResultSet::next' );
+               $this->hideDeprecated( 'BaseSearchResultSet::rewind' );
+               $this->hideDeprecated( 'BaseSearchResultSet::next' );
                $resultSet->rewind();
                $count = 0;
                while ( ( $iterResult = $resultSet->next() ) !== false ) {
@@ -29,8 +29,8 @@ class SearchResultSetTest extends MediaWikiTestCase {
        }
 
        /**
-        * @covers SearchResultSet::augmentResult
-        * @covers SearchResultSet::setAugmentedData
+        * @covers SearchResultSetTrait::augmentResult
+        * @covers SearchResultSetTrait::setAugmentedData
         */
        public function testDelayedResultAugment() {
                $result = SearchResult::newFromTitle( Title::newMainPage() );
index 86e3295..5f11a01 100644 (file)
@@ -391,6 +391,78 @@ class SpecialBlockTest extends SpecialPageTestBase {
                $this->assertSame( 0, $count );
        }
 
+       /**
+        * @dataProvider provideProcessFormErrors
+        * @covers ::processForm()
+        */
+       public function testProcessFormErrors( $data, $expected, $config = [] ) {
+               $defaultConfig = [
+                       'wgEnablePartialBlocks' => true,
+                       'wgBlockAllowsUTEdit' => true,
+               ];
+
+               $this->setMwGlobals( array_merge( $defaultConfig, $config ) );
+
+               $defaultData = [
+                       'Target' => '1.2.3.4',
+                       'Expiry' => 'infinity',
+                       'Reason' => [ 'bad reason' ],
+                       'Confirm' => false,
+                       'PageRestrictions' => '',
+                       'NamespaceRestrictions' => '',
+               ];
+
+               $context = RequestContext::getMain();
+               $page = $this->newSpecialPage();
+               $result = $page->processForm( array_merge( $defaultData, $data ), $context );
+
+               $this->assertEquals( $result[0], $expected );
+       }
+
+       public function provideProcessFormErrors() {
+               return [
+                       'Invalid expiry' => [
+                               [
+                                       'Expiry' => 'invalid',
+                               ],
+                               'ipb_expiry_invalid',
+                       ],
+                       'Expiry is in the past' => [
+                               [
+                                       'Expiry' => 'yesterday',
+                               ],
+                               'ipb_expiry_old',
+                       ],
+                       'HideUser with wrong permissions' => [
+                               [
+                                       'HideUser' => 1,
+                               ],
+                               'badaccess-group0',
+                       ],
+                       'Bad ip address' => [
+                               [
+                                       'Target' => '1.2.3.4/1234',
+                               ],
+                               'badipaddress',
+                       ],
+                       'Edit user talk page invalid with no restrictions' => [
+                               [
+                                       'EditingRestriction' => 'partial',
+                                       'DisableUTEdit' => 1,
+                               ],
+                               'ipb-prevent-user-talk-edit',
+                       ],
+                       'Edit user talk page invalid with namespace restriction != NS_USER_TALK ' => [
+                               [
+                                       'EditingRestriction' => 'partial',
+                                       'DisableUTEdit' => 1,
+                                       'NamespaceRestrictions' => NS_USER
+                               ],
+                               'ipb-prevent-user-talk-edit',
+                       ],
+               ];
+       }
+
        /**
         * @dataProvider provideCheckUnblockSelf
         * @covers ::checkUnblockSelf
index 4f9664f..acd8a19 100644 (file)
@@ -36,14 +36,6 @@ class ResourcesTest extends MediaWikiTestCase {
                );
        }
 
-       public function testVersionHash() {
-               $data = self::getAllModules();
-               foreach ( $data['modules'] as $moduleName => $module ) {
-                       $version = $module->getVersionHash( $data['context'] );
-                       $this->assertEquals( 7, strlen( $version ), "$moduleName must use ResourceLoader::makeHash" );
-               }
-       }
-
        /**
         * Verify that all modules specified as dependencies of other modules actually
         * exist and are not illegal.
index 4969a8b..d55b603 100644 (file)
@@ -6,6 +6,12 @@ return [
 
        /* Utilities */
 
+       'jquery.qunit' => [
+               'scripts' => 'resources/lib/qunitjs/qunit.js',
+               'styles' => 'resources/lib/qunitjs/qunit.css',
+               'targets' => [ 'desktop', 'mobile' ],
+       ],
+
        'test.sinonjs' => [
                'scripts' => [
                        'tests/qunit/suites/resources/test.sinonjs/index.js',