61 global $wgCaptchaSecret, $wgCaptchaDirectoryLevels;
63 $totalTime = -microtime(
true );
65 $instance = ConfirmEditHooks::getInstance();
67 $this->
fatalError(
"\$wgCaptchaClass is not FancyCaptcha.\n", 1 );
69 $backend = $instance->getBackend();
71 $deleteOldCaptchas = $this->
getOption(
'delete' );
73 $countGen = (int)$this->
getOption(
'fill' );
74 if ( !$deleteOldCaptchas ) {
75 $countAct = $instance->getCaptchaCount();
76 $this->
output(
"Current number of captchas is $countAct.\n" );
77 $countGen -= $countAct;
80 if ( $countGen <= 0 ) {
81 $this->
output(
"No need to generate anymore captchas.\n" );
87 $this->
fatalError(
"Could not create temp directory.\n", 1 );
90 $captchaScript =
'captcha.py';
93 $captchaScript =
'captcha-old.py';
96 $cmd = sprintf(
"python %s --key %s --output %s --count %s --dirs %s",
104 [
'wordlist',
'font',
'font-size',
'blacklist',
'verbose',
'threads' ] as $par
111 $this->
output(
"Generating $countGen new captchas.." );
113 $captchaTime = -microtime(
true );
115 if ( $retVal != 0 ) {
117 $this->
fatalError(
"An error occured when running $captchaScript.\n", 1 );
120 $captchaTime += microtime(
true );
121 $this->
output(
" Done.\n" );
125 "\nGenerated %d captchas in %.1f seconds\n",
132 if ( $deleteOldCaptchas ) {
133 $this->
output(
"Getting a list of old captchas to delete..." );
134 $path = $backend->getRootStoragePath() .
'/captcha-render';
135 foreach ( $backend->getFileList( [
'dir' =>
$path ] ) as
$file ) {
141 $this->
output(
" Done.\n" );
144 $this->
output(
"Copying the new captchas to storage..." );
146 $storeTime = -microtime(
true );
147 $iter =
new RecursiveIteratorIterator(
148 new RecursiveDirectoryIterator(
150 FilesystemIterator::SKIP_DOTS
152 RecursiveIteratorIterator::LEAVES_ONLY
155 $captchasGenerated = iterator_count( $iter );
160 foreach ( $iter as $fileInfo ) {
161 if ( !$fileInfo->isFile() ) {
164 list( $salt, $hash ) = $instance->hashFromImageName( $fileInfo->getBasename() );
165 $dest = $instance->imagePath( $salt, $hash );
166 $backend->prepare( [
'dir' => dirname( $dest ) ] );
169 'src' => $fileInfo->getPathname(),
174 $ret = $backend->doQuickOperations( $filesToStore );
176 $storeTime += microtime(
true );
178 $storeSucceeded =
true;
179 if ( $ret->isOK() ) {
180 $this->
output(
" Done.\n" );
183 "\nCopied %d captchas to storage in %.1f seconds\n",
188 if ( !$ret->isGood() ) {
190 "Non fatal errors:\n" .
191 Status::wrap( $ret )->getWikiText(
null,
null,
'en' ) .
195 if ( $ret->failCount ) {
196 $storeSucceeded =
false;
197 $this->
error( sprintf(
"\nFailed to copy %d captchas\n", $ret->failCount ) );
199 if ( $ret->successCount + $ret->failCount !== $captchasGenerated ) {
200 $storeSucceeded =
false;
202 sprintf(
"Internal error: captchasGenerated: %d, successCount: %d, failCount: %d\n",
203 $captchasGenerated, $ret->successCount, $ret->failCount
208 $storeSucceeded =
false;
209 $this->
output(
"Errored.\n" );
211 Status::wrap( $ret )->getWikiText(
null,
null,
'en' ) .
216 if ( $storeSucceeded && $deleteOldCaptchas ) {
217 $numOriginalFiles = count( $filesToDelete );
218 $this->
output(
"Deleting {$numOriginalFiles} old captchas...\n" );
219 $deleteTime = -microtime(
true );
220 $ret = $backend->doQuickOperations( $filesToDelete );
222 $deleteTime += microtime(
true );
223 if ( $ret->isOK() ) {
224 $this->
output(
"Done.\n" );
227 "\nDeleted %d old captchas in %.1f seconds\n",
232 if ( !$ret->isGood() ) {
234 "Non fatal errors:\n" .
235 Status::wrap( $ret )->getWikiText(
null,
null,
'en' ) .
240 $this->
output(
"Errored.\n" );
242 Status::wrap( $ret )->getWikiText(
null,
null,
'en' ) .
248 $this->
output(
"Removing temporary files..." );
250 $this->
output(
" Done.\n" );
252 $totalTime += microtime(
true );
255 "\nWhole captchas generation process took %.1f seconds\n",