pipeline {
    agent any

    environment {
        NEXUS_REGISTRY = 'https://nexus.xuqinmin.com/repository/npm-hosted/'
        GIT_URL        = 'https://xuqinmin.com/xuqmGroup/XuqmGroup-RNSDK.git'
    }

    options {
        timeout(time: 30, unit: 'MINUTES')
        buildDiscarder(logRotator(numToKeepStr: '20'))
        disableConcurrentBuilds()
    }

    parameters {
        // ── 版本升级策略 ─────────────────────────────────────────────────────
        choice(name: 'VERSION_BUMP', choices: ['patch', 'minor', 'major'],
               description: '版本升级策略: major(1.0.0→2.0.0), minor(1.0.0→1.1.0), patch(1.0.0→1.0.1)')
        booleanParam(name: 'CUSTOM_VERSION', defaultValue: false,
                     description: '勾选后使用下方自定义版本号（忽略VERSION_BUMP）')

        // ── 模块自定义版本号（仅CUSTOM_VERSION=true时生效）───────────────────
        string(name: 'COMMON_VERSION',  defaultValue: '', description: '@xuqm/rn-common 自定义版本号（仅CUSTOM_VERSION=true时生效）')
        string(name: 'IM_VERSION',      defaultValue: '', description: '@xuqm/rn-im 自定义版本号')
        string(name: 'PUSH_VERSION',    defaultValue: '', description: '@xuqm/rn-push 自定义版本号')
        string(name: 'UPDATE_VERSION',  defaultValue: '', description: '@xuqm/rn-update 自定义版本号')
        string(name: 'XWEBVIEW_VERSION',defaultValue: '', description: '@xuqm/rn-xwebview 自定义版本号')
        string(name: 'LICENSE_VERSION', defaultValue: '', description: '@xuqm/rn-license 自定义版本号')

        // ── 模块选择（每行一个）──────────────────────────────────────────
        text(name: 'MODULES', defaultValue: 'common\nim\npush\nupdate\nxwebview\nlicense',
             description: '要发布的模块，每行一个。可选: common im push update xwebview license')
    }

    stages {

        // ── 1. 检出代码 ─────────────────────────────────────────────────────────
        stage('Checkout') {
            steps {
                checkout([
                    $class: 'GitSCM',
                    branches: [[name: 'main']],
                    extensions: [[$class: 'CleanBeforeCheckout']],
                    userRemoteConfigs: scm.userRemoteConfigs
                ])
                script {
                    env.GIT_COMMIT_SHORT = bat(
                        script: '@git rev-parse --short HEAD',
                        returnStdout: true
                    ).trim()

                    // ── 解析选中的模块 ──────────────────────────────────────────
                    // 映射：参数名 → [目录, npm包名, 版本参数值]
                    def moduleMap = [
                        common:   ['packages/common',   '@xuqm/rn-common',   params.COMMON_VERSION],
                        im:       ['packages/im',        '@xuqm/rn-im',       params.IM_VERSION],
                        push:     ['packages/push',      '@xuqm/rn-push',     params.PUSH_VERSION],
                        update:   ['packages/update',    '@xuqm/rn-update',   params.UPDATE_VERSION],
                        xwebview: ['packages/xwebview',  '@xuqm/rn-xwebview', params.XWEBVIEW_VERSION],
                        license:  ['packages/license',   '@xuqm/rn-license',  params.LICENSE_VERSION],
                    ]

                    def requestedModules = params.MODULES
                        .split(/\r?\n/)
                        .collect { it.trim() }
                        .findAll  { it && moduleMap.containsKey(it) }

                    if (requestedModules.isEmpty()) {
                        error "MODULES param is empty or contains no valid module names."
                    }

                    // 保存为逗号分隔的环境变量供后续阶段使用
                    env.SELECTED_MODULES = requestedModules.join(',')

                    // ── 解析版本号 ──────────────────────────────────────────
                    for (mod in requestedModules) {
                        def (dir, pkg, ver) = moduleMap[mod]
                        if (params.CUSTOM_VERSION && ver?.trim()) {
                            // 使用自定义版本号
                            echo "Overriding ${pkg} version → ${ver.trim()}"
                            bat """
                                node -e "var fs=require('fs'),p='${dir}/package.json',j=JSON.parse(fs.readFileSync(p,'utf8'));j.version='${ver.trim()}';fs.writeFileSync(p,JSON.stringify(j,null,2)+'\\n','utf8');console.log('Updated '+p+' to '+j.version)"
                            """
                        } else if (!params.CUSTOM_VERSION) {
                            // 自动升级版本号
                            def currentVer = bat(
                                script: "@node -p \"require('./${dir}/package.json').version\"",
                                returnStdout: true
                            ).trim()
                            def parts = currentVer.tokenize('.')
                            while (parts.size() < 3) { parts.add('0') }
                            switch (params.VERSION_BUMP) {
                                case 'major':
                                    parts[0] = (parts[0].toInteger() + 1).toString()
                                    parts[1] = '0'
                                    parts[2] = '0'
                                    break
                                case 'minor':
                                    parts[1] = (parts[1].toInteger() + 1).toString()
                                    parts[2] = '0'
                                    break
                                case 'patch':
                                default:
                                    parts[2] = (parts[2].toInteger() + 1).toString()
                                    break
                            }
                            def newVer = parts.join('.')
                            echo "Auto-bumping ${pkg}: ${currentVer} → ${newVer}"
                            bat """
                                node -e "var fs=require('fs'),p='${dir}/package.json',j=JSON.parse(fs.readFileSync(p,'utf8'));j.version='${newVer}';fs.writeFileSync(p,JSON.stringify(j,null,2)+'\\n','utf8');console.log('Updated '+p+' to '+j.version)"
                            """
                        }
                    }
                }
            }
        }

        // ── 2. 发布信息 ─────────────────────────────────────────────────────────
        stage('Publish Info') {
            steps {
                script {
                    def selected = env.SELECTED_MODULES.split(',')
                    def dirMap = [
                        common:   'packages/common',
                        im:       'packages/im',
                        push:     'packages/push',
                        update:   'packages/update',
                        xwebview: 'packages/xwebview',
                        license:  'packages/license',
                    ]
                    echo "Git Commit : ${env.GIT_COMMIT_SHORT}"
                    echo "Modules to publish (${selected.size()}):"
                    for (mod in selected) {
                        def dir = dirMap[mod]
                        def ver = bat(
                            script: "@node -p \"require('./${dir}/package.json').version\"",
                            returnStdout: true
                        ).trim()
                        echo "  ${mod.padRight(10)} (${dir})  →  v${ver}"
                    }
                }
            }
        }

        // ── 3. 安装依赖 ─────────────────────────────────────────────────────────
        stage('Install Dependencies') {
            steps {
                withCredentials([usernamePassword(credentialsId: 'NEXUS_CREDS',
                                                  passwordVariable: 'NPM_PASS',
                                                  usernameVariable: 'NPM_USER')]) {
                    bat """
                        powershell -NonInteractive -Command "\$auth=[Convert]::ToBase64String([Text.Encoding]::ASCII.GetBytes(\$env:NPM_USER+':'+\$env:NPM_PASS)); Set-Content .npmrc ('@xuqm:registry=https://nexus.xuqinmin.com/repository/npm-hosted/' + [char]10 + '//nexus.xuqinmin.com/repository/npm-hosted/:_auth=' + \$auth) -Encoding ASCII"
                        npm install --legacy-peer-deps
                    """
                }
            }
        }

        // ── 4. 类型检查 ─────────────────────────────────────────────────────────
        stage('Type Check') {
            steps {
                script {
                    // 在每个选中的包目录运行类型检查（允许失败）
                    def selected = env.SELECTED_MODULES.split(',')
                    def dirMap = [
                        common:   'packages/common',
                        im:       'packages/im',
                        push:     'packages/push',
                        update:   'packages/update',
                        xwebview: 'packages/xwebview',
                        license:  'packages/license',
                    ]
                    for (mod in selected) {
                        def dir = dirMap[mod]
                        echo "Type-checking ${mod} (${dir})"
                        bat "cd ${dir} && npm run typecheck || ver>nul"
                    }
                }
            }
        }

        // ── 5. 测试 ────────────────────────────────────────────────────────────
        stage('Tests') {
            steps {
                bat 'npm test > test-results.txt 2>&1 || ver>nul'
            }
            post {
                always {
                    archiveArtifacts artifacts: 'test-results.txt', allowEmptyArchive: true
                }
            }
        }

        // ── 6. 发布到 Nexus ─────────────────────────────────────────────────────
        stage('Publish to Nexus') {
            steps {
                withCredentials([usernamePassword(credentialsId: 'NEXUS_CREDS',
                                                  passwordVariable: 'NPM_PASS',
                                                  usernameVariable: 'NPM_USER')]) {
                    script {
                        // 写入带认证信息的 .npmrc
                        bat """
                            powershell -NonInteractive -Command "\$auth=[Convert]::ToBase64String([Text.Encoding]::ASCII.GetBytes(\$env:NPM_USER+':'+\$env:NPM_PASS)); Set-Content .npmrc ('@xuqm:registry=https://nexus.xuqinmin.com/repository/npm-hosted/' + [char]10 + '//nexus.xuqinmin.com/repository/npm-hosted/:_auth=' + \$auth) -Encoding ASCII"
                        """

                        def selected = env.SELECTED_MODULES.split(',')
                        def dirMap = [
                            common:   'packages/common',
                            im:       'packages/im',
                            push:     'packages/push',
                            update:   'packages/update',
                            xwebview: 'packages/xwebview',
                            license:  'packages/license',
                        ]

                        for (mod in selected) {
                            def dir = dirMap[mod]
                            echo "Publishing ${mod} from ${dir}"
                            // 复制 .npmrc 到包目录，发布，然后清理
                            bat """
                                copy .npmrc ${dir}\\.npmrc
                                cd ${dir} && npm publish --registry %NEXUS_REGISTRY%
                                del ${dir}\\.npmrc 2>nul || exit 0
                            """
                        }
                    }
                }
            }
            post {
                always {
                    bat 'del .npmrc 2>nul || exit 0'
                }
            }
        }
    }

    post {
        success {
            script {
                def selected = env.SELECTED_MODULES?.split(',') ?: []
                echo "RN SDK 构建成功 — 已发布模块: ${selected.join(', ')} (Commit: ${env.GIT_COMMIT_SHORT})"
            }
        }
        failure {
            echo "RN SDK 构建失败，请检查日志"
        }
    }
}
