{"id":12620,"date":"2026-02-24T15:20:01","date_gmt":"2026-02-24T15:20:01","guid":{"rendered":"https:\/\/withcode.tech\/media\/?p=12620"},"modified":"2026-07-02T05:52:41","modified_gmt":"2026-07-02T05:52:41","slug":"github-actions-web-deploy-cicd","status":"publish","type":"post","link":"https:\/\/withcode.tech\/media\/github-actions-web-deploy-cicd\/","title":{"rendered":"\u3010\u5b8c\u5168\u7248\u3011GitHub Actions\u3067Web\u5236\u4f5c\u306e\u81ea\u52d5\u30c7\u30d7\u30ed\u30a4\u3092\u5b9f\u73fe\uff01CI\/CD\u69cb\u7bc9\u304b\u3089\u5b9f\u88c5\u307e\u3067\u5b9f\u4f8b\u3092\u8e0f\u307e\u3048\u3066\u8a73\u3057\u304f\u89e3\u8aac"},"content":{"rendered":"\n<div class=\"wp-block-group swl-box\"><div class=\"wp-block-group__inner-container\">\n\n<h3 class=\"wp-block-heading\">\u3053\u306e\u8a18\u4e8b\u3067\u308f\u304b\u308b\u3053\u3068<\/h3>\n\n\n<ul class=\"wp-block-list\">\n<li>GitHub Actions\u306eCI\/CD\u57fa\u672c\u6982\u5ff5\u3068\u30ef\u30fc\u30af\u30d5\u30ed\u30fc\u69cb\u6210\u8981\u7d20\u306e\u9055\u3044<\/li>\n<li>FTP\u30fbAWS S3\u30fbNetlify\u30fbVercel\u3078\u306e\u81ea\u52d5\u30c7\u30d7\u30ed\u30a4\u30ef\u30fc\u30af\u30d5\u30ed\u30fc\u306e\u66f8\u304d\u65b9<\/li>\n<li>API\u30ad\u30fc\u30fb\u30d1\u30b9\u30ef\u30fc\u30c9\u3092Secrets\u3067\u5b89\u5168\u306b\u7ba1\u7406\u3059\u308b\u624b\u9806<\/li>\n<li>\u30c6\u30b9\u30c8\u30fb\u30ea\u30f3\u30c8\u30fb\u30bb\u30ad\u30e5\u30ea\u30c6\u30a3\u30c1\u30a7\u30c3\u30af\u3092\u542b\u3080\u672c\u683c\u7684\u306aCI\/CD\u30d1\u30a4\u30d7\u30e9\u30a4\u30f3\u306e\u5b9f\u88c5\u4f8b<\/li>\n<li>\u30ef\u30fc\u30af\u30d5\u30ed\u30fc\u304c\u5b9f\u884c\u3055\u308c\u306a\u3044\u30fbFTP\u5931\u6557\u30fb\u30d3\u30eb\u30c9\u30a8\u30e9\u30fc\u306a\u3069\u30c8\u30e9\u30d6\u30eb\u30b7\u30e5\u30fc\u30c6\u30a3\u30f3\u30b0<\/li>\n<\/ul>\n\n<\/div><\/div>\n\n\n<div class=\"swell-block-balloon\"><div class=\"c-balloon -bln-left\" data-col=\"gray\"><div class=\"c-balloon__icon -square\"><img decoding=\"async\" loading=\"lazy\" src=\"https:\/\/withcode.tech\/media\/wp-content\/uploads\/2025\/06\/\u30b9\u30af\u30ea\u30fc\u30f3\u30b7\u30e7\u30c3\u30c8-2025-06-15-14.48.08.jpg\" alt=\"\" class=\"c-balloon__iconImg\" width=\"80px\" height=\"80px\"><span class=\"c-balloon__iconName\">\u751f\u5f92<\/span><\/div><div class=\"c-balloon__body -speaking -border-none\"><div class=\"c-balloon__text\">\n<p><strong>Web\u5236\u4f5c\u3067\u6bce\u56deFTP\u3067\u30d5\u30a1\u30a4\u30eb\u3092\u30a2\u30c3\u30d7\u30ed\u30fc\u30c9\u3059\u308b\u306e\u304c\u9762\u5012\u306a\u3093\u3067\u3059\u2026\u4f55\u304b\u826f\u3044\u65b9\u6cd5\u306f\u3042\u308a\u307e\u305b\u3093\u304b\uff1f<\/strong><\/p>\n<span class=\"c-balloon__shapes\"><span class=\"c-balloon__before\"><\/span><span class=\"c-balloon__after\"><\/span><\/span><\/div><\/div><\/div><\/div>\n\n<div class=\"swell-block-balloon\"><div class=\"c-balloon -bln-right\" data-col=\"gray\"><div class=\"c-balloon__icon -square\"><img decoding=\"async\" loading=\"lazy\" src=\"https:\/\/withcode.tech\/media\/wp-content\/uploads\/2025\/06\/\u30b9\u30af\u30ea\u30fc\u30f3\u30b7\u30e7\u30c3\u30c8-2025-06-15-15.11.23.jpg\" alt=\"\" class=\"c-balloon__iconImg\" width=\"80px\" height=\"80px\"><span class=\"c-balloon__iconName\">\u30da\u30f3\u535a\u58eb<\/span><\/div><div class=\"c-balloon__body -speaking -border-none\"><div class=\"c-balloon__text\">\n<p><strong>\u305d\u308c\u306a\u3089GitHub Actions\u3092\u4f7f\u3063\u305f\u81ea\u52d5\u30c7\u30d7\u30ed\u30a4\u304c\u6700\u9069\u3058\u3083\uff01\u30b3\u30fc\u30c9\u3092\u30d7\u30c3\u30b7\u30e5\u3059\u308b\u3060\u3051\u3067\u81ea\u52d5\u7684\u306b\u30b5\u30fc\u30d0\u30fc\u306b\u30c7\u30d7\u30ed\u30a4\u3055\u308c\u308b\u3093\u3058\u3083\u3088\u3002\u4eca\u65e5\u306f\u8a2d\u5b9a\u65b9\u6cd5\u304b\u3089\u5b9f\u88c5\u307e\u3067\u8a73\u3057\u304f\u89e3\u8aac\u3059\u308b\u305e\u3044\uff01<\/strong><\/p>\n<span class=\"c-balloon__shapes\"><span class=\"c-balloon__before\"><\/span><span class=\"c-balloon__after\"><\/span><\/span><\/div><\/div><\/div><\/div>\n\n\n<p class=\"wp-block-paragraph\"><strong>\u7d50\u8ad6\u304b\u3089\u8a00\u3046\u3068\u3001GitHub Actions\u3092\u4f7f\u3048\u3070\u30b3\u30fc\u30c9\u3092GitHub\u306b\u30d7\u30c3\u30b7\u30e5\u3059\u308b\u3060\u3051\u3067\u30c6\u30b9\u30c8\u30fb\u30d3\u30eb\u30c9\u30fb\u30c7\u30d7\u30ed\u30a4\u304c\u5168\u81ea\u52d5\u3067\u5b9f\u884c\u3055\u308c\u307e\u3059\u3002<\/strong>\u624b\u52d5FTP\u306e\u5de5\u6570\u3092\u30bc\u30ed\u306b\u3057\u3001\u30a2\u30c3\u30d7\u30ed\u30fc\u30c9\u6f0f\u308c\u3084\u8aa4\u64cd\u4f5c\u3082\u9632\u3052\u307e\u3059\u3002\u672c\u8a18\u4e8b\u3067\u306f\u57fa\u672c\u30bb\u30c3\u30c8\u30a2\u30c3\u30d7\u304b\u3089\u3001FTP\u30fbS3\u30fbNetlify\u30fbVercel\u3078\u306e\u5b9f\u8df5\u7684\u306a\u30c7\u30d7\u30ed\u30a4\u8a2d\u5b9a\u307e\u3067\u3001\u52d5\u304f\u30b3\u30fc\u30c9\u4ed8\u304d\u3067\u89e3\u8aac\u3057\u307e\u3059\u3002<\/p>\n\n\n\n<hr class=\"wp-block-separator has-alpha-channel-opacity\"\/>\n\n\n\n<h2 class=\"wp-block-heading\">GitHub Actions\u3068\u306f\uff1fCI\/CD\u306e\u57fa\u672c\u6982\u5ff5<\/h2>\n\n\n\n<p class=\"wp-block-paragraph\">GitHub Actions\u306f\u3001<strong>GitHub\u304c\u63d0\u4f9b\u3059\u308bCI\/CD\uff08\u7d99\u7d9a\u7684\u30a4\u30f3\u30c6\u30b0\u30ec\u30fc\u30b7\u30e7\u30f3\/\u7d99\u7d9a\u7684\u30c7\u30ea\u30d0\u30ea\u30fc\uff09\u30d7\u30e9\u30c3\u30c8\u30d5\u30a9\u30fc\u30e0<\/strong>\u3067\u3059\u3002\u30ea\u30dd\u30b8\u30c8\u30ea\u5185\u3067\u767a\u751f\u3059\u308b\u30a4\u30d9\u30f3\u30c8\uff08\u30d7\u30c3\u30b7\u30e5\u30fb\u30d7\u30eb\u30ea\u30af\u30a8\u30b9\u30c8\u30fb\u30a4\u30b7\u30e5\u30fc\u4f5c\u6210\u306a\u3069\uff09\u3092\u30c8\u30ea\u30ac\u30fc\u3068\u3057\u3066\u3001\u81ea\u52d5\u7684\u306b\u30ef\u30fc\u30af\u30d5\u30ed\u30fc\u3092\u5b9f\u884c\u3067\u304d\u307e\u3059\u3002<\/p>\n\n\n\n<h3 class=\"wp-block-heading\">\u4e3b\u8981\u306a\u69cb\u6210\u8981\u7d20<\/h3>\n\n\n\n<figure class=\"wp-block-table\"><table><thead><tr><th>\u8981\u7d20<\/th><th>\u5f79\u5272<\/th><th>\u4f8b<\/th><\/tr><\/thead><tbody><tr><td><strong>\u30ef\u30fc\u30af\u30d5\u30ed\u30fc\uff08Workflow\uff09<\/strong><\/td><td>\u81ea\u52d5\u5316\u30d7\u30ed\u30bb\u30b9\u5168\u4f53\u3002YAML\u30d5\u30a1\u30a4\u30eb\u3067\u5b9a\u7fa9<\/td><td>deploy.yml<\/td><\/tr><tr><td><strong>\u30a4\u30d9\u30f3\u30c8\uff08Event\uff09<\/strong><\/td><td>\u30ef\u30fc\u30af\u30d5\u30ed\u30fc\u3092\u958b\u59cb\u3059\u308b\u30c8\u30ea\u30ac\u30fc<\/td><td>push\u3001pull_request<\/td><\/tr><tr><td><strong>\u30b8\u30e7\u30d6\uff08Job\uff09<\/strong><\/td><td>\u30ef\u30fc\u30af\u30d5\u30ed\u30fc\u5185\u3067\u5b9f\u884c\u3055\u308c\u308b\u4e00\u9023\u306e\u30b9\u30c6\u30c3\u30d7\u306e\u96c6\u307e\u308a<\/td><td>build\u3001deploy<\/td><\/tr><tr><td><strong>\u30b9\u30c6\u30c3\u30d7\uff08Step\uff09<\/strong><\/td><td>\u30b8\u30e7\u30d6\u5185\u306e\u500b\u5225\u30bf\u30b9\u30af<\/td><td>npm install\u3001npm build<\/td><\/tr><tr><td><strong>\u30a2\u30af\u30b7\u30e7\u30f3\uff08Action\uff09<\/strong><\/td><td>\u518d\u5229\u7528\u53ef\u80fd\u306a\u30b3\u30de\u30f3\u30c9\u96c6\u5408\uff08\u30de\u30fc\u30b1\u30c3\u30c8\u30d7\u30ec\u30a4\u30b9\u3067\u516c\u958b\uff09<\/td><td>actions\/checkout@v4<\/td><\/tr><tr><td><strong>\u30e9\u30f3\u30ca\u30fc\uff08Runner\uff09<\/strong><\/td><td>\u30ef\u30fc\u30af\u30d5\u30ed\u30fc\u3092\u5b9f\u884c\u3059\u308b\u30b5\u30fc\u30d0\u30fc<\/td><td>ubuntu-latest<\/td><\/tr><\/tbody><\/table><\/figure>\n\n\n\n<h3 class=\"wp-block-heading\">CI\/CD\u306e\u5b9a\u7fa9<\/h3>\n\n\n\n<ul class=\"wp-block-list\">\n<li><strong>CI\uff08Continuous Integration \/ \u7d99\u7d9a\u7684\u30a4\u30f3\u30c6\u30b0\u30ec\u30fc\u30b7\u30e7\u30f3\uff09<\/strong>\uff1a\u958b\u767a\u8005\u304c\u30b3\u30fc\u30c9\u3092\u30ea\u30dd\u30b8\u30c8\u30ea\u306b\u983b\u7e41\u306b\u30de\u30fc\u30b8\u3057\u3001\u81ea\u52d5\u7684\u306b\u30d3\u30eb\u30c9\u3068\u30c6\u30b9\u30c8\u3092\u5b9f\u884c\u3059\u308b\u3053\u3068\u3067\u554f\u984c\u3092\u65e9\u671f\u306b\u767a\u898b\u3059\u308b\u624b\u6cd5<\/li>\n<li><strong>CD\uff08Continuous Delivery \/ \u7d99\u7d9a\u7684\u30c7\u30ea\u30d0\u30ea\u30fc\uff09<\/strong>\uff1a\u30b3\u30fc\u30c9\u306e\u5909\u66f4\u3092\u81ea\u52d5\u7684\u306b\u30d3\u30eb\u30c9\u30fb\u30c6\u30b9\u30c8\u3057\u3001\u672c\u756a\u74b0\u5883\u3078\u306e\u30c7\u30d7\u30ed\u30a4\u3092\u81ea\u52d5\u5316\u3059\u308b\u624b\u6cd5<\/li>\n<\/ul>\n\n\n\n<h3 class=\"wp-block-heading\">GitHub Actions\u306e\u30e1\u30ea\u30c3\u30c8<\/h3>\n\n\n\n<ul class=\"wp-block-list\">\n<li><strong>GitHub\u3068\u306e\u7d71\u5408<\/strong>\uff1a\u8ffd\u52a0\u30c4\u30fc\u30eb\u4e0d\u8981\u3067\u3001GitHub\u30ea\u30dd\u30b8\u30c8\u30ea\u304b\u3089\u76f4\u63a5\u4f7f\u7528\u53ef\u80fd<\/li>\n<li><strong>\u8c4a\u5bcc\u306a\u30a2\u30af\u30b7\u30e7\u30f3<\/strong>\uff1a\u30de\u30fc\u30b1\u30c3\u30c8\u30d7\u30ec\u30a4\u30b9\u306b\u6570\u5343\u306e\u30a2\u30af\u30b7\u30e7\u30f3\u304c\u516c\u958b\u3055\u308c\u3066\u3044\u308b<\/li>\n<li><strong>\u7121\u6599\u67a0\u304c\u5145\u5b9f<\/strong>\uff1a\u30d1\u30d6\u30ea\u30c3\u30af\u30ea\u30dd\u30b8\u30c8\u30ea\u306f\u7121\u6599\u3001\u30d7\u30e9\u30a4\u30d9\u30fc\u30c8\u3067\u3082\u67082,000\u5206\u307e\u3067\u7121\u6599<\/li>\n<li><strong>\u8a2d\u5b9a\u304c\u7c21\u5358<\/strong>\uff1aYAML\u30d5\u30a1\u30a4\u30eb\u3067\u76f4\u611f\u7684\u306b\u8a2d\u5b9a\u3067\u304d\u308b<\/li>\n<li><strong>\u67d4\u8edf\u6027<\/strong>\uff1a\u8907\u6570\u306eOS\u3084\u30d7\u30ed\u30b0\u30e9\u30df\u30f3\u30b0\u8a00\u8a9e\u3092\u30b5\u30dd\u30fc\u30c8<\/li>\n<\/ul>\n\n\n\n<hr class=\"wp-block-separator has-alpha-channel-opacity\"\/>\n\n\n\n<h2 class=\"wp-block-heading\">\u81ea\u52d5\u30c7\u30d7\u30ed\u30a4\u304c\u3082\u305f\u3089\u30596\u3064\u306e\u30e1\u30ea\u30c3\u30c8<\/h2>\n\n\n<div class=\"swell-block-balloon\"><div class=\"c-balloon -bln-left\" data-col=\"gray\"><div class=\"c-balloon__icon -square\"><img decoding=\"async\" loading=\"lazy\" src=\"https:\/\/withcode.tech\/media\/wp-content\/uploads\/2025\/06\/\u30b9\u30af\u30ea\u30fc\u30f3\u30b7\u30e7\u30c3\u30c8-2025-06-15-14.48.08.jpg\" alt=\"\" class=\"c-balloon__iconImg\" width=\"80px\" height=\"80px\"><span class=\"c-balloon__iconName\">\u751f\u5f92<\/span><\/div><div class=\"c-balloon__body -speaking -border-none\"><div class=\"c-balloon__text\">\n<p><strong>\u81ea\u52d5\u30c7\u30d7\u30ed\u30a4\u306b\u3059\u308b\u3068\u3069\u3093\u306a\u826f\u3044\u3053\u3068\u304c\u3042\u308b\u3093\u3067\u3059\u304b\uff1f<\/strong><\/p>\n<span class=\"c-balloon__shapes\"><span class=\"c-balloon__before\"><\/span><span class=\"c-balloon__after\"><\/span><\/span><\/div><\/div><\/div><\/div>\n\n<div class=\"swell-block-balloon\"><div class=\"c-balloon -bln-right\" data-col=\"gray\"><div class=\"c-balloon__icon -square\"><img decoding=\"async\" loading=\"lazy\" src=\"https:\/\/withcode.tech\/media\/wp-content\/uploads\/2025\/06\/\u30b9\u30af\u30ea\u30fc\u30f3\u30b7\u30e7\u30c3\u30c8-2025-06-15-15.11.23.jpg\" alt=\"\" class=\"c-balloon__iconImg\" width=\"80px\" height=\"80px\"><span class=\"c-balloon__iconName\">\u30da\u30f3\u535a\u58eb<\/span><\/div><div class=\"c-balloon__body -speaking -border-none\"><div class=\"c-balloon__text\">\n<p><strong>\u305f\u304f\u3055\u3093\u3042\u308b\u305e\uff01\u6642\u9593\u306e\u7bc0\u7d04\u306f\u3082\u3061\u308d\u3093\u3001\u30df\u30b9\u306e\u524a\u6e1b\u3001\u30c1\u30fc\u30e0\u958b\u767a\u306e\u52b9\u7387\u5316\u306a\u3069\u3001\u30e1\u30ea\u30c3\u30c8\u3060\u3089\u3051\u306a\u3093\u3058\u3083<\/strong><\/p>\n<span class=\"c-balloon__shapes\"><span class=\"c-balloon__before\"><\/span><span class=\"c-balloon__after\"><\/span><\/span><\/div><\/div><\/div><\/div>\n\n\n<ol class=\"wp-block-list\">\n<li><strong>\u4eba\u7684\u30df\u30b9\u306e\u524a\u6e1b<\/strong>\uff1a\u30d5\u30a1\u30a4\u30eb\u306e\u30a2\u30c3\u30d7\u30ed\u30fc\u30c9\u6f0f\u308c\u30fb\u8aa4\u3063\u305f\u30d5\u30a1\u30a4\u30eb\u9001\u4fe1\u30fb\u30c7\u30d7\u30ed\u30a4\u624b\u9806\u306e\u5b9f\u884c\u30df\u30b9\u304c\u30bc\u30ed\u306b\u306a\u308b<\/li>\n<li><strong>\u30c7\u30d7\u30ed\u30a4\u6642\u9593\u306e\u5927\u5e45\u77ed\u7e2e<\/strong>\uff1a\u624b\u52d5\u306710\u5206\u304b\u304b\u308b\u4f5c\u696d\u304c\u6570\u79d2\u301c\u6570\u5206\u306b\u3002\u8907\u6570\u74b0\u5883\u3078\u306e\u4e26\u884c\u30c7\u30d7\u30ed\u30a4\u3082\u53ef\u80fd<\/li>\n<li><strong>\u958b\u767a\u901f\u5ea6\u306e\u5411\u4e0a<\/strong>\uff1a\u30b3\u30fc\u30c9\u3092\u66f8\u3044\u305f\u3089\u3059\u3050\u306b\u672c\u756a\u53cd\u6620\u53ef\u80fd\u3002\u5c0f\u3055\u306a\u5909\u66f4\u3092\u983b\u7e41\u306b\u30ea\u30ea\u30fc\u30b9\u3067\u304d\u308b<\/li>\n<li><strong>\u54c1\u8cea\u306e\u5b89\u5b9a\u5316<\/strong>\uff1a\u30c7\u30d7\u30ed\u30a4\u524d\u306b\u5fc5\u305a\u81ea\u52d5\u30c6\u30b9\u30c8\u304c\u5b9f\u884c\u3055\u308c\u308b\u3002\u9759\u7684\u89e3\u6790\u3084\u30bb\u30ad\u30e5\u30ea\u30c6\u30a3\u30c1\u30a7\u30c3\u30af\u3092\u7d44\u307f\u8fbc\u3081\u308b<\/li>\n<li><strong>\u30c1\u30fc\u30e0\u958b\u767a\u306e\u52b9\u7387\u5316<\/strong>\uff1a\u30c7\u30d7\u30ed\u30a4\u65b9\u6cd5\u304cYAML\u3067\u30c9\u30ad\u30e5\u30e1\u30f3\u30c8\u5316\u3055\u308c\u308b\u3002\u8ab0\u304c\u3044\u3064\u4f55\u3092\u30c7\u30d7\u30ed\u30a4\u3057\u305f\u304b\u5c65\u6b74\u304c\u6b8b\u308b<\/li>\n<li><strong>\u30ed\u30fc\u30eb\u30d0\u30c3\u30af\u306e\u7c21\u5358\u5316<\/strong>\uff1a\u554f\u984c\u304c\u767a\u751f\u3057\u3066\u3082\u524d\u30d0\u30fc\u30b8\u30e7\u30f3\u306b\u5373\u5ea7\u306b\u623b\u305b\u308b<\/li>\n<\/ol>\n\n\n\n<hr class=\"wp-block-separator has-alpha-channel-opacity\"\/>\n\n\n\n<h2 class=\"wp-block-heading\">GitHub Actions\u306e\u57fa\u672c\u30bb\u30c3\u30c8\u30a2\u30c3\u30d7<\/h2>\n\n\n\n<h3 class=\"wp-block-heading\">\u30b9\u30c6\u30c3\u30d71\uff1a\u30ea\u30dd\u30b8\u30c8\u30ea\u3068\u30ef\u30fc\u30af\u30d5\u30ed\u30fc\u30c7\u30a3\u30ec\u30af\u30c8\u30ea\u306e\u6e96\u5099<\/h3>\n\n\n<div class=\"hcb_wrap\"><pre class=\"prism line-numbers lang-bash\" data-lang=\"Bash\"><code># \u65b0\u3057\u3044\u30d7\u30ed\u30b8\u30a7\u30af\u30c8\u306e\u5834\u5408\nmkdir my-website\ncd my-website\ngit init\ngit remote add origin https:\/\/github.com\/your-username\/my-website.git\n\n# \u65e2\u5b58\u30d7\u30ed\u30b8\u30a7\u30af\u30c8\u306e\u5834\u5408\u306f\u3001\u30ea\u30dd\u30b8\u30c8\u30ea\u3092\u30af\u30ed\u30fc\u30f3\ngit clone https:\/\/github.com\/your-username\/my-website.git\ncd my-website\n\n# \u30ef\u30fc\u30af\u30d5\u30ed\u30fc\u30c7\u30a3\u30ec\u30af\u30c8\u30ea\u3092\u4f5c\u6210\nmkdir -p .github\/workflows<\/code><\/pre><\/div>\n\n\n<h3 class=\"wp-block-heading\">\u30b9\u30c6\u30c3\u30d72\uff1a\u57fa\u672c\u7684\u306a\u30ef\u30fc\u30af\u30d5\u30ed\u30fc\u30d5\u30a1\u30a4\u30eb\u306e\u4f5c\u6210<\/h3>\n\n\n<div class=\"hcb_wrap\"><pre class=\"prism line-numbers lang-yaml\" data-lang=\"YAML\"><code># .github\/workflows\/test.yml\nname: Test Workflow\n\n# \u3053\u306e\u30ef\u30fc\u30af\u30d5\u30ed\u30fc\u3092\u5b9f\u884c\u3059\u308b\u30c8\u30ea\u30ac\u30fc\non:\n  push:\n    branches:\n      - main  # main\u30d6\u30e9\u30f3\u30c1\u3078\u306e\u30d7\u30c3\u30b7\u30e5\u6642\u306b\u5b9f\u884c\n  pull_request:\n    branches:\n      - main  # main\u30d6\u30e9\u30f3\u30c1\u3078\u306ePR\u6642\u306b\u5b9f\u884c\n\n# \u30b8\u30e7\u30d6\u306e\u5b9a\u7fa9\njobs:\n  test:\n    # \u5b9f\u884c\u74b0\u5883\n    runs-on: ubuntu-latest\n\n    # \u30b9\u30c6\u30c3\u30d7\u306e\u5b9a\u7fa9\n    steps:\n      # 1. \u30ea\u30dd\u30b8\u30c8\u30ea\u306e\u30b3\u30fc\u30c9\u3092\u30c1\u30a7\u30c3\u30af\u30a2\u30a6\u30c8\n      - name: Checkout code\n        uses: actions\/checkout@v4\n\n      # 2. Node.js\u306e\u30bb\u30c3\u30c8\u30a2\u30c3\u30d7\n      - name: Setup Node.js\n        uses: actions\/setup-node@v4\n        with:\n          node-version: '20'\n\n      # 3. \u4f9d\u5b58\u95a2\u4fc2\u306e\u30a4\u30f3\u30b9\u30c8\u30fc\u30eb\n      - name: Install dependencies\n        run: npm ci\n\n      # 4. \u30c6\u30b9\u30c8\u306e\u5b9f\u884c\n      - name: Run tests\n        run: npm test\n\n      # 5. \u30d3\u30eb\u30c9\u306e\u5b9f\u884c\n      - name: Build\n        run: npm run build<\/code><\/pre><\/div>\n\n\n<p class=\"wp-block-paragraph\">YAML\u306e\u4e3b\u8981\u30ad\u30fc\u30ef\u30fc\u30c9\u306e\u5f79\u5272\uff1a<strong>name<\/strong>\uff08GitHub\u4e0a\u3067\u306e\u8868\u793a\u540d\uff09\u30fb<strong>on<\/strong>\uff08\u30c8\u30ea\u30ac\u30fc\uff09\u30fb<strong>jobs<\/strong>\uff08\u30b8\u30e7\u30d6\u5b9a\u7fa9\uff09\u30fb<strong>runs-on<\/strong>\uff08\u5b9f\u884c\u74b0\u5883\uff09\u30fb<strong>steps<\/strong>\uff08\u51e6\u7406\u30b9\u30c6\u30c3\u30d7\uff09\u30fb<strong>uses<\/strong>\uff08\u65e2\u5b58\u30a2\u30af\u30b7\u30e7\u30f3\u547c\u3073\u51fa\u3057\uff09\u30fb<strong>run<\/strong>\uff08\u30b7\u30a7\u30eb\u30b3\u30de\u30f3\u30c9\u5b9f\u884c\uff09<\/p>\n\n\n\n<hr class=\"wp-block-separator has-alpha-channel-opacity\"\/>\n\n\n\n<h2 class=\"wp-block-heading\">\u5b9f\u8df5\uff1aFTP\u30b5\u30fc\u30d0\u30fc\u3078\u306e\u81ea\u52d5\u30c7\u30d7\u30ed\u30a4<\/h2>\n\n\n\n<h3 class=\"wp-block-heading\">\u30b9\u30c6\u30c3\u30d71\uff1a\u30b7\u30fc\u30af\u30ec\u30c3\u30c8\u60c5\u5831\u306e\u8a2d\u5b9a<\/h3>\n\n\n\n<p class=\"wp-block-paragraph\"><strong><span class=\"swl-marker mark_yellow\">\u30d1\u30b9\u30ef\u30fc\u30c9\u306a\u3069\u306e\u6a5f\u5bc6\u60c5\u5831\u306f\u7d76\u5bfe\u306b\u30b3\u30fc\u30c9\u306b\u76f4\u63a5\u66f8\u304b\u305a\u3001GitHub\u306eSecrets\u3092\u4f7f\u7528\u3057\u307e\u3059\u3002<\/span><\/strong><\/p>\n\n\n\n<ol class=\"wp-block-list\">\n<li>GitHub\u30ea\u30dd\u30b8\u30c8\u30ea\u30da\u30fc\u30b8\u3067\u300cSettings\u300d\u30bf\u30d6\u3092\u30af\u30ea\u30c3\u30af<\/li>\n<li>\u5de6\u30b5\u30a4\u30c9\u30d0\u30fc\u306e\u300cSecrets and variables\u300d\u2192\u300cActions\u300d\u3092\u30af\u30ea\u30c3\u30af<\/li>\n<li>\u300cNew repository secret\u300d\u30dc\u30bf\u30f3\u3092\u30af\u30ea\u30c3\u30af<\/li>\n<li>\u4ee5\u4e0b\u306e\u30b7\u30fc\u30af\u30ec\u30c3\u30c8\u3092\u767b\u9332\uff1a\n<ul>\n<li><code>FTP_SERVER<\/code>\uff1aFTP\u30b5\u30fc\u30d0\u30fc\u306e\u30a2\u30c9\u30ec\u30b9\uff08\u4f8b\uff1aftp.example.com\uff09<\/li>\n<li><code>FTP_USERNAME<\/code>\uff1aFTP\u30e6\u30fc\u30b6\u30fc\u540d<\/li>\n<li><code>FTP_PASSWORD<\/code>\uff1aFTP\u30d1\u30b9\u30ef\u30fc\u30c9<\/li>\n<\/ul>\n<\/li>\n<\/ol>\n\n\n\n<h3 class=\"wp-block-heading\">\u30b9\u30c6\u30c3\u30d72\uff1aFTP\u30c7\u30d7\u30ed\u30a4\u30ef\u30fc\u30af\u30d5\u30ed\u30fc\u306e\u4f5c\u6210<\/h3>\n\n\n<div class=\"hcb_wrap\"><pre class=\"prism line-numbers lang-yaml\" data-lang=\"YAML\"><code># .github\/workflows\/deploy-ftp.yml\nname: Deploy to FTP Server\n\non:\n  push:\n    branches:\n      - main  # main\u30d6\u30e9\u30f3\u30c1\u3078\u306e\u30d7\u30c3\u30b7\u30e5\u6642\u306b\u5b9f\u884c\n\njobs:\n  deploy:\n    runs-on: ubuntu-latest\n\n    steps:\n      # 1. \u30ea\u30dd\u30b8\u30c8\u30ea\u3092\u30c1\u30a7\u30c3\u30af\u30a2\u30a6\u30c8\n      - name: Checkout code\n        uses: actions\/checkout@v4\n\n      # 2. Node.js\u306e\u30bb\u30c3\u30c8\u30a2\u30c3\u30d7\uff08\u30d3\u30eb\u30c9\u304c\u5fc5\u8981\u306a\u5834\u5408\uff09\n      - name: Setup Node.js\n        uses: actions\/setup-node@v4\n        with:\n          node-version: '20'\n\n      # 3. \u4f9d\u5b58\u95a2\u4fc2\u306e\u30a4\u30f3\u30b9\u30c8\u30fc\u30eb\n      - name: Install dependencies\n        run: npm ci\n\n      # 4. \u30d7\u30ed\u30b8\u30a7\u30af\u30c8\u306e\u30d3\u30eb\u30c9\n      - name: Build project\n        run: npm run build\n\n      # 5. FTP\u30c7\u30d7\u30ed\u30a4\n      - name: Deploy via FTP\n        uses: SamKirkland\/FTP-Deploy-Action@v4.3.5\n        with:\n          # FTP\u63a5\u7d9a\u60c5\u5831\uff08Secrets\u304b\u3089\u53d6\u5f97\uff09\n          server: ${{ secrets.FTP_SERVER }}\n          username: ${{ secrets.FTP_USERNAME }}\n          password: ${{ secrets.FTP_PASSWORD }}\n\n          # \u30c7\u30d7\u30ed\u30a4\u8a2d\u5b9a\n          local-dir: .\/dist\/        # \u30ed\u30fc\u30ab\u30eb\u306e\u30c7\u30d7\u30ed\u30a4\u5bfe\u8c61\u30c7\u30a3\u30ec\u30af\u30c8\u30ea\n          server-dir: \/public_html\/ # \u30b5\u30fc\u30d0\u30fc\u5074\u306e\u30c7\u30a3\u30ec\u30af\u30c8\u30ea\n\n          # \u9664\u5916\u30d5\u30a1\u30a4\u30eb\u30fb\u30c7\u30a3\u30ec\u30af\u30c8\u30ea\n          exclude: |\n            **\/.git*\n            **\/.git*\/**\n            **\/node_modules\/**\n            **\/.env\n            **\/.DS_Store\n\n          # \u305d\u306e\u4ed6\u306e\u8a2d\u5b9a\n          dangerous-clean-slate: false  # \u30b5\u30fc\u30d0\u30fc\u5074\u3092\u5b8c\u5168\u524a\u9664\u3057\u306a\u3044\n          dry-run: false                # \u5b9f\u969b\u306b\u30c7\u30d7\u30ed\u30a4\u3092\u5b9f\u884c<\/code><\/pre><\/div>\n\n\n<figure class=\"wp-block-table\"><table><thead><tr><th>\u8a2d\u5b9a\u9805\u76ee<\/th><th>\u8aac\u660e<\/th><\/tr><\/thead><tbody><tr><td><code>local-dir<\/code><\/td><td>\u30a2\u30c3\u30d7\u30ed\u30fc\u30c9\u3059\u308b\u30ed\u30fc\u30ab\u30eb\u30c7\u30a3\u30ec\u30af\u30c8\u30ea\uff08\u672b\u5c3e\u306e\/\u304c\u91cd\u8981\uff09<\/td><\/tr><tr><td><code>server-dir<\/code><\/td><td>\u30b5\u30fc\u30d0\u30fc\u5074\u306e\u30a2\u30c3\u30d7\u30ed\u30fc\u30c9\u5148\u30c7\u30a3\u30ec\u30af\u30c8\u30ea<\/td><\/tr><tr><td><code>exclude<\/code><\/td><td>\u30a2\u30c3\u30d7\u30ed\u30fc\u30c9\u304b\u3089\u9664\u5916\u3059\u308b\u30d5\u30a1\u30a4\u30eb\u30d1\u30bf\u30fc\u30f3<\/td><\/tr><tr><td><code>dangerous-clean-slate<\/code><\/td><td>true\u306b\u3059\u308b\u3068\u30b5\u30fc\u30d0\u30fc\u5074\u3092\u5b8c\u5168\u524a\u9664\u3057\u3066\u304b\u3089\u30a2\u30c3\u30d7\u30ed\u30fc\u30c9\uff08\u901a\u5e38\u306ffalse\uff09<\/td><\/tr><tr><td><code>dry-run<\/code><\/td><td>true\u306b\u3059\u308b\u3068\u5b9f\u969b\u306e\u30a2\u30c3\u30d7\u30ed\u30fc\u30c9\u306f\u884c\u308f\u305a\u30c6\u30b9\u30c8\u306e\u307f\u5b9f\u884c<\/td><\/tr><\/tbody><\/table><\/figure>\n\n\n\n<hr class=\"wp-block-separator has-alpha-channel-opacity\"\/>\n\n\n\n<h2 class=\"wp-block-heading\">\u591a\u69d8\u306a\u30c7\u30d7\u30ed\u30a4\u5148\u3078\u306e\u5bfe\u5fdc<\/h2>\n\n\n\n<h3 class=\"wp-block-heading\">1. AWS S3\u3078\u306e\u81ea\u52d5\u30c7\u30d7\u30ed\u30a4<\/h3>\n\n\n<div class=\"hcb_wrap\"><pre class=\"prism line-numbers lang-yaml\" data-lang=\"YAML\"><code># .github\/workflows\/deploy-s3.yml\nname: Deploy to AWS S3\n\non:\n  push:\n    branches:\n      - main\n\njobs:\n  deploy:\n    runs-on: ubuntu-latest\n\n    steps:\n      - name: Checkout code\n        uses: actions\/checkout@v4\n\n      - name: Setup Node.js\n        uses: actions\/setup-node@v4\n        with:\n          node-version: '20'\n\n      - name: Install dependencies\n        run: npm ci\n\n      - name: Build\n        run: npm run build\n\n      # AWS\u8a8d\u8a3c\u60c5\u5831\u306e\u8a2d\u5b9a\n      - name: Configure AWS credentials\n        uses: aws-actions\/configure-aws-credentials@v4\n        with:\n          aws-access-key-id: ${{ secrets.AWS_ACCESS_KEY_ID }}\n          aws-secret-access-key: ${{ secrets.AWS_SECRET_ACCESS_KEY }}\n          aws-region: ap-northeast-1\n\n      # S3\u3078\u306e\u30c7\u30d7\u30ed\u30a4\n      - name: Deploy to S3\n        run: |\n          aws s3 sync .\/dist s3:\/\/your-bucket-name --delete\n\n      # CloudFront\u306e\u30ad\u30e3\u30c3\u30b7\u30e5\u524a\u9664\uff08\u30aa\u30d7\u30b7\u30e7\u30f3\uff09\n      - name: Invalidate CloudFront\n        run: |\n          aws cloudfront create-invalidation \\\n            --distribution-id ${{ secrets.CLOUDFRONT_DISTRIBUTION_ID }} \\\n            --paths \"\/*\"<\/code><\/pre><\/div>\n\n\n<p class=\"wp-block-paragraph\"><strong>\u5fc5\u8981\u306aSecrets\uff1a<\/strong><code>AWS_ACCESS_KEY_ID<\/code>\u30fb<code>AWS_SECRET_ACCESS_KEY<\/code>\u30fb<code>CLOUDFRONT_DISTRIBUTION_ID<\/code>\uff08CloudFront\u4f7f\u7528\u6642\uff09<\/p>\n\n\n\n<h3 class=\"wp-block-heading\">2. Netlify\u3078\u306e\u81ea\u52d5\u30c7\u30d7\u30ed\u30a4<\/h3>\n\n\n<div class=\"hcb_wrap\"><pre class=\"prism line-numbers lang-yaml\" data-lang=\"YAML\"><code># .github\/workflows\/deploy-netlify.yml\nname: Deploy to Netlify\n\non:\n  push:\n    branches:\n      - main\n\njobs:\n  deploy:\n    runs-on: ubuntu-latest\n\n    steps:\n      - name: Checkout code\n        uses: actions\/checkout@v4\n\n      - name: Setup Node.js\n        uses: actions\/setup-node@v4\n        with:\n          node-version: '20'\n\n      - name: Install dependencies\n        run: npm ci\n\n      - name: Build\n        run: npm run build\n\n      # Netlify\u3078\u306e\u30c7\u30d7\u30ed\u30a4\n      - name: Deploy to Netlify\n        uses: nwtgck\/actions-netlify@v3.0\n        with:\n          publish-dir: '.\/dist'\n          production-branch: main\n          github-token: ${{ secrets.GITHUB_TOKEN }}\n          deploy-message: \"Deploy from GitHub Actions\"\n        env:\n          NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }}\n          NETLIFY_SITE_ID: ${{ secrets.NETLIFY_SITE_ID }}<\/code><\/pre><\/div>\n\n\n<p class=\"wp-block-paragraph\"><strong>\u5fc5\u8981\u306aSecrets\uff1a<\/strong><code>NETLIFY_AUTH_TOKEN<\/code>\uff08Netlify\u306e\u500b\u4eba\u30a2\u30af\u30bb\u30b9\u30c8\u30fc\u30af\u30f3\uff09\u30fb<code>NETLIFY_SITE_ID<\/code>\uff08Netlify\u30b5\u30a4\u30c8\u306eID\uff09<\/p>\n\n\n\n<h3 class=\"wp-block-heading\">3. Vercel\u3078\u306e\u81ea\u52d5\u30c7\u30d7\u30ed\u30a4<\/h3>\n\n\n<div class=\"hcb_wrap\"><pre class=\"prism line-numbers lang-yaml\" data-lang=\"YAML\"><code># .github\/workflows\/deploy-vercel.yml\nname: Deploy to Vercel\n\non:\n  push:\n    branches:\n      - main\n\njobs:\n  deploy:\n    runs-on: ubuntu-latest\n\n    steps:\n      - name: Checkout code\n        uses: actions\/checkout@v4\n\n      - name: Setup Node.js\n        uses: actions\/setup-node@v4\n        with:\n          node-version: '20'\n\n      - name: Install Vercel CLI\n        run: npm install --global vercel@latest\n\n      # Vercel\u3078\u306e\u30c7\u30d7\u30ed\u30a4\n      - name: Pull Vercel Environment Information\n        run: vercel pull --yes --environment=production --token=${{ secrets.VERCEL_TOKEN }}\n\n      - name: Build Project Artifacts\n        run: vercel build --prod --token=${{ secrets.VERCEL_TOKEN }}\n\n      - name: Deploy to Vercel\n        run: vercel deploy --prebuilt --prod --token=${{ secrets.VERCEL_TOKEN }}<\/code><\/pre><\/div>\n\n\n<p class=\"wp-block-paragraph\"><strong>\u5fc5\u8981\u306aSecrets\uff1a<\/strong><code>VERCEL_TOKEN<\/code>\u30fb<code>VERCEL_ORG_ID<\/code>\uff08\u30aa\u30d7\u30b7\u30e7\u30f3\uff09\u30fb<code>VERCEL_PROJECT_ID<\/code>\uff08\u30aa\u30d7\u30b7\u30e7\u30f3\uff09<\/p>\n\n\n\n<hr class=\"wp-block-separator has-alpha-channel-opacity\"\/>\n\n\n\n<h2 class=\"wp-block-heading\">\u9ad8\u5ea6\u306aCI\/CD\u30d1\u30a4\u30d7\u30e9\u30a4\u30f3\u306e\u69cb\u7bc9<\/h2>\n\n\n<div class=\"swell-block-balloon\"><div class=\"c-balloon -bln-left\" data-col=\"gray\"><div class=\"c-balloon__icon -square\"><img decoding=\"async\" loading=\"lazy\" src=\"https:\/\/withcode.tech\/media\/wp-content\/uploads\/2025\/06\/\u30b9\u30af\u30ea\u30fc\u30f3\u30b7\u30e7\u30c3\u30c8-2025-06-15-14.48.08.jpg\" alt=\"\" class=\"c-balloon__iconImg\" width=\"80px\" height=\"80px\"><span class=\"c-balloon__iconName\">\u751f\u5f92<\/span><\/div><div class=\"c-balloon__body -speaking -border-none\"><div class=\"c-balloon__text\">\n<p><strong>\u3082\u3063\u3068\u672c\u683c\u7684\u306a\u30d1\u30a4\u30d7\u30e9\u30a4\u30f3\u3092\u4f5c\u308a\u305f\u3044\u3067\u3059\uff01<\/strong><\/p>\n<span class=\"c-balloon__shapes\"><span class=\"c-balloon__before\"><\/span><span class=\"c-balloon__after\"><\/span><\/span><\/div><\/div><\/div><\/div>\n\n<div class=\"swell-block-balloon\"><div class=\"c-balloon -bln-right\" data-col=\"gray\"><div class=\"c-balloon__icon -square\"><img decoding=\"async\" loading=\"lazy\" src=\"https:\/\/withcode.tech\/media\/wp-content\/uploads\/2025\/06\/\u30b9\u30af\u30ea\u30fc\u30f3\u30b7\u30e7\u30c3\u30c8-2025-06-15-15.11.23.jpg\" alt=\"\" class=\"c-balloon__iconImg\" width=\"80px\" height=\"80px\"><span class=\"c-balloon__iconName\">\u30da\u30f3\u535a\u58eb<\/span><\/div><div class=\"c-balloon__body -speaking -border-none\"><div class=\"c-balloon__text\">\n<p><strong>\u3088\u3057\uff01\u305d\u308c\u306a\u3089\u30c6\u30b9\u30c8\u3001\u30ea\u30f3\u30c8\u3001\u30bb\u30ad\u30e5\u30ea\u30c6\u30a3\u30c1\u30a7\u30c3\u30af\u3082\u542b\u3081\u305f\u5b8c\u5168\u306aCI\/CD\u30d1\u30a4\u30d7\u30e9\u30a4\u30f3\u3092\u4f5c\u3063\u3066\u307f\u308b\u304b\u306e\u3046<\/strong><\/p>\n<span class=\"c-balloon__shapes\"><span class=\"c-balloon__before\"><\/span><span class=\"c-balloon__after\"><\/span><\/span><\/div><\/div><\/div><\/div>\n\n\n<p class=\"wp-block-paragraph\">\u54c1\u8cea\u30c1\u30a7\u30c3\u30af\u2192\u30c6\u30b9\u30c8\u2192\u30bb\u30ad\u30e5\u30ea\u30c6\u30a3\u2192\u30d3\u30eb\u30c9\u2192\u30c7\u30d7\u30ed\u30a4\u306e5\u6bb5\u968e\u3067\u69cb\u6210\u3055\u308c\u308b\u672c\u683c\u7684\u306a\u30d1\u30a4\u30d7\u30e9\u30a4\u30f3\u3067\u3059\u3002\u30c6\u30b9\u30c8\u3068\u30bb\u30ad\u30e5\u30ea\u30c6\u30a3\u30c1\u30a7\u30c3\u30af\u306f\u4e26\u5217\u5b9f\u884c\u3059\u308b\u3053\u3068\u3067\u6642\u9593\u3092\u77ed\u7e2e\u3057\u307e\u3059\u3002<\/p>\n\n\n<div class=\"hcb_wrap\"><pre class=\"prism line-numbers lang-yaml\" data-lang=\"YAML\"><code># .github\/workflows\/ci-cd.yml\nname: CI\/CD Pipeline\n\non:\n  push:\n    branches:\n      - main\n      - develop\n  pull_request:\n    branches:\n      - main\n      - develop\n\n# \u74b0\u5883\u5909\u6570\u306e\u5b9a\u7fa9\nenv:\n  NODE_VERSION: '20'\n\njobs:\n  # \u30b8\u30e7\u30d61\uff1a\u30b3\u30fc\u30c9\u54c1\u8cea\u30c1\u30a7\u30c3\u30af\n  quality-check:\n    runs-on: ubuntu-latest\n\n    steps:\n      - name: Checkout code\n        uses: actions\/checkout@v4\n\n      - name: Setup Node.js\n        uses: actions\/setup-node@v4\n        with:\n          node-version: ${{ env.NODE_VERSION }}\n          cache: 'npm'\n\n      - name: Install dependencies\n        run: npm ci\n\n      # ESLint\u306b\u3088\u308b\u9759\u7684\u89e3\u6790\n      - name: Run ESLint\n        run: npm run lint\n\n      # Prettier\u306b\u3088\u308b\u30d5\u30a9\u30fc\u30de\u30c3\u30c8\u30c1\u30a7\u30c3\u30af\n      - name: Check code formatting\n        run: npm run format:check\n\n      # TypeScript\u306e\u578b\u30c1\u30a7\u30c3\u30af\n      - name: Type check\n        run: npm run type-check\n\n  # \u30b8\u30e7\u30d62\uff1a\u30c6\u30b9\u30c8\u5b9f\u884c\n  test:\n    runs-on: ubuntu-latest\n    needs: quality-check\n\n    steps:\n      - name: Checkout code\n        uses: actions\/checkout@v4\n\n      - name: Setup Node.js\n        uses: actions\/setup-node@v4\n        with:\n          node-version: ${{ env.NODE_VERSION }}\n          cache: 'npm'\n\n      - name: Install dependencies\n        run: npm ci\n\n      # \u30e6\u30cb\u30c3\u30c8\u30c6\u30b9\u30c8\n      - name: Run unit tests\n        run: npm run test:unit\n\n      # E2E\u30c6\u30b9\u30c8\n      - name: Run E2E tests\n        run: npm run test:e2e\n\n      # \u30ab\u30d0\u30ec\u30c3\u30b8\u30ec\u30dd\u30fc\u30c8\u306e\u751f\u6210\n      - name: Generate coverage report\n        run: npm run test:coverage\n\n      # \u30ab\u30d0\u30ec\u30c3\u30b8\u3092Codecov\u306b\u30a2\u30c3\u30d7\u30ed\u30fc\u30c9\n      - name: Upload coverage to Codecov\n        uses: codecov\/codecov-action@v4\n        with:\n          token: ${{ secrets.CODECOV_TOKEN }}\n          files: .\/coverage\/coverage-final.json\n\n  # \u30b8\u30e7\u30d63\uff1a\u30bb\u30ad\u30e5\u30ea\u30c6\u30a3\u30c1\u30a7\u30c3\u30af\n  security:\n    runs-on: ubuntu-latest\n    needs: quality-check\n\n    steps:\n      - name: Checkout code\n        uses: actions\/checkout@v4\n\n      - name: Setup Node.js\n        uses: actions\/setup-node@v4\n        with:\n          node-version: ${{ env.NODE_VERSION }}\n\n      - name: Install dependencies\n        run: npm ci\n\n      # \u4f9d\u5b58\u95a2\u4fc2\u306e\u8106\u5f31\u6027\u30c1\u30a7\u30c3\u30af\n      - name: Run npm audit\n        run: npm audit --audit-level=moderate\n\n      # Snyk\u306b\u3088\u308b\u30bb\u30ad\u30e5\u30ea\u30c6\u30a3\u30b9\u30ad\u30e3\u30f3\n      - name: Run Snyk security scan\n        uses: snyk\/actions\/node@master\n        env:\n          SNYK_TOKEN: ${{ secrets.SNYK_TOKEN }}\n\n  # \u30b8\u30e7\u30d64\uff1a\u30d3\u30eb\u30c9\n  build:\n    runs-on: ubuntu-latest\n    needs: [test, security]\n\n    steps:\n      - name: Checkout code\n        uses: actions\/checkout@v4\n\n      - name: Setup Node.js\n        uses: actions\/setup-node@v4\n        with:\n          node-version: ${{ env.NODE_VERSION }}\n          cache: 'npm'\n\n      - name: Install dependencies\n        run: npm ci\n\n      - name: Build project\n        run: npm run build\n\n      # \u30d3\u30eb\u30c9\u6210\u679c\u7269\u3092\u30a2\u30fc\u30c6\u30a3\u30d5\u30a1\u30af\u30c8\u3068\u3057\u3066\u4fdd\u5b58\n      - name: Upload build artifacts\n        uses: actions\/upload-artifact@v4\n        with:\n          name: build-artifacts\n          path: dist\/\n          retention-days: 7\n\n  # \u30b8\u30e7\u30d65\uff1a\u30c7\u30d7\u30ed\u30a4\uff08main\u30d6\u30e9\u30f3\u30c1\u306e\u307f\uff09\n  deploy:\n    runs-on: ubuntu-latest\n    needs: build\n    if: github.ref == 'refs\/heads\/main' &amp;&amp; github.event_name == 'push'\n\n    steps:\n      - name: Checkout code\n        uses: actions\/checkout@v4\n\n      # \u30d3\u30eb\u30c9\u6210\u679c\u7269\u3092\u30c0\u30a6\u30f3\u30ed\u30fc\u30c9\n      - name: Download build artifacts\n        uses: actions\/download-artifact@v4\n        with:\n          name: build-artifacts\n          path: dist\/\n\n      # FTP\u30c7\u30d7\u30ed\u30a4\n      - name: Deploy to production\n        uses: SamKirkland\/FTP-Deploy-Action@v4.3.5\n        with:\n          server: ${{ secrets.FTP_SERVER }}\n          username: ${{ secrets.FTP_USERNAME }}\n          password: ${{ secrets.FTP_PASSWORD }}\n          local-dir: .\/dist\/\n          server-dir: \/public_html\/\n\n      # Slack\u306b\u901a\u77e5\uff08\u30aa\u30d7\u30b7\u30e7\u30f3\uff09\n      - name: Notify deployment success\n        uses: 8398a7\/action-slack@v3\n        with:\n          status: ${{ job.status }}\n          text: 'Deployment to production completed successfully!'\n          webhook_url: ${{ secrets.SLACK_WEBHOOK }}\n        if: always()<\/code><\/pre><\/div>\n\n\n<p class=\"wp-block-paragraph\">\u3053\u306e\u30d1\u30a4\u30d7\u30e9\u30a4\u30f3\u306e\u7279\u5fb4\uff1a<strong>\u6bb5\u968e\u7684\u306a\u5b9f\u884c<\/strong>\uff08\u54c1\u8cea\u30c1\u30a7\u30c3\u30af\u2192\u30c6\u30b9\u30c8\u2192\u30bb\u30ad\u30e5\u30ea\u30c6\u30a3\u2192\u30d3\u30eb\u30c9\u2192\u30c7\u30d7\u30ed\u30a4\uff09\u3001<strong>\u4e26\u5217\u5b9f\u884c<\/strong>\uff08\u30c6\u30b9\u30c8\u3068\u30bb\u30ad\u30e5\u30ea\u30c6\u30a3\u30c1\u30a7\u30c3\u30af\u3092\u540c\u6642\u5b9f\u884c\uff09\u3001<strong>\u6761\u4ef6\u5206\u5c90<\/strong>\uff08\u30c7\u30d7\u30ed\u30a4\u306fmain\u30d6\u30e9\u30f3\u30c1\u3078\u306e\u30d7\u30c3\u30b7\u30e5\u6642\u306e\u307f\uff09\u3001<strong>\u30a2\u30fc\u30c6\u30a3\u30d5\u30a1\u30af\u30c8\u7ba1\u7406<\/strong>\uff08\u30d3\u30eb\u30c9\u6210\u679c\u7269\u3092\u4fdd\u5b58\u30fb\u518d\u5229\u7528\uff09\u3001<strong>Slack\u901a\u77e5<\/strong>\uff08\u30c7\u30d7\u30ed\u30a4\u5b8c\u4e86\u306e\u81ea\u52d5\u901a\u77e5\uff09<\/p>\n\n\n\n<hr class=\"wp-block-separator has-alpha-channel-opacity\"\/>\n\n\n\n<h2 class=\"wp-block-heading\">\u30c8\u30e9\u30d6\u30eb\u30b7\u30e5\u30fc\u30c6\u30a3\u30f3\u30b0<\/h2>\n\n\n\n<figure class=\"wp-block-table\"><table><thead><tr><th>\u554f\u984c<\/th><th>\u4e3b\u306a\u539f\u56e0<\/th><th>\u89e3\u6c7a\u6cd5<\/th><\/tr><\/thead><tbody><tr><td><strong>\u30ef\u30fc\u30af\u30d5\u30ed\u30fc\u304c\u5b9f\u884c\u3055\u308c\u306a\u3044<\/strong><\/td><td>YAML\u30d5\u30a1\u30a4\u30eb\u306e\u914d\u7f6e\u5834\u6240\u304c\u9593\u9055\u3063\u3066\u3044\u308b\u30fb\u69cb\u6587\u30a8\u30e9\u30fc\u30fb\u30c8\u30ea\u30ac\u30fc\u30d6\u30e9\u30f3\u30c1\u540d\u304c\u4e0d\u4e00\u81f4<\/td><td><code>.github\/workflows\/<\/code>\u306b\u914d\u7f6e\u3055\u308c\u3066\u3044\u308b\u304b\u78ba\u8a8d\u3002YAML Linter\u3067\u69cb\u6587\u30c1\u30a7\u30c3\u30af\u3002GitHub\u306e\u300cActions\u300d\u30bf\u30d6\u3067\u30a8\u30e9\u30fc\u30e1\u30c3\u30bb\u30fc\u30b8\u3092\u78ba\u8a8d<\/td><\/tr><tr><td><strong>FTP\u30c7\u30d7\u30ed\u30a4\u304c\u5931\u6557\u3059\u308b<\/strong><\/td><td>FTP\u8a8d\u8a3c\u60c5\u5831\u304c\u9593\u9055\u3063\u3066\u3044\u308b\u30fb\u30b5\u30fc\u30d0\u30fc\u5074\u306e\u30c7\u30a3\u30ec\u30af\u30c8\u30ea\u30d1\u30b9\u304c\u9593\u9055\u3063\u3066\u3044\u308b\u30fb\u30d5\u30a1\u30a4\u30a2\u30a6\u30a9\u30fc\u30eb\u3067\u30d6\u30ed\u30c3\u30af\u3055\u308c\u3066\u3044\u308b<\/td><td>Secrets\u306e\u5024\u3092\u518d\u78ba\u8a8d\u3002<code>dry-run: true<\/code>\u3067\u30c6\u30b9\u30c8\u5b9f\u884c\u3002\u30b5\u30fc\u30d0\u30fc\u7ba1\u7406\u8005\u306bGitHub Actions\u306eIP\u30a2\u30c9\u30ec\u30b9\u3092\u8a31\u53ef\u3057\u3066\u3082\u3089\u3046<\/td><\/tr><tr><td><strong>\u30d3\u30eb\u30c9\u304c\u5931\u6557\u3059\u308b<\/strong><\/td><td>\u74b0\u5883\u5909\u6570\u304c\u8a2d\u5b9a\u3055\u308c\u3066\u3044\u306a\u3044\u30fbNode.js\u306e\u30d0\u30fc\u30b8\u30e7\u30f3\u304c\u5408\u308f\u306a\u3044\u30fb\u4f9d\u5b58\u95a2\u4fc2\u306e\u30a4\u30f3\u30b9\u30c8\u30fc\u30eb\u30a8\u30e9\u30fc<\/td><td>\u5fc5\u8981\u306a\u74b0\u5883\u5909\u6570\u3092Secrets\u307e\u305f\u306fenv\u306b\u8ffd\u52a0\u3002<code>node-version<\/code>\u3092package.json\u306e<code>engines<\/code>\u3068\u5408\u308f\u305b\u308b\u3002<code>npm ci<\/code>\u306e\u4ee3\u308f\u308a\u306b<code>npm install<\/code>\u3092\u8a66\u3059<\/td><\/tr><\/tbody><\/table><\/figure>\n\n\n\n<hr class=\"wp-block-separator has-alpha-channel-opacity\"\/>\n\n\n\n<h2 class=\"wp-block-heading\">\u3088\u304f\u3042\u308b\u8cea\u554f<\/h2>\n\n\n\n<h3 class=\"wp-block-heading\">GitHub Actions\u306f\u6709\u6599\u3067\u3059\u304b\uff1f<\/h3>\n\n\n\n<p class=\"wp-block-paragraph\">\u30d1\u30d6\u30ea\u30c3\u30af\u30ea\u30dd\u30b8\u30c8\u30ea\u306f\u5b8c\u5168\u7121\u6599\u3067\u5229\u7528\u3067\u304d\u307e\u3059\u3002\u30d7\u30e9\u30a4\u30d9\u30fc\u30c8\u30ea\u30dd\u30b8\u30c8\u30ea\u306fGitHub\u30d7\u30e9\u30f3\u3054\u3068\u306b\u7121\u6599\u67a0\u304c\u3042\u308a\u3001Free\uff08\u67082,000\u5206\uff09\u30fbPro\uff08\u67083,000\u5206\uff09\u30fbTeam\uff08\u67083,000\u5206\uff09\u30fbEnterprise\uff08\u670850,000\u5206\uff09\u3067\u3059\u3002ubuntu-latest\u74b0\u5883\u306f1\u5206\u3042\u305f\u308a1\u5206\u3068\u3057\u3066\u8a08\u7b97\u3055\u308c\u307e\u3059\u3002\u4e00\u822c\u7684\u306aWeb\u5236\u4f5c\u30d7\u30ed\u30b8\u30a7\u30af\u30c8\u3067\u3042\u308c\u3070\u3001\u7121\u6599\u67a0\u3067\u5341\u5206\u5bfe\u5fdc\u3067\u304d\u307e\u3059\u3002<\/p>\n\n\n\n<h3 class=\"wp-block-heading\">FTP\u63a5\u7d9a\u60c5\u5831\u3092\u30b3\u30fc\u30c9\u306b\u66f8\u3044\u3066\u3057\u307e\u3044\u307e\u3057\u305f\u3002\u3069\u3046\u3059\u308c\u3070\u3044\u3044\u3067\u3059\u304b\uff1f<\/h3>\n\n\n\n<p class=\"wp-block-paragraph\">\u5373\u5ea7\u306bFTP\u30d1\u30b9\u30ef\u30fc\u30c9\u3092\u5909\u66f4\u3057\u3066\u304f\u3060\u3055\u3044\u3002GitHub\u306eSecrets\u306b\u65b0\u3057\u3044\u8a8d\u8a3c\u60c5\u5831\u3092\u767b\u9332\u3057\u76f4\u3057\u3001\u30ef\u30fc\u30af\u30d5\u30ed\u30fc\u30d5\u30a1\u30a4\u30eb\u304b\u3089\u8a8d\u8a3c\u60c5\u5831\u3092\u524a\u9664\u3057\u3066\u30b3\u30df\u30c3\u30c8\u3057\u3066\u304f\u3060\u3055\u3044\u3002\u3082\u3057\u65e2\u306bGit\u5c65\u6b74\u306b\u30d1\u30b9\u30ef\u30fc\u30c9\u304c\u542b\u307e\u308c\u3066\u3044\u308b\u5834\u5408\u306f\u3001<code>git filter-repo<\/code>\u3067\u5c65\u6b74\u304b\u3089\u524a\u9664\u3059\u308b\u5fc5\u8981\u304c\u3042\u308a\u307e\u3059\u3002GitHub\u5074\u306e\u81ea\u52d5\u30b9\u30ad\u30e3\u30f3\u3067\u691c\u51fa\u3055\u308c\u305f\u5834\u5408\u306f\u30a2\u30e9\u30fc\u30c8\u304c\u5c4a\u304d\u307e\u3059\u3002<\/p>\n\n\n\n<h3 class=\"wp-block-heading\">\u30c7\u30d7\u30ed\u30a4\u306b\u5931\u6557\u3057\u305f\u5834\u5408\u3001\u524d\u306e\u30d0\u30fc\u30b8\u30e7\u30f3\u306b\u623b\u305b\u307e\u3059\u304b\uff1f<\/h3>\n\n\n\n<p class=\"wp-block-paragraph\">\u306f\u3044\u3001\u53ef\u80fd\u3067\u3059\u3002GitHub\u306e\u300cActions\u300d\u30bf\u30d6\u304b\u3089\u4ee5\u524d\u306e\u6210\u529f\u3057\u305f\u30ef\u30fc\u30af\u30d5\u30ed\u30fc\u3092\u9078\u629e\u3057\u3001\u300cRe-run jobs\u300d\u3067\u518d\u5b9f\u884c\u3059\u308b\u3053\u3068\u3067\u524d\u306e\u30b3\u30fc\u30c9\u3092\u518d\u30c7\u30d7\u30ed\u30a4\u3067\u304d\u307e\u3059\u3002\u307e\u305f\u3001Git\u30ea\u30dd\u30b8\u30c8\u30ea\u3067\u524d\u306e\u30b3\u30df\u30c3\u30c8\u306b\u623b\u3057\u3066\u304b\u3089\u30d7\u30c3\u30b7\u30e5\u3059\u308b\u3053\u3068\u3067\u3082\u540c\u69d8\u306e\u52b9\u679c\u304c\u5f97\u3089\u308c\u307e\u3059\u3002FTP\u30c7\u30d7\u30ed\u30a4\u306e\u5834\u5408\u306f\u30b5\u30fc\u30d0\u30fc\u4e0a\u306e\u30d5\u30a1\u30a4\u30eb\u3082\u66f4\u65b0\u3055\u308c\u307e\u3059\u3002<\/p>\n\n\n\n<h3 class=\"wp-block-heading\">develop\u30d6\u30e9\u30f3\u30c1\u306f\u30b9\u30c6\u30fc\u30b8\u30f3\u30b0\u74b0\u5883\u3001main\u30d6\u30e9\u30f3\u30c1\u306f\u672c\u756a\u74b0\u5883\u306b\u5206\u3051\u3089\u308c\u307e\u3059\u304b\uff1f<\/h3>\n\n\n\n<p class=\"wp-block-paragraph\">\u306f\u3044\u3001\u3067\u304d\u307e\u3059\u3002\u30ef\u30fc\u30af\u30d5\u30ed\u30fc\u30d5\u30a1\u30a4\u30eb\u306e<code>on.push.branches<\/code>\u306b\u4e21\u65b9\u306e\u30d6\u30e9\u30f3\u30c1\u3092\u6307\u5b9a\u3057\u3001\u30c7\u30d7\u30ed\u30a4\u30b8\u30e7\u30d6\u306b<code>if: github.ref == 'refs\/heads\/main'<\/code>\u3084<code>if: github.ref == 'refs\/heads\/develop'<\/code>\u306e\u6761\u4ef6\u3092\u8ffd\u52a0\u3059\u308b\u3060\u3051\u3067\u3059\u3002\u305d\u308c\u305e\u308c\u306b\u7570\u306a\u308bSecrets\uff08\u30b9\u30c6\u30fc\u30b8\u30f3\u30b0\u7528FTP\u30b5\u30fc\u30d0\u30fc\u30fb\u672c\u756a\u7528FTP\u30b5\u30fc\u30d0\u30fc\uff09\u3092\u8a2d\u5b9a\u3059\u308b\u3053\u3068\u3067\u3001\u30d6\u30e9\u30f3\u30c1\u3054\u3068\u306b\u7570\u306a\u308b\u74b0\u5883\u3078\u30c7\u30d7\u30ed\u30a4\u3067\u304d\u307e\u3059\u3002<\/p>\n\n\n\n<h3 class=\"wp-block-heading\">WordPress\u30b5\u30a4\u30c8\u3067\u3082GitHub Actions\u306f\u4f7f\u3048\u307e\u3059\u304b\uff1f<\/h3>\n\n\n\n<p class=\"wp-block-paragraph\">\u4f7f\u3048\u307e\u3059\u3002WordPress\u30c6\u30fc\u30de\u3084\u30d7\u30e9\u30b0\u30a4\u30f3\u306e\u958b\u767a\u306bGitHub\u3092\u4f7f\u3063\u3066\u3044\u308b\u5834\u5408\u3001FTP\u30c7\u30d7\u30ed\u30a4\u30a2\u30af\u30b7\u30e7\u30f3\u3067\u30c6\u30fc\u30de\u30d5\u30a1\u30a4\u30eb\u3092\u30b5\u30fc\u30d0\u30fc\u306e<code>wp-content\/themes\/<\/code>\u306b\u81ea\u52d5\u8ee2\u9001\u3067\u304d\u307e\u3059\u3002\u30d3\u30eb\u30c9\u30b9\u30c6\u30c3\u30d7\u304c\u4e0d\u8981\u306a\u5834\u5408\u306f\u3001\u30c1\u30a7\u30c3\u30af\u30a2\u30a6\u30c8\u2192FTP\u30c7\u30d7\u30ed\u30a4\u306e2\u30b9\u30c6\u30c3\u30d7\u306e\u30b7\u30f3\u30d7\u30eb\u306a\u30ef\u30fc\u30af\u30d5\u30ed\u30fc\u3067\u52d5\u4f5c\u3057\u307e\u3059\u3002\u305f\u3060\u3057<code>wp-config.php<\/code>\u306a\u3069\u306e\u6a5f\u5bc6\u30d5\u30a1\u30a4\u30eb\u306f\u5fc5\u305a<code>exclude<\/code>\u30ea\u30b9\u30c8\u306b\u5165\u308c\u3066\u304f\u3060\u3055\u3044\u3002<\/p>\n\n\n\n<hr class=\"wp-block-separator has-alpha-channel-opacity\"\/>\n\n\n<div class=\"swell-block-balloon\"><div class=\"c-balloon -bln-left\" data-col=\"gray\"><div class=\"c-balloon__icon -square\"><img decoding=\"async\" loading=\"lazy\" src=\"https:\/\/withcode.tech\/media\/wp-content\/uploads\/2025\/06\/\u30b9\u30af\u30ea\u30fc\u30f3\u30b7\u30e7\u30c3\u30c8-2025-06-15-15.06.05.jpg\" alt=\"\" class=\"c-balloon__iconImg\" width=\"80px\" height=\"80px\"><span class=\"c-balloon__iconName\">\u751f\u5f92<\/span><\/div><div class=\"c-balloon__body -speaking -border-none\"><div class=\"c-balloon__text\">\n<p><strong>GitHub Actions\u3001\u601d\u3063\u305f\u3088\u308a\u7c21\u5358\u306b\u8a2d\u5b9a\u3067\u304d\u307e\u3059\u306d\uff01<\/strong><\/p>\n<span class=\"c-balloon__shapes\"><span class=\"c-balloon__before\"><\/span><span class=\"c-balloon__after\"><\/span><\/span><\/div><\/div><\/div><\/div>\n\n<div class=\"swell-block-balloon\"><div class=\"c-balloon -bln-right\" data-col=\"gray\"><div class=\"c-balloon__icon -square\"><img decoding=\"async\" loading=\"lazy\" src=\"https:\/\/withcode.tech\/media\/wp-content\/uploads\/2025\/06\/\u30b9\u30af\u30ea\u30fc\u30f3\u30b7\u30e7\u30c3\u30c8-2025-06-15-15.12.07.jpg\" alt=\"\" class=\"c-balloon__iconImg\" width=\"80px\" height=\"80px\"><span class=\"c-balloon__iconName\">\u30da\u30f3\u535a\u58eb<\/span><\/div><div class=\"c-balloon__body -speaking -border-none\"><div class=\"c-balloon__text\">\n<p><strong>\u305d\u306e\u901a\u308a\u3058\u3083\uff01\u6700\u521d\u306f\u57fa\u672c\u7684\u306a\u30c7\u30d7\u30ed\u30a4\u304b\u3089\u59cb\u3081\u3066\u3001\u5f90\u3005\u306b\u30c6\u30b9\u30c8\u3084\u30bb\u30ad\u30e5\u30ea\u30c6\u30a3\u30c1\u30a7\u30c3\u30af\u3092\u8ffd\u52a0\u3057\u3066\u3044\u304f\u306e\u304c\u30b3\u30c4\u3058\u3083\u305e\u3002\u4e00\u5ea6\u8a2d\u5b9a\u3059\u308c\u3070\u3001\u958b\u767a\u52b9\u7387\u304c\u683c\u6bb5\u306b\u4e0a\u304c\u308b\u3093\u3058\u3083\uff01<\/strong><\/p>\n<span class=\"c-balloon__shapes\"><span class=\"c-balloon__before\"><\/span><span class=\"c-balloon__after\"><\/span><\/span><\/div><\/div><\/div><\/div>\n\n<div class=\"swell-block-balloon\"><div class=\"c-balloon -bln-left\" data-col=\"gray\"><div class=\"c-balloon__icon -square\"><img decoding=\"async\" loading=\"lazy\" src=\"https:\/\/withcode.tech\/media\/wp-content\/uploads\/2025\/06\/\u30b9\u30af\u30ea\u30fc\u30f3\u30b7\u30e7\u30c3\u30c8-2025-06-15-14.48.08.jpg\" alt=\"\" class=\"c-balloon__iconImg\" width=\"80px\" height=\"80px\"><span class=\"c-balloon__iconName\">\u751f\u5f92<\/span><\/div><div class=\"c-balloon__body -speaking -border-none\"><div class=\"c-balloon__text\">\n<p><strong>\u308f\u304b\u308a\u307e\u3057\u305f\uff01\u65e9\u901f\u30d7\u30ed\u30b8\u30a7\u30af\u30c8\u306b\u5c0e\u5165\u3057\u3066\u307f\u307e\u3059\uff01<\/strong><\/p>\n<span class=\"c-balloon__shapes\"><span class=\"c-balloon__before\"><\/span><span class=\"c-balloon__after\"><\/span><\/span><\/div><\/div><\/div><\/div>\n\n\n<hr class=\"wp-block-separator has-alpha-channel-opacity\"\/>\n\n\n\n<h2 class=\"wp-block-heading\">\u307e\u3068\u3081<\/h2>\n\n\n\n<p class=\"wp-block-paragraph\">GitHub Actions\u3092\u4f7f\u3063\u305f\u81ea\u52d5\u30c7\u30d7\u30ed\u30a4\u306f\u3001\u6700\u521d\u306e\u8a2d\u5b9a\u306b\u5c11\u3057\u6642\u9593\u304c\u304b\u304b\u308a\u307e\u3059\u304c\u3001\u4e00\u5ea6\u69cb\u7bc9\u3059\u308c\u3070\u9577\u671f\u7684\u306b\u5927\u304d\u306a\u4fa1\u5024\u3092\u3082\u305f\u3089\u3057\u307e\u3059\u3002\u307e\u305a\u306f\u30b7\u30f3\u30d7\u30eb\u306aFTP\u30c7\u30d7\u30ed\u30a4\u304b\u3089\u8a66\u3057\u3066\u307f\u3066\u304f\u3060\u3055\u3044\u3002<\/p>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u30ef\u30fc\u30af\u30d5\u30ed\u30fc\u30d5\u30a1\u30a4\u30eb\u306f<code>.github\/workflows\/<\/code>\u306b\u914d\u7f6e\u3059\u308bYAML\u30d5\u30a1\u30a4\u30eb\u3002\u30ef\u30fc\u30af\u30d5\u30ed\u30fc\u30fb\u30a4\u30d9\u30f3\u30c8\u30fb\u30b8\u30e7\u30d6\u30fb\u30b9\u30c6\u30c3\u30d7\u30fb\u30a2\u30af\u30b7\u30e7\u30f3\u30fb\u30e9\u30f3\u30ca\u30fc\u306e6\u8981\u7d20\u3067\u69cb\u6210\u3055\u308c\u308b<\/li>\n<li>\u6a5f\u5bc6\u60c5\u5831\uff08FTP\u30d1\u30b9\u30ef\u30fc\u30c9\u30fbAPI\u30ad\u30fc\uff09\u306f\u5fc5\u305aGitHub Secrets\u306b\u767b\u9332\u3057\u3001YAML\u3067\u306f<code>${{ secrets.NAME }}<\/code>\u3067\u53c2\u7167\u3059\u308b<\/li>\n<li>FTP\u30c7\u30d7\u30ed\u30a4\u306f<code>SamKirkland\/FTP-Deploy-Action<\/code>\u3067\u5b9f\u73fe\u3002<code>dry-run: true<\/code>\u3067\u30c6\u30b9\u30c8\u5b9f\u884c\u3067\u304d\u308b<\/li>\n<li>AWS S3\u30fbNetlify\u30fbVercel\u3078\u306e\u30c7\u30d7\u30ed\u30a4\u3082\u5c02\u7528\u30a2\u30af\u30b7\u30e7\u30f3\u3067\u540c\u69d8\u306e\u30b7\u30f3\u30d7\u30eb\u306a\u8a2d\u5b9a\u3067\u5b9f\u73fe\u53ef\u80fd<\/li>\n<li>\u672c\u683c\u7684\u306aCI\/CD\u306f\u54c1\u8cea\u30c1\u30a7\u30c3\u30af\u2192\u30c6\u30b9\u30c8\u2192\u30bb\u30ad\u30e5\u30ea\u30c6\u30a3\u2192\u30d3\u30eb\u30c9\u2192\u30c7\u30d7\u30ed\u30a4\u306e5\u6bb5\u968e\u69cb\u6210\u3002test\u3068security\u306f\u4e26\u5217\u5b9f\u884c\u3067\u6642\u9593\u77ed\u7e2e<\/li>\n<li>\u30c8\u30e9\u30d6\u30eb\u6642\u306fGitHub\u306e\u300cActions\u300d\u30bf\u30d6\u306e\u30ed\u30b0\u3092\u78ba\u8a8d\u3059\u308b\u306e\u304c\u6700\u77ed\u306e\u89e3\u6c7a\u7b56<\/li>\n<\/ul>\n\n\n\n<hr class=\"wp-block-separator has-alpha-channel-opacity\"\/>\n\n\n\n<h2 class=\"wp-block-heading\">WithCode\u3092\u4f53\u9a13\u3067\u304d\u308b\u521d\u7d1a\u30b3\u30fc\u30b9\u516c\u958b\u4e2d\uff01<\/h2>\n\n\n<div class=\"wp-block-image\">\n<figure class=\"alignleft size-full\"><img decoding=\"async\" src=\"https:\/\/withcode.tech\/media\/wp-content\/uploads\/2024\/01\/withcode-beginner-course.jpg\" alt=\"WithCode\u521d\u7d1a\u30b3\u30fc\u30b9\" class=\"wp-image-12345\"\/><\/figure>\n<\/div>\n\n\n<h3 class=\"wp-block-heading\">\u521d\u7d1a\u30b3\u30fc\u30b9\uff08\u00a549,800\uff09\u304c\u5b8c\u5168\u7121\u6599\u306b\uff01<\/h3>\n\n\n\n<ul class=\"wp-block-list\">\n<li><strong>\u671f\u9593\uff1a<\/strong>1\u9031\u9593<\/li>\n<li><strong>\u5b66\u7fd2\u5185\u5bb9\uff1a<\/strong>\u30ed\u30fc\u30c9\u30de\u30c3\u30d7\/\u57fa\u790e\u77e5\u8b58\/\u74b0\u5883\u69cb\u7bc9\/HTML\/CSS\/LP\u30fb\u30dd\u30fc\u30c8\u30d5\u30a9\u30ea\u30aa\u4f5c\u6210<br><strong><span class=\"swl-marker mark_yellow\">\u2192 \u6b63\u3057\u3044\u5b66\u7fd2\u65b9\u6cd5\u3067\u300c\u78ba\u304b\u306a\u6210\u9577\u300d\u3092\u5b9f\u611f\u3067\u304d\u308b\u30ab\u30ea\u30ad\u30e5\u30e9\u30e0<\/span><\/strong><\/li>\n<\/ul>\n\n\n\n<p class=\"wp-block-paragraph\">\u526f\u696d\u30fb\u30d5\u30ea\u30fc\u30e9\u30f3\u30b9\u304c\u4e3b\u6d41\u306b\u306a\u3063\u3066\u3044\u308b\u4eca\u3053\u305d\u3001\u81ea\u3089\u306e\u30b9\u30ad\u30eb\u3067\u7a3c\u3052\u308b\u4eba\u6750\u3092\u76ee\u6307\u3057\u3066\u307f\u307e\u305b\u3093\u304b\uff1f<\/p>\n\n\n\n<p class=\"wp-block-paragraph\">\u672a\u7d4c\u9a13\u3067\u3082\u5fc3\u914d\u3059\u308b\u3053\u3068\u306f\u3042\u308a\u307e\u305b\u3093\u3002\u521d\u7d1a\u30b3\u30fc\u30b9\u3092\u53d7\u8b1b\u3055\u308c\u308b\u65b9\u306e\u5927\u591a\u6570\u306f\u30d7\u30ed\u30b0\u30e9\u30df\u30f3\u30b0\u672a\u7d4c\u9a13\u3067\u3059\u3002\u307e\u305a\u306f\u7121\u6599\u30ab\u30a6\u30f3\u30bb\u30ea\u30f3\u30b0\u3067\u3001\u60a9\u307f\u3084\u4e0d\u5b89\u3092\u304a\u805e\u304b\u305b\u304f\u3060\u3055\u3044\uff01<\/p>\n\n\n\n<div class=\"wp-block-buttons is-layout-flex wp-block-buttons-is-layout-flex\">\n<div class=\"wp-block-button\"><a class=\"wp-block-button__link has-white-color has-text-color has-background has-link-color wp-element-button\" href=\"https:\/\/withcode.tech\/reservation\/\" style=\"background-color:#ffbf00\"><strong>\u516c\u5f0f\u30b5\u30a4\u30c8\u304b\u3089\u7121\u6599\u30ab\u30a6\u30f3\u30bb\u30ea\u30f3\u30b0\u306b\u7533\u3057\u8fbc\u3080 \u2192<\/strong><\/a><\/div>\n<\/div>\n\n","protected":false},"excerpt":{"rendered":"<p>GitHub Actions\u3067FTP\u30fbAWS S3\u30fbNetlify\u30fbVercel\u3078\u306eCI\/CD\u81ea\u52d5\u30c7\u30d7\u30ed\u30a4\u3092\u69cb\u7bc9\u3059\u308b\u65b9\u6cd5\u3092\u89e3\u8aac\u3002Secrets\u8a2d\u5b9a\u30fb\u57fa\u672c\u30ef\u30fc\u30af\u30d5\u30ed\u30fc\u304b\u30895\u6bb5\u968e\u306e\u672c\u683c\u30d1\u30a4\u30d7\u30e9\u30a4\u30f3\u5b9f\u88c5\u30fb\u30c8\u30e9\u30d6\u30eb\u30b7\u30e5\u30fc\u30c6\u30a3\u30f3\u30b0\u307e\u3067\u52d5\u304f\u30b3\u30fc\u30c9\u4ed8\u304d\u3067\u5b8c\u5168\u89e3\u8aac\u3002<\/p>\n","protected":false},"author":2,"featured_media":0,"comment_status":"closed","ping_status":"closed","sticky":false,"template":"","format":"standard","meta":{"om_disable_all_campaigns":false,"_monsterinsights_skip_tracking":false,"_monsterinsights_sitenote_active":false,"_monsterinsights_sitenote_note":"","_monsterinsights_sitenote_category":0,"swell_btn_cv_data":"","footnotes":"","vk-ltc-link":"","vk-ltc-target":""},"categories":[9,382],"tags":[],"class_list":["post-12620","post","type-post","status-publish","format-standard","hentry","category-learning-content","category-382"],"aioseo_notices":[],"_links":{"self":[{"href":"https:\/\/withcode.tech\/media\/wp-json\/wp\/v2\/posts\/12620","targetHints":{"allow":["GET"]}}],"collection":[{"href":"https:\/\/withcode.tech\/media\/wp-json\/wp\/v2\/posts"}],"about":[{"href":"https:\/\/withcode.tech\/media\/wp-json\/wp\/v2\/types\/post"}],"author":[{"embeddable":true,"href":"https:\/\/withcode.tech\/media\/wp-json\/wp\/v2\/users\/2"}],"replies":[{"embeddable":true,"href":"https:\/\/withcode.tech\/media\/wp-json\/wp\/v2\/comments?post=12620"}],"version-history":[{"count":3,"href":"https:\/\/withcode.tech\/media\/wp-json\/wp\/v2\/posts\/12620\/revisions"}],"predecessor-version":[{"id":14042,"href":"https:\/\/withcode.tech\/media\/wp-json\/wp\/v2\/posts\/12620\/revisions\/14042"}],"wp:attachment":[{"href":"https:\/\/withcode.tech\/media\/wp-json\/wp\/v2\/media?parent=12620"}],"wp:term":[{"taxonomy":"category","embeddable":true,"href":"https:\/\/withcode.tech\/media\/wp-json\/wp\/v2\/categories?post=12620"},{"taxonomy":"post_tag","embeddable":true,"href":"https:\/\/withcode.tech\/media\/wp-json\/wp\/v2\/tags?post=12620"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}