<?php

function disjoint_set_find(&$parents, $x) {
    if ($parents[$x] !== $x) {
        return $parents[$x] = disjoint_set_find($parents, $parents[$x]);
    }
    return $x;
}

function disjoint_set_union(&$parents, $x, $y) {
    $x = disjoint_set_find($parents, $x);
    $y = disjoint_set_find($parents, $y);

    if ($x !== $y) {
        if (rand(0, 1) == 0) {
            $parents[$x] = $y;
        } else {
            $parents[$y] = $x;
        }
    }
}

function split_file($file, $chunks, $undo) {
    $cpp_name = "$file.cpp";

    echo "Processing file $cpp_name".PHP_EOL;

    $new_files = array();
    foreach (range(0, $chunks - 1) as $n) {
        $new_files[] = "$file$n.cpp";
    }

    $is_generated = (strpos($file, 'td/generate/') === 0);

    $cmake_file = $is_generated ? 'td/generate/CMakeLists.txt' : 'CMakeLists.txt';
    $cmake = file_get_contents($cmake_file);

    $cmake_cpp_name = $cpp_name;
    $cmake_new_files = $new_files;
    if ($is_generated) {
        foreach ($cmake_new_files as &$file_ref) {
            $file_ref = str_replace('td/generate/auto/td', '${TD_AUTO_INCLUDE_DIR}', $file_ref);
        }
        $cmake_cpp_name = str_replace('td/generate/auto/td', '${TD_AUTO_INCLUDE_DIR}', $cmake_cpp_name);
    }

    if ($undo) {
        foreach ($new_files as $file) {
            if (file_exists($file)) {
                echo "Unlinking ".$file.PHP_EOL;
                unlink($file);
            }
        }

        if (strpos($cmake, $cmake_cpp_name) === false) {
            $cmake = str_replace(implode(PHP_EOL.'  ', $cmake_new_files), $cmake_cpp_name, $cmake);
            file_put_contents($cmake_file, $cmake);
        }

        return;
    }

    if (strpos($cmake, $cmake_cpp_name) !== false) {
        $cmake = str_replace($cmake_cpp_name, implode(PHP_EOL.'  ', $cmake_new_files), $cmake);
        file_put_contents($cmake_file, $cmake);
    }

    if (!file_exists($cpp_name)) {
        echo "ERROR: skip nonexistent file $cpp_name".PHP_EOL;
        return;
    }

    $lines = file($cpp_name);
    $depth = 0;
    $target_depth = 1 + $is_generated;
    $is_static = false;
    $in_define = false;
    $in_comment = false;
    $current = '';
    $common = '';
    $functions = array();
    $namespace_begin = '';
    $namespace_end = '';
    foreach ($lines as $line) {
        $add_depth = strpos($line, 'namespace ') === 0 ? 1 : (strpos($line, '}  // namespace') === 0 ? -1 : 0);
        if ($add_depth) {
            # namespace begin/end
            if ($add_depth > 0) {
              $depth += $add_depth;
            }
            if ($depth <= $target_depth) {
                if ($add_depth > 0) {
                    $namespace_begin .= $line;
                } else {
                    $namespace_end .= $line;
                }
            }
            if ($add_depth < 0) {
              $depth += $add_depth;
            }
            if ($is_static) {
                $common .= $current;
            } else {
                $functions[] = $current;
            }
            $common .= $line;
            $current = '';
            $is_static = false;
            $in_define = false;
            continue;
        }

        if (strpos($line, '#undef') === 0 && !trim($current)) {
            continue;
        }

        if ($in_comment && strpos($line, '*/') === 0) {
            $in_comment = false;
            continue;
        }
        if (strpos($line, '/*') === 0) {
            $in_comment = true;
        }
        if ($in_comment) {
            continue;
        }

        if ($depth !== $target_depth) {
            $common .= $line;
            continue;
        }

        if (strpos($line, 'static ') === 0 && $depth === $target_depth) {
            $is_static = true;
        }
        if (!trim($current) && strpos($line, '#define ') === 0) {
            $is_static = true;
            $in_define = true;
        }

        $current .= $line;
        if ((strpos($line, '}') === 0 || ($in_define && !trim($line)) || preg_match('/^[a-z].*;\s*$/i', $line)) && $depth === $target_depth) {
            # block end
            if ($is_static) {
                $common .= $current;
            } else {
                $functions[] = $current;
            }
            $current = '';
            $is_static = false;
            $in_define = false;
        }
    }
    $current = trim($current);
    if (!empty($current)) {
        fwrite(STDERR, "ERROR: $current".PHP_EOL);
        exit();
    }

    if (count($functions) < $chunks) {
        fwrite(STDERR, "ERROR: file is too small to be split more".PHP_EOL);
        return;
    }

    $deps = array();  // all functions from the same subarray must be in the same file
    $parents = array();
    foreach ($functions as $i => $f) {
        if (preg_match_all('/(?J)create_handler<(?<name>[A-Z][A-Za-z]*)>|'.
                           '(?<name>[A-Z][A-Za-z]*) (final )?: public (Td::ResultHandler|Request)|'.
                           '(CREATE_REQUEST|CREATE_NO_ARGS_REQUEST)[(](?<name>[A-Z][A-Za-z]*)|'.
                           '(?<name>complete_pending_preauthentication_requests)|'.
                           '(?<name>get_message_history_slice)|'.
                           '(Up|Down)load(?!ManagerCallback)[a-zA-Z]+C(?<name>allback)|(up|down)load_[a-z_]*_c(?<name>allback)_|'.
                           '(?<name>lazy_to_json)|'.
                           '(?<name>LogEvent)[^sA]|'.
                           '(?<name>parse)[(]|'.
                           '(?<name>store)[(]/', $f, $matches, PREG_SET_ORDER)) {
            foreach ($matches as $match) {
                $name = $match['name'];
                if ($name === 'parse' || $name === 'store') {
                    if ($is_generated) {
                        continue;
                    }
                    $name = 'LogEvent';
                }
                $deps[$name][] = $i;
            }
        }
        $parents[$i] = $i;
    }

    foreach ($deps as $func_ids) {
        foreach ($func_ids as $func_id) {
            disjoint_set_union($parents, $func_ids[0], $func_id);
        }
    }
    $sets = array();
    $set_sizes = array();
    foreach ($functions as $i => $f) {
        $parent = disjoint_set_find($parents, $i);
        if (!isset($sets[$parent])) {
            $sets[$parent] = '';
            $set_sizes[$parent] = 0;
        }
        $sets[$parent] .= $f;
        $set_sizes[$parent] += preg_match('/Td::~?Td/', $f) ? 1000000 : strlen($f);
    }
    arsort($set_sizes);

    $files = array_fill(0, $chunks, '');
    $file_sizes = array_fill(0, $chunks, 0);
    foreach ($set_sizes as $parent => $size) {
        $file_id = array_search(min($file_sizes), $file_sizes);
        $files[$file_id] .= $sets[$parent];
        $file_sizes[$file_id] += $size;
    }

    foreach ($files as $n => $f) {
        $new_content = $common.$namespace_begin.$f.$namespace_end;

        $std_methods = array();
        preg_match_all('/std::[a-z_0-9]*|td::unique(?!_)/', $new_content, $std_methods);
        $std_methods = array_unique($std_methods[0]);

        $needed_std_headers = array();
        $type_headers = array(
            'std::move' => '',
            'std::vector' => '',
            'std::string' => '',
            'std::uint32_t' => '',
            'std::int32_t' => '',
            'std::int64_t' => '',
            'td::unique' => 'algorithm',
            'std::count_if' => 'algorithm',
            'std::fill' => 'algorithm',
            'std::find' => 'algorithm',
            'std::is_sorted' => 'algorithm',
            'std::lower_bound' => 'algorithm',
            'std::max' => 'algorithm',
            'std::merge' => 'algorithm',
            'std::min' => 'algorithm',
            'std::partial_sort' => 'algorithm',
            'std::partition' => 'algorithm',
            'std::remove' => 'algorithm',
            'std::reverse' => 'algorithm',
            'std::rotate' => 'algorithm',
            'std::sort' => 'algorithm',
            'std::stable_sort' => 'algorithm',
            'std::upper_bound' => 'algorithm',
            'std::abs' => 'cmath',
            'std::isfinite' => 'cmath',
            'std::function' => 'functional',
            'std::greater' => 'functional',
            'std::reference_wrapper' => 'functional',
            'std::make_move_iterator' => 'iterator',
            'std::numeric_limits' => 'limits',
            'std::map' => 'map',
            'std::multimap' => 'map',
            'std::make_shared' => 'memory',
            'std::shared_ptr' => 'memory',
            'std::multiset' => 'set',
            'std::set' => 'set',
            'std::get' => 'tuple',
            'std::make_tuple' => 'tuple',
            'std::tie' => 'tuple',
            'std::tuple' => 'tuple',
            'std::decay_t' => 'type_traits',
            'std::is_same' => 'type_traits',
            'std::unordered_map' => 'unordered_map',
            'std::unordered_set' => 'unordered_set',
            'std::make_pair' => 'utility',
            'std::pair' => 'utility',
            'std::swap' => 'utility');
        foreach ($type_headers as $type => $header) {
            if (in_array($type, $std_methods)) {
                $std_methods = array_diff($std_methods, array($type));
                if ($header && !in_array($header, $needed_std_headers)) {
                    $needed_std_headers[] = $header;
                }
            }
        }

        if (!$std_methods) { // know all needed std headers
            $new_content = preg_replace_callback(
                '/#include <([a-z_]*)>/',
                function ($matches) use ($needed_std_headers) {
                    if (in_array($matches[1], $needed_std_headers)) {
                        return $matches[0];
                    }
                    return '';
                },
                $new_content
            );
        }

        if (!preg_match('/Td::~?Td/', $new_content)) {  // destructor Td::~Td needs to see definitions of all forward-declared classes
            $td_methods = array(
                'AccentColorId' => 'AccentColorId',
                'account_manager[_(-](?![.]get[(][)])|AccountManager[^;>]' => 'AccountManager',
                'animations_manager[_(-](?![.]get[(][)])|AnimationsManager[^;>]' => 'AnimationsManager',
                'attach_menu_manager[_(-](?![.]get[(][)])|AttachMenuManager[^;>]' => 'AttachMenuManager',
                'audios_manager[_(-](?![.]get[(][)])|AudiosManager' => 'AudiosManager',
                'auth_manager[_(-](?![.]get[(][)])|AuthManager' => 'AuthManager',
                'AutoDownloadSettings|[a-z_]*auto_download_settings' => 'AutoDownloadSettings',
                'autosave_manager[_(-](?![.]get[(][)])|AutosaveManager' => 'AutosaveManager',
                'BackgroundId' => 'BackgroundId',
                'background_manager[_(-](?![.]get[(][)])|BackgroundManager' => 'BackgroundManager',
                'BackgroundType' => 'BackgroundType',
                'Birthdate' => 'Birthdate',
                'BotMenuButton|[a-z_]*_menu_button' => 'BotMenuButton',
                'boost_manager[_(-](?![.]get[(][)])|BoostManager' => 'BoostManager',
                'bot_info_manager[_(-](?![.]get[(][)])|BotInfoManager' => 'BotInfoManager',
                'BusinessAwayMessage' => 'BusinessAwayMessage',
                'BusinessChatLink' => 'BusinessChatLink',
                'BusinessConnectedBot' => 'BusinessConnectedBot',
                'BusinessConnectionId' => 'BusinessConnectionId',
                'business_connection_manager[_(-](?![.]get[(][)])|BusinessConnectionManager' => 'BusinessConnectionManager',
                'BusinessGreetingMessage' => 'BusinessGreetingMessage',
                'BusinessInfo|business_info' => 'BusinessInfo',
                'BusinessIntro' => 'BusinessIntro',
                'business_manager[_(-](?![.]get[(][)])|BusinessManager' => 'BusinessManager',
                'BusinessRecipients' => 'BusinessRecipients',
                'BusinessWorkHours' => 'BusinessWorkHours',
                'callback_queries_manager[_(-](?![.]get[(][)])|CallbackQueriesManager' => 'CallbackQueriesManager',
                'CallId' => 'CallId',
                'call_manager[_(-](?![.]get[(][)])|CallManager' => 'CallManager',
                'ChannelId' => 'ChannelId',
                'channel_recommendation_manager[_(-](?![.]get[(][)])|ChannelRecommendationManager' => 'ChannelRecommendationManager',
                'ChatId' => 'ChatId',
                'chat_manager[_(-](?![.]get[(][)])|ChatManager([^ ;.]| [^*])' => 'ChatManager',
                'common_dialog_manager[_(-](?![.]get[(][)])|CommonDialogManager' => 'CommonDialogManager',
                'country_info_manager[_(-](?![.]get[(][)])|CountryInfoManager' => 'CountryInfoManager',
                'CustomEmojiId' => 'CustomEmojiId',
                'device_token_manager[_(-](?![.]get[(][)])|DeviceTokenManager' => 'DeviceTokenManager',
                'DialogAction[^M]' => 'DialogAction',
                'dialog_action_manager[_(-](?![.]get[(][)])|DialogActionManager' => 'DialogActionManager',
                'DialogFilter[^A-Z]' => 'DialogFilter',
                'DialogFilterId' => 'DialogFilterId',
                'dialog_filter_manager[_(-](?![.]get[(][)])|DialogFilterManager' => 'DialogFilterManager',
                'DialogId' => 'DialogId',
                'dialog_invite_link_manager[_(-](?![.]get[(][)])|DialogInviteLinkManager' => 'DialogInviteLinkManager',
                'DialogListId' => 'DialogListId',
                'DialogLocation' => 'DialogLocation',
                'dialog_manager[_(-](?![.]get[(][)])|DialogManager' => 'DialogManager',
                'DialogParticipantFilter' => 'DialogParticipantFilter',
                'dialog_participant_manager[_(-](?![.]get[(][)])|DialogParticipantManager' => 'DialogParticipantManager',
                'DialogSource' => 'DialogSource',
                'documents_manager[_(-](?![.]get[(][)])|DocumentsManager' => 'DocumentsManager',
                'download_manager[_(-](?![.]get[(][)])|DownloadManager[^C]' => 'DownloadManager',
                'DownloadManagerCallback' => 'DownloadManagerCallback',
                'EmailVerification' => 'EmailVerification',
                'EmojiGroup' => 'EmojiGroup',
                'EmojiStatus|[a-z_]*_emoji_status' => 'EmojiStatus',
                'FactCheck' => 'FactCheck',
                'file_reference_manager[_(-](?![.]get[(][)])|FileReferenceManager|file_references[)]' => 'FileReferenceManager',
                'file_manager[_(-](?![.]get[(][)])|FileManager([^ ;.]| [^*])|update_file[)]' => 'files/FileManager',
                'FolderId' => 'FolderId',
                'forum_topic_manager[_(-](?![.]get[(][)])|ForumTopicManager' => 'ForumTopicManager',
                'game_manager[_(-](?![.]get[(][)])|GameManager' => 'GameManager',
                'G[(][)]|Global[^A-Za-z]' => 'Global',
                'GlobalPrivacySettings' => 'GlobalPrivacySettings',
                'GroupCallId' => 'GroupCallId',
                'group_call_manager[_(-](?![.]get[(][)])|GroupCallManager' => 'GroupCallManager',
                'hashtag_hints[_(-](?![.]get[(][)])|HashtagHints' => 'HashtagHints',
                'inline_message_manager[_(-](?![.]get[(][)])|InlineMessageManager' => 'InlineQueriesManager',
                'inline_queries_manager[_(-](?![.]get[(][)])|InlineQueriesManager' => 'InlineQueriesManager',
                'InputBusinessChatLink' => 'InputBusinessChatLink',
                'language_pack_manager[_(-]|LanguagePackManager' => 'LanguagePackManager',
                'link_manager[_(-](?![.]get[(][)])|LinkManager' => 'LinkManager',
                'LogeventIdWithGeneration|add_log_event|delete_log_event|get_erase_log_event_promise|parse_time|store_time' => 'logevent/LogEventHelper',
                'MessageCopyOptions' => 'MessageCopyOptions',
                'MessageForwardInfo|LastForwardedMessageInfo|forward_info' => 'MessageForwardInfo',
                'MessageFullId' => 'MessageFullId',
                'MessageId' => 'MessageId',
                'message_import_manager[_(-](?![.]get[(][)])|MessageImportManager' => 'MessageImportManager',
                'MessageLinkInfo' => 'MessageLinkInfo',
                'MessageQuote' => 'MessageQuote',
                'MessageReaction|UnreadMessageReaction|[a-z_]*message[a-z_]*reaction' => 'MessageReaction',
                'MessageSearchOffset' => 'MessageSearchOffset',
                '[a-z_]*_message_sender' => 'MessageSender',
                'messages_manager[_(-](?![.]get[(][)])|MessagesManager' => 'MessagesManager',
                'MessageThreadInfo' => 'MessageThreadInfo',
                'MessageTtl' => 'MessageTtl',
                'MissingInvitee' => 'MissingInvitee',
                'notification_manager[_(-](?![.]get[(][)])|NotificationManager|notifications[)]' => 'NotificationManager',
                'notification_settings_manager[_(-](?![.]get[(][)])|NotificationSettingsManager' => 'NotificationSettingsManager',
                'option_manager[_(-](?![.]get[(][)])|OptionManager' => 'OptionManager',
                'password_manager[_(-](?![.]get[(][)])|PasswordManager' => 'PasswordManager',
                'people_nearby_manager[_(-](?![.]get[(][)])|PeopleNearbyManager' => 'PeopleNearbyManager',
                'phone_number_manager[_(-](?![.]get[(][)])|PhoneNumberManager' => 'PhoneNumberManager',
                'PhotoSizeSource' => 'PhotoSizeSource',
                'poll_manager[_(-](?![.]get[(][)])|PollManager' => 'PollManager',
                'privacy_manager[_(-](?![.]get[(][)])|PrivacyManager' => 'PrivacyManager',
                'PublicDialogType|get_public_dialog_type' => 'PublicDialogType',
                'quick_reply_manager[_(-](?![.]get[(][)])|QuickReplyManager' => 'QuickReplyManager',
                'ReactionListType|[a-z_]*_reaction_list_type' => 'ReactionListType',
                'reaction_manager[_(-](?![.]get[(][)])|ReactionManager' => 'ReactionManager',
                'ReactionNotificationSettings' => 'ReactionNotificationSettings',
                'ReactionNotificationsFrom' => 'ReactionNotificationsFrom',
                'ReactionType|[a-z_]*_reaction_type' => 'ReactionType',
                'RequestActor|RequestOnceActor' => 'RequestActor',
                'saved_messages_manager[_(-](?![.]get[(][)])|SavedMessagesManager' => 'SavedMessagesManager',
                'ScopeNotificationSettings|[a-z_]*_scope_notification_settings' => 'ScopeNotificationSettings',
                'SecretChatActor' => 'SecretChatActor',
                'secret_chats_manager[_(-]|SecretChatsManager' => 'SecretChatsManager',
                'secure_manager[_(-](?![.]get[(][)])|SecureManager' => 'SecureManager',
                'SentEmailCode' => 'SentEmailCode',
                'SharedDialog' => 'SharedDialog',
                'sponsored_message_manager[_(-](?![.]get[(][)])|SponsoredMessageManager' => 'SponsoredMessageManager',
                'state_manager[_(-](?![.]get[(][)])|StateManager' => 'StateManager',
                'statistics_manager[_(-](?![.]get[(][)])|StatisticsManager' => 'StatisticsManager',
                'StickerSetId' => 'StickerSetId',
                'stickers_manager[_(-](?![.]get[(][)])|StickersManager' => 'StickersManager',
                'storage_manager[_(-](?![.]get[(][)])|StorageManager' => 'StorageManager',
                'StoryId' => 'StoryId',
                'StoryListId' => 'StoryListId',
                'story_manager[_(-](?![.]get[(][)])|StoryManager' => 'StoryManager',
                'SuggestedAction|[a-z_]*_suggested_action' => 'SuggestedAction',
                'td_api' => 'td_api',
                'td_db[(][)]|TdDb[^A-Za-z]' => 'TdDb',
                'telegram_api' => 'telegram_api',
                'theme_manager[_(-](?![.]get[(][)])|ThemeManager' => 'ThemeManager',
                'time_zone_manager[_(-](?![.]get[(][)])|TimeZoneManager' => 'TimeZoneManager',
                'TopDialogCategory|get_top_dialog_category' => 'TopDialogCategory',
                'top_dialog_manager[_(-](?![.]get[(][)])|TopDialogManager' => 'TopDialogManager',
                'translation_manager[_(-](?![.]get[(][)])|TranslationManager' => 'TranslationManager',
                'transcription_manager[_(-](?![.]get[(][)])|TranscriptionManager' => 'TranscriptionManager',
                'updates_manager[_(-](?![.]get[(][)])|UpdatesManager|get_difference[)]|updateSentMessage|dummyUpdate' => 'UpdatesManager',
                'UserId' => 'UserId',
                'user_manager[_(-](?![.]get[(][)])|UserManager([^ ;.]| [^*])' => 'UserManager',
                'video_notes_manager[_(-](?![.]get[(][)])|VideoNotesManager' => 'VideoNotesManager',
                'videos_manager[_(-](?![.]get[(][)])|VideosManager' => 'VideosManager',
                'voice_notes_manager[_(-](?![.]get[(][)])|VoiceNotesManager' => 'VoiceNotesManager',
                'WebPageId(Hash)?' => 'WebPageId',
                'web_pages_manager[_(-](?![.]get[(][)])|WebPagesManager' => 'WebPagesManager');

            foreach ($td_methods as $pattern => $header) {
                if (strpos($cpp_name, $header) !== false) {
                    continue;
                }

                $include_name = '#include "td/telegram/'.$header.'.h"';
                if (strpos($new_content, $include_name) !== false && preg_match('/[^a-zA-Z0-9_]('.$pattern.')/', str_replace($include_name, '', $new_content)) === 0) {
                    $new_content = str_replace($include_name, '', $new_content);
                }
            }
        } else {
            $new_content = preg_replace_callback(
                '|#include "[a-z_A-Z/0-9.]*"|',
                function ($matches) {
                    if (strpos($matches[0], "Manager") !== false || strpos($matches[0], "HashtagHints") !== false || strpos($matches[0], "Td.h") !== false) {
                        return $matches[0];
                    }
                    return '';
                },
                $new_content
            );
        }

        if (!file_exists($new_files[$n]) || file_get_contents($new_files[$n]) !== $new_content) {
            echo "Writing file ".$new_files[$n].PHP_EOL;
            file_put_contents($new_files[$n], $new_content);
        }
    }
}

if (in_array('--help', $argv) || in_array('-h', $argv)) {
    echo "Usage: php SplitSource.php [OPTION]...\n".
         "Splits some source files to reduce a maximum amount of RAM needed for compiling a single file.\n".
         "  -u, --undo Undo all source code changes.\n".
         "  -h, --help Show this help.\n";
    exit(2);
}

$undo = in_array('--undo', $argv) || in_array('-u', $argv);
$files = array('td/telegram/ChatManager' => 10,
               'td/telegram/MessagesManager' => 50,
               'td/telegram/NotificationManager' => 10,
               'td/telegram/StickersManager' => 10,
               'td/telegram/Td' => 50,
               'td/telegram/UserManager' => 10,
               'td/generate/auto/td/telegram/td_api' => 10,
               'td/generate/auto/td/telegram/td_api_json' => 10,
               'td/generate/auto/td/telegram/telegram_api' => 10);

foreach ($files as $file => $chunks) {
    split_file($file, $chunks, $undo);
}