Trello + Sapper JAMstack (3๋ถ€)

14429 ๋‹จ์–ด trellosveltejavascriptsapper

Sapper ์„ค์ •



Sapper ์„ค์ •์€ ์„ค์น˜๊ฐ€ ๋งค์šฐ ์‰ฝ์Šต๋‹ˆ๋‹ค. ์ž์„ธํ•œ ๋‚ด์šฉ์€ ํ™•์ธํ•˜์‹ญ์‹œ์˜คhere. ์ €๋Š” ๊ฐœ์ธ์ ์œผ๋กœ ๋กค์—…์„ ์„ ํ˜ธํ•ฉ๋‹ˆ๋‹ค.

# for Rollup
npx degit "sveltejs/sapper-template#rollup" my-app
# for webpack
npx degit "sveltejs/sapper-template#webpack" my-app
cd my-app

npm install
npm run dev & open http://localhost:3000


ํŽธ์ง‘๊ธฐ์—์„œ ํ”„๋กœ์ ํŠธ๋ฅผ ์—ด๋ฉด ์ฒ˜์Œ์—๋Š” ์œ„์••์ ์œผ๋กœ ๋ณด์ผ ์ˆ˜ ์žˆ์ง€๋งŒ Sapper์˜ SSG ๋ถ€๋ถ„์„ ์ƒ์„ฑํ•  ๋•Œ ๋งค์šฐ ์ง๊ด€์ ์ธ ํ˜ผ๋ž€์Šค๋Ÿฌ์šด ๋””๋ ‰ํ† ๋ฆฌ ๊ตฌ์กฐ๋ฅผ ์•Œ ์ˆ˜ ์žˆ์Šต๋‹ˆ๋‹ค.

โ””โ”€โ”€โ”€routes
โ”‚   โ”‚   index.svelte
โ”‚   โ”‚   about.svelte
โ”‚   โ”‚   _layout.svelte
|   |   _error.svlete
โ”‚   โ””โ”€โ”€โ”€blog
โ”‚       โ”‚   [slug].svelte
โ”‚       โ”‚   index.svelte
โ”‚       โ”‚   [slug].json.js
โ”‚       โ”‚   _posts.js
โ”‚       โ”‚   index.json.js


Sapper์—์„œ ์‹œ์ž‘ ๋ถ€๋ถ„์— _ ๋ฐ‘์ค„์ด ์žˆ๋Š” ํŒŒ์ผ์€ ์ˆจ๊น€/๊ฐœ์ธ ํŒŒ์ผ์ž…๋‹ˆ๋‹ค. Trello์—์„œ API๋ฅผ ํ†ตํ•ด ๋ธ”๋กœ๊ทธ ๊ฒŒ์‹œ๋ฌผ์„ ๊ฐ€์ ธ์˜ค๋Š” ๋กœ์ง์€ ๋ธ”๋กœ๊ทธ ๊ฒŒ์‹œ๋ฌผ์— ์ ํ•ฉํ•œ JSON ํŒŒ์ผ์„ ์ƒ์„ฑํ•˜๋Š” _posts.js ํŒŒ์ผ์—์„œ ๋ฐœ์ƒํ•ฉ๋‹ˆ๋‹ค. [slug].svelte ๋ฐ index.svelte์—๋Š” ๊ฒŒ์‹œ๋ฌผ ๋ชฉ๋ก๊ณผ ๊ฒŒ์‹œ๋ฌผ ์ž์ฒด๋ฅผ ๋‹ด๋‹นํ•˜๋Š” ํ…œํ”Œ๋ฆฟ์ด ํฌํ•จ๋˜์–ด ์žˆ์Šต๋‹ˆ๋‹ค. [slug].json.js ๋ฐ index.json.js JSON์„ ๋‚ด๋ณด๋ƒ…๋‹ˆ๋‹ค. ์ด ํŒŒ์ผ ์—†์ด๋„ ๊ฐ€๋Šฅํ•˜์ง€๋งŒ SSG๋Š” ์ˆ˜ํ–‰ํ–ˆ์ง€๋งŒ ์„ฑ๋Šฅ์— ํฐ ํƒ€๊ฒฉ์„ ์ฃผ๋ฉฐ .
_posts.js ๋ฐ index.svelte์— ์ฑ„์šธ ์ˆ˜ ์žˆ๋Š” ๋ธ”๋กœ๊ทธ ๊ฒŒ์‹œ๋ฌผ์— ๋Œ€ํ•œ ๋ฐ์ดํ„ฐ๊ฐ€ ์žˆ๋Š” JSON ๊ฐœ์ฒด๋ฅผ ๋‚ด๋ณด๋‚ด๋Š” [slug].svelte์— ๋Œ€ํ•œ ์Šค๋‹ˆํŽซ

//_posts.js

import fetch from "node-fetch";
let data = {};
let url = `https://api.trello.com/1/lists/5f538d3a842e0a3b6ce9f259/cards?key={key}&token={token}`; //this is from your previous post
let trello = async function getPosts() {
  const res = await fetch(url);
  data = await res.json();
  return (data = data.map((e) => {

    const months = ["jan", "feb", "mar","apr", "may", "jun", "jul", "aug", "sep", "oct", "nov", "dec"];
    let posted_date = new Date(e.dateLastActivity)
    let formatted_date = `${posted_date.getDate()} ${months[posted_date.getMonth()]} ${posted_date.getFullYear()}`
    let tags = e.labels.map(e=>{
      return {
        name:e.name,
        color:e.color
      }
    })

    return { title: e.name, slug: e.name.toLowerCase().replace(/[^\w ]+/g,'').replace(/ +/g,'-'), desc: e.desc, image:e.cover.scaled,time:formatted_date,tags };
  }));
};

export default trello;


index.json.js ๋ฐ [slug].json.js๋ฅผ ์•ฝ๊ฐ„ ์ˆ˜์ •ํ•˜๋ฉด ์ž๋™์œผ๋กœ Trello ํ•ญ๋ชฉ์„ ๋ธ”๋กœ๊ทธ ๊ฒŒ์‹œ๋ฌผ๋กœ ๊ฐ€์ ธ์˜ฌ ์ˆ˜ ์žˆ์Šต๋‹ˆ๋‹ค.


//index.json.js
import trello from "./_posts.js";

export async function get(req, res) {
  res.writeHead(200, {
    "Content-Type": "application/json",
  });
  let data = await trello();
  let posts = data;
  posts = posts.map((post) => {
    return {
      title: post.title,
      slug: post.slug,
      time: post.time,
      tags: post.tags
    };
  });
  res.end(JSON.stringify(posts));
}




//[slug].json.js
import trello from './_posts.js';


const lookup = new Map();


export async function get(req, res, next) {
    // the `slug` parameter is available because
    // this file is called [slug].json.js
    const { slug } = req.params;
    let data = await trello()
    data.forEach(e => {
        lookup.set(e.slug, JSON.stringify(e));
    });
    if (lookup.has(slug)) {
        res.writeHead(200, {
            'Content-Type': 'application/json'
        });

        res.end(lookup.get(slug));
    } else {
        res.writeHead(404, {
            'Content-Type': 'application/json'
        });

        res.end(JSON.stringify({
            message: `Not found`
        }));
    }


}

์ข‹์€ ์›นํŽ˜์ด์ง€ ์ฆ๊ฒจ์ฐพ๊ธฐ