Creating a commit with multiple files to Github with JS on the web

Mon site est entirely static . Il est construit avec Hugo et hébergé avec Zeit . Je suis assez content de la configuration, je reçois des versions quasi instantanées et une livraison de contenu CDN très rapide et je peux faire tout ce dont j'ai besoin car je ne dois gérer aucun état.

J'ai créé un simple UI pour ce site ainsi que mon podcast creator qui me permet de publier rapidement un nouveau contenu sur mon site hébergé de manière statique.

Alors. Comment l'ai-je fait?

C’est une combinaison de Firebase Auth et de mon Repo Github, EditorJS pour créer le contenu (c’est soigné) et Octokat.js pour s’engager dans le repo, puis de l’intégration Github de Zeit pour effectuer ma construction hugo. Avec cette configuration, je peux avoir un CMS statique entièrement auto-hébergé, similaire à la façon dont un utilisateur peut créer des publications dans un CMS sauvegardé sur une base de données comme Wordpress.

Dans cet article, je vais me concentrer sur une partie de l'infrastructure, à savoir l'envoi de plusieurs fichiers dans Github, car il m'a fallu un peu de temps pour y arriver.

Le code entier peut être vu sur mon repo .

Si vous construisez une interface Web qui doit s’engager directement dans Github, la meilleure bibliothèque que j’ai trouvée est Octokat. Elle fonctionne avec CORS et semble gérer l’ensemble de la surface de l’API de Github.

Git peut être une bête complexe quand il s'agit de comprendre le fonctionnement de l'arbre, des branches et d'autres éléments, j'ai donc pris des décisions qui ont facilité la tâche.

  1. Je ne pourrai que pousser sur la branche principale connue sous le nom de heads/master .
  2. Je saurai où certains fichiers seront stockés (Hugo me force à avoir une structure de répertoires spécifique)

Dans cet esprit, le processus général pour créer un commit avec plusieurs fichiers est le suivant:

Obtenez une référence au repo.

  1. Obtenez une référence à la pointe de l’arbre sur la branche heads/master .
  2. Pour chaque fichier que nous voulons valider, créez un blob , puis stockez les références à l'identifiant, au chemin d'accès et au mode de sha dans un tableau.
  3. Créez un nouveau tree contenant tous les blobs à ajouter à la référence à la pointe de l’arbre heads/master et stockez le nouveau pointeur sha dans cet arbre.
  4. Créez un commit qui pointe vers cette nouvelle arborescence, puis appuyez sur la branche heads/master .

Le code suit à peu près ce flux. Étant donné que je peux assumer la structure du chemin pour certaines entrées, je n'ai pas besoin de créer d'interface utilisateur ni de gestion complexes pour les fichiers.

const createCommit = async (repositoryUrl, filename, data, images, commitMessage, recording) => {
  try {
    const token = localStorage.getItem('accessToken');
    const github = new Octokat({ 'token': token });
    const [user, repoName] = repositoryUrl.split('/');

    if(user === null || repoName === null) {
      alert('Please specifiy a repo');
      return;
    }
    
    const markdownPath = `site/content/${filename}.markdown`.toLowerCase();
    let repo = await github.repos(user, repoName).fetch();
    let main = await repo.git.refs('heads/master').fetch();
    let treeItems = [];

    for(let image of images) {
      let imageGit = await repo.git.blobs.create({ content: image.data, encoding: 'base64' });
      let imagePath = `site/static/images/${image.name}`.toLowerCase();
      treeItems.push({
        path: imagePath,
        sha: imageGit.sha,
        mode: "100644",
        type: "blob"
        });
    }

    if (recording) {
      let audioGit = await repo.git.blobs.create({ content: recording.data, encoding: 'base64' });
      let audioPath = `site/static/audio/${recording.name}.${recording.extension}`.toLowerCase();
      treeItems.push({
        path: audioPath,
        sha: audioGit.sha,
        mode: "100644",
        type: "blob"
        });
    }

    let markdownFile = await repo.git.blobs.create({ content: btoa(jsonEncode(data)), encoding: 'base64' });
    treeItems.push({
      path: markdownPath,
      sha: markdownFile.sha,
      mode: "100644",
      type: "blob"
    });

    let tree = await repo.git.trees.create({
      tree: treeItems,
      base_tree: main.object.sha
    });
  
    let commit = await repo.git.commits.create({
      message: `Created via Web - ${commitMessage}`,
      tree: tree.sha,
      parents: [main.object.sha]});

    main.update({sha: commit.sha})

    logToToast('Posted');
  } catch (err) {
    console.error(err);
    logToToast(err);
  }
}

Faites-moi savoir si vous avez fait quelque chose de similaire avec l'hébergement statique. Je suis très heureux de pouvoir construire une interface moderne pour une infrastructure d'hébergement entièrement sans serveur.

Qu'en est-il de Zeit?

Eh bien, c'est juste un peu tout automatique maintenant. J'utilise static-builder pour exécuter la commande hugo et c'est à peu près tout. :)

Paul Kinlan

Trying to make the web and developers better.

RSS Github Medium