<?php

declare(strict_types=1);

use dokuwiki\test\rector\DokuWikiPtlnRector;
use dokuwiki\test\rector\DokuWikiRenamePrintToEcho;
use Rector\Caching\ValueObject\Storage\FileCacheStorage;
use Rector\CodeQuality\Rector\Array_\CallableThisArrayToAnonymousFunctionRector;
use Rector\CodeQuality\Rector\Concat\JoinStringConcatRector;
use Rector\CodeQuality\Rector\FunctionLike\SimplifyUselessVariableRector;
use Rector\CodeQuality\Rector\If_\CombineIfRector;
use Rector\CodeQuality\Rector\If_\ExplicitBoolCompareRector;
use Rector\CodeQuality\Rector\If_\SimplifyIfElseToTernaryRector;
use Rector\CodeQuality\Rector\If_\SimplifyIfReturnBoolRector;
use Rector\CodeQuality\Rector\Isset_\IssetOnPropertyObjectToPropertyExistsRector;
use Rector\CodingStyle\Rector\Catch_\CatchExceptionNameMatchingTypeRector;
use Rector\CodingStyle\Rector\Class_\AddArrayDefaultToArrayPropertyRector;
use Rector\CodingStyle\Rector\Closure\StaticClosureRector;
use Rector\CodingStyle\Rector\Encapsed\EncapsedStringsToSprintfRector;
use Rector\CodingStyle\Rector\Encapsed\WrapEncapsedVariableInCurlyBracesRector;
use Rector\CodingStyle\Rector\FuncCall\StrictArraySearchRector;
use Rector\CodingStyle\Rector\PostInc\PostIncDecToPreIncDecRector;
use Rector\CodingStyle\Rector\Stmt\NewlineAfterStatementRector;
use Rector\CodingStyle\Rector\String_\SymplifyQuoteEscapeRector;
use Rector\Config\RectorConfig;
use Rector\DeadCode\Rector\ClassMethod\RemoveUselessParamTagRector;
use Rector\DeadCode\Rector\If_\RemoveAlwaysTrueIfConditionRector;
use Rector\DeadCode\Rector\If_\RemoveUnusedNonEmptyArrayBeforeForeachRector;
use Rector\DeadCode\Rector\Property\RemoveUselessVarTagRector;
use Rector\DeadCode\Rector\StaticCall\RemoveParentCallWithoutParentRector;
use Rector\DeadCode\Rector\Stmt\RemoveUnreachableStatementRector;
use Rector\Php71\Rector\FuncCall\CountOnNullRector;
use Rector\Php71\Rector\FuncCall\RemoveExtraParametersRector;
use Rector\Php80\Rector\Identical\StrEndsWithRector;
use Rector\Php80\Rector\Identical\StrStartsWithRector;
use Rector\Php80\Rector\NotIdentical\StrContainsRector;
use Rector\Renaming\Rector\FuncCall\RenameFunctionRector;
use Rector\Renaming\Rector\Name\RenameClassRector;
use Rector\Set\ValueObject\LevelSetList;
use Rector\Set\ValueObject\SetList;
use Rector\Strict\Rector\Empty_\DisallowedEmptyRuleFixerRector;
use Rector\TypeDeclaration\Rector\Property\TypedPropertyFromAssignsRector;

