Translate extension for MediaWiki
 
Loading...
Searching...
No Matches
TranslatablePageParser.php
1<?php
2declare( strict_types = 1 );
3
4namespace MediaWiki\Extension\Translate\PageTranslation;
5
7
16 private $placeholderFactory;
17
18 public function __construct( ParsingPlaceholderFactory $placeholderFactory ) {
19 $this->placeholderFactory = $placeholderFactory;
20 }
21
22 public function containsMarkup( string $text ): bool {
23 $nowiki = [];
24 $text = $this->armourNowiki( $nowiki, $text );
25 return preg_match( '~</?translate[ >]~', $text ) !== 0;
26 }
27
32 public function cleanupTags( string $text ): string {
33 $nowiki = [];
34 $text = $this->armourNowiki( $nowiki, $text );
35 $text = preg_replace( '~<translate( nowrap)?>\n?~s', '', $text );
36 $text = preg_replace( '~\n?</translate>~s', '', $text );
37 // Markers: headers and the rest
38 $ic = preg_quote( TranslationUnit::UNIT_MARKER_INVALID_CHARS, '~' );
39 $text = preg_replace( "~(^=.*=) <!--T:[^$ic]+-->$~um", '\1', $text );
40 $text = preg_replace( "~<!--T:[^$ic]+-->[\n ]?~um", '', $text );
41 // Remove variables
42 $unit = new TranslationUnit( $text );
43 $text = $unit->getTextForTrans();
44
45 $text = $this->unarmourNowiki( $nowiki, $text );
46 return $text;
47 }
48
50 public function parse( string $text ): ParserOutput {
51 $nowiki = [];
52 $text = $this->armourNowiki( $nowiki, $text );
53
54 $sections = [];
55 $tagPlaceHolders = [];
56
57 while ( true ) {
58 $re = '~(<translate(?: nowrap)?>)(.*?)</translate>~s';
59 $matches = [];
60 $ok = preg_match( $re, $text, $matches, PREG_OFFSET_CAPTURE );
61
62 if ( $ok === 0 || $ok === false ) {
63 break; // No match or failure
64 }
65
66 $contentWithTags = $matches[0][0];
67 $contentWithoutTags = $matches[2][0];
68 // These are offsets to the content inside the tags in $text
69 $offsetStart = $matches[0][1];
70 $offsetEnd = $offsetStart + strlen( $contentWithTags );
71
72 // Replace the whole match with a placeholder
73 $ph = $this->placeholderFactory->make();
74 $text = substr( $text, 0, $offsetStart ) . $ph . substr( $text, $offsetEnd );
75
76 if ( preg_match( '~<translate( nowrap)?>~', $contentWithoutTags ) !== 0 ) {
77 throw new ParsingFailure(
78 'Nested tags',
79 [ 'pt-parse-nested', $contentWithoutTags ]
80 );
81 }
82
83 $openTag = $matches[1][0];
84 $canWrap = $openTag !== '<translate nowrap>';
85
86 // Parse the content inside the tags
87 $contentWithoutTags = $this->unarmourNowiki( $nowiki, $contentWithoutTags );
88 $parse = $this->parseSection( $contentWithoutTags, $canWrap );
89
90 // Update list of sections and the template with the results
91 $sections += $parse['sections'];
92 $tagPlaceHolders[$ph] = new Section( $openTag, $parse['template'], '</translate>' );
93 }
94
95 $prettyTemplate = $text;
96 foreach ( $tagPlaceHolders as $ph => $value ) {
97 $prettyTemplate = str_replace( $ph, '[...]', $prettyTemplate );
98 }
99
100 if ( preg_match( '~<translate( nowrap)?>~', $text ) !== 0 ) {
101 throw new ParsingFailure(
102 'Unmatched opening tag',
103 [ 'pt-parse-open', $prettyTemplate ]
104 );
105 } elseif ( str_contains( $text, '</translate>' ) ) {
106 throw new ParsingFailure(
107 "Unmatched closing tag",
108 [ 'pt-parse-close', $prettyTemplate ]
109 );
110 }
111
112 $text = $this->unarmourNowiki( $nowiki, $text );
113
114 return new ParserOutput( $text, $tagPlaceHolders, $sections );
115 }
116
123 public function parseSection( string $text, bool $canWrap ): array {
124 $flags = PREG_SPLIT_NO_EMPTY | PREG_SPLIT_DELIM_CAPTURE;
125 $parts = preg_split( '~(^\s*|\s*\n\n\s*|\s*$)~', $text, -1, $flags );
126
127 $inline = preg_match( '~\n~', $text ) === 0;
128
129 $template = '';
130 $sections = [];
131
132 foreach ( $parts as $_ ) {
133 if ( trim( $_ ) === '' ) {
134 $template .= $_;
135 } else {
136 $ph = $this->placeholderFactory->make();
137 $tpsection = $this->parseUnit( $_ );
138 $tpsection->setIsInline( $inline );
139 $tpsection->setCanWrap( $canWrap );
140 $sections[$ph] = $tpsection;
141 $template .= $ph;
142 }
143 }
144
145 return [
146 'template' => $template,
147 'sections' => $sections,
148 ];
149 }
150
157 public function parseUnit( string $content ): TranslationUnit {
158 $re = '~<!--T:(.*?)-->~';
159 $matches = [];
160 $count = preg_match_all( $re, $content, $matches, PREG_SET_ORDER );
161
162 if ( $count > 1 ) {
163 throw new ParsingFailure(
164 'Multiple translation unit markers',
165 [ 'pt-shake-multiple', $content ]
166 );
167 }
168
169 // If no id given in the source, default to a new section id
170 $id = TranslationUnit::NEW_UNIT_ID;
171 if ( $count === 1 ) {
172 foreach ( $matches as $match ) {
173 [ /*full*/, $id ] = $match;
174
175 // Currently handle only these two standard places.
176 // Is this too strict?
177 $rer1 = '~^<!--T:(.*?)-->( |\n)~'; // Normal sections
178 $rer2 = '~\s*<!--T:(.*?)-->$~m'; // Sections with title
179 $content = preg_replace( $rer1, '', $content );
180 $content = preg_replace( $rer2, '', $content );
181
182 if ( preg_match( $re, $content ) === 1 ) {
183 throw new ParsingFailure(
184 'Translation unit marker is in unsupported position',
185 [ 'pt-shake-position', $content ]
186 );
187 } elseif ( trim( $content ) === '' ) {
188 throw new ParsingFailure(
189 'Translation unit has no content besides marker',
190 [ 'pt-shake-empty', $id ]
191 );
192 }
193 }
194 }
195
196 return new TranslationUnit( $content, $id );
197 }
198
200 public function armourNowiki( array &$holders, string $text ): string {
201 $re = '~(<nowiki>)(.*?)(</nowiki>)~s';
202
203 while ( preg_match( $re, $text, $matches ) ) {
204 $ph = $this->placeholderFactory->make();
205 $text = str_replace( $matches[0], $ph, $text );
206 $holders[$ph] = $matches[0];
207 }
208
209 return $text;
210 }
211
213 public function unarmourNowiki( array $holders, string $text ): string {
214 return strtr( $text, $holders );
215 }
216}
return[ 'Translate:ConfigHelper'=> static function():ConfigHelper { return new ConfigHelper();}, 'Translate:CsvTranslationImporter'=> static function(MediaWikiServices $services):CsvTranslationImporter { return new CsvTranslationImporter( $services->getWikiPageFactory());}, 'Translate:EntitySearch'=> static function(MediaWikiServices $services):EntitySearch { return new EntitySearch($services->getMainWANObjectCache(), $services->getCollationFactory() ->makeCollation( 'uca-default-u-kn'), MessageGroups::singleton(), $services->getNamespaceInfo(), $services->get( 'Translate:MessageIndex'), $services->getTitleParser(), $services->getTitleFormatter());}, 'Translate:ExternalMessageSourceStateImporter'=> static function(MediaWikiServices $services):ExternalMessageSourceStateImporter { return new ExternalMessageSourceStateImporter($services->getMainConfig(), $services->get( 'Translate:GroupSynchronizationCache'), $services->getJobQueueGroup(), LoggerFactory::getInstance( 'Translate.GroupSynchronization'), $services->get( 'Translate:MessageIndex'));}, 'Translate:FileFormatFactory'=> static function(MediaWikiServices $services):FileFormatFactory { return new FileFormatFactory( $services->getObjectFactory());}, 'Translate:GroupSynchronizationCache'=> static function(MediaWikiServices $services):GroupSynchronizationCache { return new GroupSynchronizationCache( $services->get( 'Translate:PersistentCache'));}, 'Translate:HookRunner'=> static function(MediaWikiServices $services):HookRunner { return new HookRunner( $services->getHookContainer());}, 'Translate:MessageBundleStore'=> static function(MediaWikiServices $services):MessageBundleStore { return new MessageBundleStore($services->get( 'Translate:RevTagStore'), $services->getJobQueueGroup(), $services->getLanguageNameUtils(), $services->get( 'Translate:MessageIndex'));}, 'Translate:MessageGroupReviewStore'=> static function(MediaWikiServices $services):MessageGroupReviewStore { return new MessageGroupReviewStore($services->getDBLoadBalancer(), $services->get( 'Translate:HookRunner'));}, 'Translate:MessageGroupStatsTableFactory'=> static function(MediaWikiServices $services):MessageGroupStatsTableFactory { return new MessageGroupStatsTableFactory($services->get( 'Translate:ProgressStatsTableFactory'), $services->getDBLoadBalancer(), $services->getLinkRenderer(), $services->get( 'Translate:MessageGroupReviewStore'), $services->getMainConfig() ->get( 'TranslateWorkflowStates') !==false);}, 'Translate:MessageIndex'=> static function(MediaWikiServices $services):MessageIndex { $params=$services->getMainConfig() ->get( 'TranslateMessageIndex');if(is_string( $params)) { $params=(array) $params;} $class=array_shift( $params);return new $class( $params);}, 'Translate:MessagePrefixStats'=> static function(MediaWikiServices $services):MessagePrefixStats { return new MessagePrefixStats( $services->getTitleParser());}, 'Translate:ParsingPlaceholderFactory'=> static function():ParsingPlaceholderFactory { return new ParsingPlaceholderFactory();}, 'Translate:PersistentCache'=> static function(MediaWikiServices $services):PersistentCache { return new PersistentDatabaseCache($services->getDBLoadBalancer(), $services->getJsonCodec());}, 'Translate:ProgressStatsTableFactory'=> static function(MediaWikiServices $services):ProgressStatsTableFactory { return new ProgressStatsTableFactory($services->getLinkRenderer(), $services->get( 'Translate:ConfigHelper'));}, 'Translate:RevTagStore'=> static function(MediaWikiServices $services):RevTagStore { return new RevTagStore($services->getDBLoadBalancerFactory());}, 'Translate:SubpageListBuilder'=> static function(MediaWikiServices $services):SubpageListBuilder { return new SubpageListBuilder($services->get( 'Translate:TranslatableBundleFactory'), $services->getLinkBatchFactory());}, 'Translate:TranslatableBundleExporter'=> static function(MediaWikiServices $services):TranslatableBundleExporter { return new TranslatableBundleExporter($services->get( 'Translate:SubpageListBuilder'), $services->getWikiExporterFactory(), $services->getDBLoadBalancer());}, 'Translate:TranslatableBundleFactory'=> static function(MediaWikiServices $services):TranslatableBundleFactory { return new TranslatableBundleFactory($services->get( 'Translate:TranslatablePageStore'), $services->get( 'Translate:MessageBundleStore'));}, 'Translate:TranslatableBundleImporter'=> static function(MediaWikiServices $services):TranslatableBundleImporter { return new TranslatableBundleImporter($services->getWikiImporterFactory(), $services->get( 'Translate:TranslatablePageParser'), $services->getRevisionLookup());}, 'Translate:TranslatableBundleMover'=> static function(MediaWikiServices $services):TranslatableBundleMover { return new TranslatableBundleMover($services->getMovePageFactory(), $services->getJobQueueGroup(), $services->getLinkBatchFactory(), $services->get( 'Translate:TranslatableBundleFactory'), $services->get( 'Translate:SubpageListBuilder'), $services->getMainConfig() ->get( 'TranslatePageMoveLimit'));}, 'Translate:TranslatableBundleStatusStore'=> static function(MediaWikiServices $services):TranslatableBundleStatusStore { return new TranslatableBundleStatusStore($services->getDBLoadBalancer() ->getConnection(DB_PRIMARY), $services->getCollationFactory() ->makeCollation( 'uca-default-u-kn'), $services->getDBLoadBalancer() ->getMaintenanceConnectionRef(DB_PRIMARY));}, 'Translate:TranslatablePageParser'=> static function(MediaWikiServices $services):TranslatablePageParser { return new TranslatablePageParser($services->get( 'Translate:ParsingPlaceholderFactory'));}, 'Translate:TranslatablePageStore'=> static function(MediaWikiServices $services):TranslatablePageStore { return new TranslatablePageStore($services->get( 'Translate:MessageIndex'), $services->getJobQueueGroup(), $services->get( 'Translate:RevTagStore'), $services->getDBLoadBalancer(), $services->get( 'Translate:TranslatableBundleStatusStore'), $services->get( 'Translate:TranslatablePageParser'),);}, 'Translate:TranslationStashReader'=> static function(MediaWikiServices $services):TranslationStashReader { $db=$services->getDBLoadBalancer() ->getConnection(DB_REPLICA);return new TranslationStashStorage( $db);}, 'Translate:TranslationStatsDataProvider'=> static function(MediaWikiServices $services):TranslationStatsDataProvider { return new TranslationStatsDataProvider(new ServiceOptions(TranslationStatsDataProvider::CONSTRUCTOR_OPTIONS, $services->getMainConfig()), $services->getObjectFactory(), $services->getDBLoadBalancer());}, 'Translate:TranslationUnitStoreFactory'=> static function(MediaWikiServices $services):TranslationUnitStoreFactory { return new TranslationUnitStoreFactory( $services->getDBLoadBalancer());}, 'Translate:TranslatorActivity'=> static function(MediaWikiServices $services):TranslatorActivity { $query=new TranslatorActivityQuery($services->getMainConfig(), $services->getDBLoadBalancer());return new TranslatorActivity($services->getMainObjectStash(), $query, $services->getJobQueueGroup());}, 'Translate:TtmServerFactory'=> static function(MediaWikiServices $services):TtmServerFactory { $config=$services->getMainConfig();$default=$config->get( 'TranslateTranslationDefaultService');if( $default===false) { $default=null;} return new TtmServerFactory( $config->get( 'TranslateTranslationServices'), $default);}]
@phpcs-require-sorted-array
Represents a parsing output produced by TranslatablePageParser.
Represents any kind of failure to parse a translatable page source code.
Generates ParserOutput from text or removes all tags from a text.
cleanupTags(string $text)
Remove all opening and closing translate tags following the same whitespace rules as the regular pars...
parseUnit(string $content)
Checks if this unit already contains a section marker.
parseSection(string $text, bool $canWrap)
Splits the content marked with <translate> tags into translation units, which are separated with two ...
This class represents one translation unit in a translatable page.
Create unique placeholders that can be used when parsing (wiki)text.