xsshend 0.5.2

Simple CLI tool for uploading files to multiple SSH servers
Documentation
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
name: Release

on:
  push:
    tags: [ 'v*' ]

env:
  CARGO_TERM_COLOR: always

permissions:
  contents: write
  pages: write
  id-token: write
  actions: write

jobs:
  # Vérification de cohérence pré-release
  pre-release-checks:
    name: Pre-release Consistency Checks
    runs-on: ubuntu-latest
    outputs:
      version: ${{ steps.version-check.outputs.version }}
      tag: ${{ steps.version-check.outputs.tag }}
      is-consistent: ${{ steps.version-check.outputs.is-consistent }}
    
    steps:
    - name: Checkout repository
      uses: actions/checkout@v4

    - name: Install Rust
      uses: dtolnay/rust-toolchain@stable

    - name: Version consistency check
      id: version-check
      uses: actions/github-script@v7
      with:
        script: |
          const fs = require('fs');
          const path = require('path');
          
          // Récupérer le tag depuis l'événement
          const tag = context.ref.replace('refs/tags/', '');
          const expectedVersion = tag.startsWith('v') ? tag.slice(1) : tag;
          
          console.log(`Tag: ${tag}`);
          console.log(`Expected version: ${expectedVersion}`);
          
          // Lire la version dans Cargo.toml
          const cargoTomlPath = path.join(process.env.GITHUB_WORKSPACE, 'Cargo.toml');
          const cargoToml = fs.readFileSync(cargoTomlPath, 'utf8');
          const cargoVersionMatch = cargoToml.match(/^version\s*=\s*"([^"]+)"/m);
          const cargoVersion = cargoVersionMatch ? cargoVersionMatch[1] : null;
          
          // Lire la version dans main.rs
          const mainRsPath = path.join(process.env.GITHUB_WORKSPACE, 'src', 'main.rs');
          const mainRs = fs.readFileSync(mainRsPath, 'utf8');
          const mainVersionMatch = mainRs.match(/\.version\s*\(\s*"([^"]+)"\s*\)/);
          const mainVersion = mainVersionMatch ? mainVersionMatch[1] : null;
          
          console.log(`Cargo.toml version: ${cargoVersion}`);
          console.log(`main.rs version: ${mainVersion}`);
          
          // Vérifier la cohérence
          let isConsistent = true;
          let errors = [];
          
          if (!cargoVersion) {
            errors.push('❌ Impossible de lire la version dans Cargo.toml');
            isConsistent = false;
          }
          
          if (!mainVersion) {
            errors.push('❌ Impossible de lire la version dans src/main.rs');
            isConsistent = false;
          }
          
          if (cargoVersion && cargoVersion !== expectedVersion) {
            errors.push(`❌ Version dans Cargo.toml (${cargoVersion}) != tag (${expectedVersion})`);
            isConsistent = false;
          }
          
          if (mainVersion && mainVersion !== expectedVersion) {
            errors.push(`❌ Version dans main.rs (${mainVersion}) != tag (${expectedVersion})`);
            isConsistent = false;
          }
          
          if (cargoVersion && mainVersion && cargoVersion !== mainVersion) {
            errors.push(`❌ Version Cargo.toml (${cargoVersion}) != main.rs (${mainVersion})`);
            isConsistent = false;
          }
          
          // Vérifier si la version existe déjà sur crates.io
          try {
            const response = await fetch(`https://crates.io/api/v1/crates/xsshend/${expectedVersion}`);
            if (response.ok) {
              console.log(`⚠️  Version ${expectedVersion} existe déjà sur crates.io`);
            } else {
              console.log(`✅ Version ${expectedVersion} n'existe pas encore sur crates.io`);
            }
          } catch (error) {
            console.log(`⚠️  Impossible de vérifier crates.io: ${error.message}`);
          }
          
          // Vérifier si la release GitHub existe déjà
          try {
            const { data: release } = await github.rest.repos.getReleaseByTag({
              owner: context.repo.owner,
              repo: context.repo.repo,
              tag: tag
            });
            errors.push(`❌ Release GitHub ${tag} existe déjà (créée le ${release.created_at})`);
            isConsistent = false;
          } catch (error) {
            if (error.status === 404) {
              console.log(`✅ Release GitHub ${tag} n'existe pas encore`);
            } else {
              console.log(`⚠️  Impossible de vérifier la release GitHub: ${error.message}`);
            }
          }
          
          // Afficher le résultat
          if (isConsistent) {
            console.log('✅ Toutes les vérifications de cohérence sont passées');
            core.summary.addHeading('🎉 Vérifications de cohérence réussies');
            core.summary.addTable([
              ['Élément', 'Version', 'Status'],
              ['Tag Git', tag, '✅'],
              ['Cargo.toml', cargoVersion || 'N/A', cargoVersion === expectedVersion ? '✅' : '❌'],
              ['main.rs', mainVersion || 'N/A', mainVersion === expectedVersion ? '✅' : '❌']
            ]);
          } else {
            console.log('❌ Des incohérences de version ont été détectées:');
            errors.forEach(error => console.log(error));
            
            core.summary.addHeading('❌ Incohérences détectées');
            core.summary.addList(errors);
            core.summary.addHeading('🔧 Actions requises');
            core.summary.addRaw(`
            Pour corriger ces problèmes:
            1. Mettez à jour la version dans \`Cargo.toml\` vers \`${expectedVersion}\`
            2. Mettez à jour la version dans \`src/main.rs\` vers \`${expectedVersion}\`
            3. Committez les changements
            4. Supprimez et recréez le tag si nécessaire
            `);
            
            core.setFailed('Version consistency check failed');
          }
          
          // Écrire le résumé
          await core.summary.write();
          
          // Définir les outputs
          core.setOutput('version', expectedVersion);
          core.setOutput('tag', tag);
          core.setOutput('is-consistent', isConsistent.toString());
          
          return {
            version: expectedVersion,
            tag: tag,
            isConsistent: isConsistent
          };

  # Tests avant release
  test:
    name: Test before release
    runs-on: ubuntu-latest
    needs: pre-release-checks
    if: needs.pre-release-checks.outputs.is-consistent == 'true'
    
    steps:
    - name: Checkout repository
      uses: actions/checkout@v4

    - name: Install Rust
      uses: dtolnay/rust-toolchain@stable
      with:
        components: rustfmt, clippy

    - name: Set up Rust cache
      uses: actions/cache@v3
      with:
        path: |
          ~/.cargo/bin/
          ~/.cargo/registry/index/
          ~/.cargo/git/db/
          target/
        key: ${{ runner.os }}-cargo-release-${{ hashFiles('**/Cargo.lock') }}

    - name: Run tests
      run: cargo test --verbose

    - name: Build release
      run: cargo build --release --verbose

    - name: Run release binary test
      run: |
        ./target/release/xsshend --version
        ./target/release/xsshend --help

  # Build release
  build:
    name: Build Release
    runs-on: ubuntu-latest
    needs: [pre-release-checks, test]
    if: needs.pre-release-checks.outputs.is-consistent == 'true'

    steps:
    - name: Checkout repository
      uses: actions/checkout@v4

    - name: Install Rust
      uses: dtolnay/rust-toolchain@stable

    - name: Set up Rust cache
      uses: actions/cache@v3
      with:
        path: |
          ~/.cargo/bin/
          ~/.cargo/registry/index/
          ~/.cargo/registry/cache/
          ~/.cargo/git/db/
          target/
        key: ${{ runner.os }}-cargo-release-${{ hashFiles('**/Cargo.lock') }}

    - name: Build release binary
      run: cargo build --release --verbose

    - name: Test release binary
      run: |
        ./target/release/xsshend --version
        ./target/release/xsshend --help

    - name: Create binary archive
      run: |
        ARCHIVE_NAME="xsshend-${{ needs.pre-release-checks.outputs.version }}-linux-x64"
        tar czf "${ARCHIVE_NAME}.tar.gz" -C target/release xsshend
        echo "ARCHIVE=${ARCHIVE_NAME}.tar.gz" >> $GITHUB_ENV

    - name: Upload build artifact
      uses: actions/upload-artifact@v4
      with:
        name: release-binary
        path: ${{ env.ARCHIVE }}

  # Publication sur crates.io
  publish:
    name: Publish to crates.io
    runs-on: ubuntu-latest
    needs: [pre-release-checks, build]
    if: needs.pre-release-checks.outputs.is-consistent == 'true'
    
    steps:
    - name: Checkout repository
      uses: actions/checkout@v4

    - name: Install Rust
      uses: dtolnay/rust-toolchain@stable

    - name: Set up Rust cache
      uses: actions/cache@v3
      with:
        path: |
          ~/.cargo/bin/
          ~/.cargo/registry/index/
          ~/.cargo/registry/cache/
          ~/.cargo/git/db/
          target/
        key: ubuntu-latest-cargo-publish-${{ hashFiles('**/Cargo.lock') }}

    - name: Publish to crates.io
      env:
        CARGO_REGISTRY_TOKEN: ${{ secrets.CARGO_REGISTRY_TOKEN }}
      run: |
        echo "Publishing xsshend version ${{ needs.pre-release-checks.outputs.version }} to crates.io..."
        cargo publish --verbose

  # Déploiement de la documentation GitHub Pages (depuis main)
  deploy-docs:
    name: Deploy Documentation to GitHub Pages
    runs-on: ubuntu-latest
    needs: [pre-release-checks, publish]
    if: needs.pre-release-checks.outputs.is-consistent == 'true'
    continue-on-error: true  # Ne pas faire échouer le workflow si Pages échoue
    
    permissions:
      contents: write
    
    steps:
    - name: Checkout main branch for docs
      uses: actions/checkout@v4
      with:
        ref: main  # Checkout main au lieu du tag pour éviter les restrictions
    
    - name: Deploy to GitHub Pages (via gh-pages branch)
      uses: peaceiris/actions-gh-pages@v3
      with:
        github_token: ${{ secrets.GITHUB_TOKEN }}
        publish_dir: ./docs
        enable_jekyll: true
        cname: # Optionnel: votre domaine custom
      continue-on-error: true

  # Création de la release GitHub
  release:
    name: Create GitHub Release
    runs-on: ubuntu-latest
    needs: [pre-release-checks, build, publish]
    if: needs.pre-release-checks.outputs.is-consistent == 'true'
    
    steps:
    - name: Checkout repository
      uses: actions/checkout@v4

    - name: Download all artifacts
      uses: actions/download-artifact@v4
      with:
        path: artifacts/

    - name: Prepare release assets
      run: |
        mkdir -p release-assets/
        find artifacts/ -name "*.tar.gz" -o -name "*.zip" | while read file; do
          cp "$file" release-assets/
        done
        # Si aucun fichier trouvé, créer un placeholder
        if [ ! "$(ls -A release-assets/)" ]; then
          echo "Aucun asset trouvé dans artifacts/" > release-assets/README.txt
        fi
        ls -la release-assets/

    - name: Generate release notes
      id: release-notes
      uses: actions/github-script@v7
      with:
        script: |
          const fs = require('fs');
          const path = require('path');
          
          // Lire les notes de release depuis RELEASE_NOTES_v0.2.2.md si elles existent
          const releaseNotesPath = path.join(process.env.GITHUB_WORKSPACE, 'RELEASE_NOTES_${{ needs.pre-release-checks.outputs.tag }}.md');
          let releaseBody = '';
          
          if (fs.existsSync(releaseNotesPath)) {
            releaseBody = fs.readFileSync(releaseNotesPath, 'utf8');
            console.log('✅ Release notes trouvées dans le fichier local');
          } else {
            // Générer des notes automatiques
            releaseBody = `## xsshend ${{ needs.pre-release-checks.outputs.version }}

            ### 🚀 Nouvelle version

            Cette release contient les dernières améliorations et corrections de bugs pour xsshend.

            ### 📦 Téléchargements

            Le binaire Linux x64 est disponible en téléchargement ci-dessous.

            ### 📋 Installation

            \`\`\`bash
            # Avec cargo (recommandé)
            cargo install xsshend

            # Ou téléchargez le binaire depuis les assets de cette release
            \`\`\`

            ### 🔗 Liens utiles

            - 📚 [Documentation](https://docs.rs/xsshend)
            - 📦 [crates.io](https://crates.io/crates/xsshend)
            - 🐛 [Signaler un bug](https://github.com/${{ github.repository }}/issues)

            ---
            *Généré automatiquement le $(date)*`;
            
            console.log('📝 Notes de release générées automatiquement');
          }
          
          core.setOutput('body', releaseBody);
          return releaseBody;

    - name: Create GitHub Release
      uses: softprops/action-gh-release@v1
      with:
        tag_name: ${{ needs.pre-release-checks.outputs.tag }}
        name: xsshend ${{ needs.pre-release-checks.outputs.version }}
        body: ${{ steps.release-notes.outputs.body }}
        draft: false
        prerelease: false
        files: release-assets/*
        token: ${{ secrets.GITHUB_TOKEN }}

    - name: Post-release verification
      uses: actions/github-script@v7
      with:
        script: |
          console.log('🎉 Release créée avec succès !');
          
          // Vérifier crates.io
          const version = '${{ needs.pre-release-checks.outputs.version }}';
          try {
            const response = await fetch(`https://crates.io/api/v1/crates/xsshend/${version}`);
            if (response.ok) {
              console.log(`✅ Version ${version} disponible sur crates.io`);
            } else {
              console.log(`⚠️  Version ${version} pas encore disponible sur crates.io (peut prendre quelques minutes)`);
            }
          } catch (error) {
            console.log(`❌ Erreur lors de la vérification crates.io: ${error.message}`);
          }
          
          // Créer un résumé final
          core.summary.addHeading('🎉 Release ${{ needs.pre-release-checks.outputs.tag }} créée avec succès');
          core.summary.addTable([
            ['Élément', 'Status', 'Lien'],
            ['GitHub Release', '✅ Créée', `https://github.com/${{ github.repository }}/releases/tag/${{ needs.pre-release-checks.outputs.tag }}`],
            ['crates.io', '⏳ En cours', `https://crates.io/crates/xsshend/${version}`],
            ['Documentation API', '⏳ En cours', `https://docs.rs/xsshend/${version}`],
            ['Documentation Web', '✅ Déployée', `https://willisback.github.io/xsshend/`]
          ]);
          
          await core.summary.write();

  # Nettoyage des artefacts
  cleanup:
    name: Cleanup artifacts
    runs-on: ubuntu-latest
    needs: [release]
    if: always()
    
    permissions:
      actions: write
    
    steps:
    - name: Delete build artifacts
      uses: geekyeggo/delete-artifact@v4
      with:
        name: release-binary
      continue-on-error: true