return static function (RectorConfig $rectorConfig): void {
    // FIXME we may want to autoload these later
    require_once __DIR__ . '/rector/DokuWikiPtlnRector.php';
    require_once __DIR__ . '/rector/DokuWikiRenamePrintToEcho.php';

    $rectorConfig->paths([
        __DIR__ . '/../inc/',
        __DIR__ . '/../lib/',
        __DIR__ . '/../bin/',
        __DIR__ . '/../*.php',
    ]);

    $rectorConfig->bootstrapFiles([
        __DIR__ . '/../inc/init.php',
    ]);

    $rectorConfig->importNames();
    $rectorConfig->importShortClasses(false);
    $rectorConfig->cacheClass(FileCacheStorage::class);
    $rectorConfig->cacheDirectory(__DIR__ . '/.rector-cache');

    // supported minimum PHP version can be overridden by environment variable
    [$major, $minor] = explode('.', $_SERVER['RECTOR_MIN_PHP'] ?? '' ?: '7.4');
    $phpset = LevelSetList::class . '::UP_TO_PHP_' . $major . $minor;
    echo "Using PHP set $phpset\n";

    // define sets of rules
    $rectorConfig->sets([
        constant($phpset),
        SetList::CODE_QUALITY,
        SetList::DEAD_CODE,
        SetList::CODING_STYLE,
    ]);

    // future rules for which we have polyfills
    $rectorConfig->rule(StrContainsRector::class);
    $rectorConfig->rule(StrEndsWithRector::class);
    $rectorConfig->rule(StrStartsWithRector::class);

    $rectorConfig->skip([
        // skip paths
        __DIR__ . '/../inc/lang/*',
        __DIR__ . '/../lib/plugins/*/_test/*',
        __DIR__ . '/../lib/tpl/*/_test/*',
        __DIR__ . '/../lib/plugins/*/lang/*',
        __DIR__ . '/../lib/tpl/*/lang/*',
        __DIR__ . '/../lib/plugins/*/conf/*', // maybe later
        __DIR__ . '/../lib/tpl/*/conf/*',  // maybe later
        __DIR__ . '/../lib/plugins/*/vendor/*',
        __DIR__ . '/../lib/tpl/*/vendor/*',
        __DIR__ . '/../lib/plugins/*/skel/*', // dev plugin
        __DIR__ . '/../inc/deprecated.php',
        __DIR__ . '/../inc/form.php',

        // third party libs, not yet moved to composer
        __DIR__ . '/../inc/DifferenceEngine.php',
        __DIR__ . '/../inc/JpegMeta.php',
        __DIR__ . '/../lib/plugins/authad/adLDAP',

        // skip rules
        SimplifyIfElseToTernaryRector::class,
        NewlineAfterStatementRector::class,
        CombineIfRector::class,
        ExplicitBoolCompareRector::class,
        IssetOnPropertyObjectToPropertyExistsRector::class, // maybe?
        SymplifyQuoteEscapeRector::class,
        CatchExceptionNameMatchingTypeRector::class,
        EncapsedStringsToSprintfRector::class,
        CallableThisArrayToAnonymousFunctionRector::class,
        StaticClosureRector::class,
        SimplifyUselessVariableRector::class, // seems to strip constructor property initializations
        PostIncDecToPreIncDecRector::class,
        RemoveUselessParamTagRector::class,
        DisallowedEmptyRuleFixerRector::class,
        CountOnNullRector::class, // adds unwanted is_countable checks?
        RemoveParentCallWithoutParentRector::class,
        WrapEncapsedVariableInCurlyBracesRector::class,
        SimplifyIfReturnBoolRector::class,
        StrictArraySearchRector::class, // we cannot assume strict search is always wanted
        AddArrayDefaultToArrayPropertyRector::class, // may break code differentiating between null and empty array
        RemoveUselessVarTagRector::class,
        TypedPropertyFromAssignsRector::class, // maybe?
        JoinStringConcatRector::class, // this does not count variables, so it creates overlong lines
        RemoveExtraParametersRector::class, // this actually broke code
        RemoveUnusedNonEmptyArrayBeforeForeachRector::class, // seems unreliable when checking on array keys
        RemoveAlwaysTrueIfConditionRector::class, // fails with if(defined(...)) constructs
        RemoveUnreachableStatementRector::class, // fails GOTO in authpdo -> should be rewritten with exceptions
    ]);

    $rectorConfig->ruleWithConfiguration(RenameClassRector::class, [
        // see inc/deprecated.php
        'RemoteAccessDeniedException' => 'dokuwiki\Remote\AccessDeniedException',
        'RemoteException' => 'dokuwiki\Remote\RemoteException',
        'setting' => 'dokuwiki\plugin\config\core\Setting\Setting',
        'setting_authtype' => 'dokuwiki\plugin\config\core\Setting\SettingAuthtype',
        'setting_string' => 'dokuwiki\plugin\config\core\Setting\SettingString',
        'PageChangelog' => 'dokuwiki\ChangeLog\PageChangeLog',
        'MediaChangelog' => 'dokuwiki\ChangeLog\MediaChangeLog',
        'Input' => 'dokuwiki\Input\Input',
        'PostInput' => 'dokuwiki\Input\Post',
        'GetInput' => 'dokuwiki\Input\Get',
        'ServerInput' => 'dokuwiki\Input\Server',
        'PassHash' => 'dokuwiki\PassHash',
        'HTTPClientException' => 'dokuwiki\HTTP\HTTPClientException',
        'HTTPClient' => 'dokuwiki\HTTP\HTTPClient',
        'DokuHTTPClient' => 'dokuwiki\HTTP\DokuHTTPClient',
        'Doku_Plugin_Controller' => 'dokuwiki\Extension\PluginController',
        'Doku_Indexer' => 'dokuwiki\Search\Indexer',
        'IXR_Client' => 'dokuwiki\Remote\IXR\Client',
        'IXR_ClientMulticall' => 'IXR\Client\ClientMulticall',
        'IXR_Server' => 'IXR\Server\Server',
        'IXR_IntrospectionServer' => 'IXR\Server\IntrospectionServer',
        'IXR_Request' => 'IXR\Request\Request',
        'IXR_Message' => 'R\Message\Message',
        'IXR_Error' => 'XR\Message\Error',
        'IXR_Date' => 'IXR\DataType\Date',
        'IXR_Base64' => 'IXR\DataType\Base64',
        'IXR_Value' => 'IXR\DataType\Value',

        // see inc/legacy.php
        'Doku_Event_Handler' => 'dokuwiki\Extension\EventHandler',
        'Doku_Event' => 'dokuwiki\Extension\Event',
        'DokuWiki_Action_Plugin' => 'dokuwiki\Extension\ActionPlugin',
        'DokuWiki_Admin_Plugin' => 'dokuwiki\Extension\AdminPlugin',
        'DokuWiki_Auth_Plugin' => 'dokuwiki\Extension\AuthPlugin',
        'DokuWiki_CLI_Plugin' => 'dokuwiki\Extension\CLIPlugin',
        'DokuWiki_Plugin' => 'dokuwiki\Extension\Plugin',
        'DokuWiki_Remote_Plugin' => 'dokuwiki\Extension\RemotePlugin',
        'DokuWiki_Syntax_Plugin' => 'dokuwiki\Extension\SyntaxPlugin',
    ]);

    $rectorConfig->ruleWithConfiguration(RenameFunctionRector::class, [
        // see inc/deprecated.php
        'Doku_Lexer_Escape' => 'dokuwiki\Parsing\Lexer\Lexer::escape',

        // see inc/utf8.php
        'utf8_isASCII' => 'dokuwiki\Utf8\Clean::isASCII',
        'utf8_strip' => 'dokuwiki\Utf8\Clean::strip',
        'utf8_check' => 'dokuwiki\Utf8\Clean::isUtf8',
        'utf8_basename' => 'dokuwiki\Utf8\PhpString::basename',
        'utf8_strlen' => 'dokuwiki\Utf8\PhpString::strlen',
        'utf8_substr' => 'dokuwiki\Utf8\PhpString::substr',
        'utf8_substr_replace' => 'dokuwiki\Utf8\PhpString::substr_replace',
        'utf8_ltrim' => 'dokuwiki\Utf8\PhpString::ltrim',
        'utf8_rtrim' => 'dokuwiki\Utf8\PhpString::rtrim',
        'utf8_trim' => 'dokuwiki\Utf8\PhpString::trim',
        'utf8_strtolower' => 'dokuwiki\Utf8\PhpString::strtolower',
        'utf8_strtoupper' => 'dokuwiki\Utf8\PhpString::strtoupper',
        'utf8_ucfirst' => 'dokuwiki\Utf8\PhpString::ucfirst',
        'utf8_ucwords' => 'dokuwiki\Utf8\PhpString::ucwords',
        'utf8_deaccent' => 'dokuwiki\Utf8\Clean::deaccent',
        'utf8_romanize' => 'dokuwiki\Utf8\Clean::romanize',
        'utf8_stripspecials' => 'dokuwiki\Utf8\Clean::stripspecials',
        'utf8_strpos' => 'dokuwiki\Utf8\PhpString::strpos',
        'utf8_tohtml' => 'dokuwiki\Utf8\Conversion::toHtml',
        'utf8_unhtml' => 'dokuwiki\Utf8\Conversion::fromHtml',
        'utf8_to_unicode' => 'dokuwiki\Utf8\Conversion::fromUtf8',
        'unicode_to_utf8' => 'dokuwiki\Utf8\Conversion::toUtf8',
        'utf8_to_utf16be' => 'dokuwiki\Utf8\Conversion::toUtf16be',
        'utf16be_to_utf8' => 'dokuwiki\Utf8\Conversion::fromUtf16be',
        'utf8_bad_replace' => 'dokuwiki\Utf8\Clean::replaceBadBytes',
        'utf8_correctIdx' => 'dokuwiki\Utf8\Clean::correctIdx',
    ]);

    $rectorConfig->rule(DokuWikiPtlnRector::class);
    $rectorConfig->rule(DokuWikiRenamePrintToEcho::class);
};
