Sunday, December 15, 2019

Workflow for moving an existing VS Code project to Github

Summary

This post is a step by step instruction on how to push an existing Visual Studio Code project to Github.  I'll demonstrate steps from both the GUI and command line to accomplish this.

Step 1:  Create the Github Repository

Screen-shot below of a basic repo.  Note that I use Github's stock Node.js .gitignore file.


Resulting repo:


Step 2:  Create a Local Git Repository

Command-line and GUI methods below:
$ git init .

 

Step 3:  Add Remote (Github) Repository

git remote add origin https://github.com/joeywhelan/containertest

Step 4:  Pull from Github

This step will pull down the existing files in the start-up repo - most importantly, the .gitignore file.

Command-line + GUI methods:

$ git pull origin main

Result below.  Note all the untracked files from the node_modules directory are now hidden via the .gitignore from Github.


Step 5:  Add all of the local files to the local Git Staging Area

$ git add .

Results:

Step 6:  Commit all to the local Git repo.

$ git commit -m "first commit" .

Results:

Step 7:  Rename the local 'master' branch to main and push local files to Github. 

$ git branch -m master main
$ git push -u origin main

Results:

Copyright ©1993-2024 Joey E Whelan, All rights reserved.

Monday, December 9, 2019

Google Cloud Run Quickstart

Summary

This post is a continuation of the previous quick start on containerizing a Node.js app.  In this post, I'll use the same simple app + Dockerfile and deploy the app to Google's Cloud Run platform.

Cloud Run is an alternate method for server-less app deployments.  Google Cloud Functions is other method.  Whereas Cloud Functions has very specific language requirements, Cloud Run has none.  Whatever you build in a container is fair game.  The only requirement is the app needs to respond to HTTP requests as a trigger.  Cloud Run provides similar auto-scaling capabilities as Cloud Functions making it an excellent choice for a self-managing app deployment.

Step 1:  Create a Google Cloud Project

 

 

Step 2:  Enable Cloud Build and Cloud Run APIs

 

Step 3:  Build the Container

$ gcloud config set project cloudrun-quickstart-261521 
Updated property [core/project].
$ gcloud builds submit --tag gcr.io/cloudrun-quickstart-261521/cloudrun-quickstartCreating temporary tarball archive of 331 file(s) totalling 1.6 MiB before compression.
Uploading tarball of [.] to [gs://cloudrun-quickstart-261521_cloudbuild/source/1575934972.69-7637137b7675474ab861a2f4185529c6.tgz]
Created [https://cloudbuild.googleapis.com/v1/projects/cloudrun-quickstart-261521/builds/7591715a-3e15-4323-b570-0c2dc191fb3a].
Logs are available at [https://console.cloud.google.com/gcr/builds/7591715a-3e15-4323-b570-0c2dc191fb3a?project=945345104488].
...
DONE
--------------------------------------------------------------------------------------------------------------------------------------------

ID                                    CREATE_TIME                DURATION  SOURCE                                                                                                IMAGES                                                           STATUS
7591715a-3e15-4323-b570-0c2dc191fb3a  2019-12-09T23:42:55+00:00  57S       gs://cloudrun-quickstart-261521_cloudbuild/source/1575934972.69-7637137b7675474ab861a2f4185529c6.tgz  gcr.io/cloudrun-quickstart-261521/cloudrun-quickstart (+1 more)  SUCCESS

Step 4:  Deploy to Cloud Run

Screen shots below of the Cloud Run console.  Note in the second screen shot the concurrency controls available.  You can specify the number of requests per container and the max container auto-scale growth.




Step 5:  Execute

$ curl -i https://cloudrun-quickstart-6saiqefrtq-uc.a.run.app 
HTTP/2 200 
x-powered-by: Express
content-type: text/html; charset=utf-8
etag: W/"b-Kq5sNclPz7QV2+lfQIuc6R7oRu0"
x-cloud-trace-context: 4f70f43d41086a3d8ea2320a166a5f89;o=1
date: Mon, 09 Dec 2019 23:58:10 GMT
server: Google Frontend
content-length: 11
alt-svc: quic=":443"; ma=2592000; v="46,43",h3-Q050=":443"; ma=2592000,h3-Q049=":443"; ma=2592000,h3-Q048=":443"; ma=2592000,h3-Q046=":443"; ma=2592000,h3-Q043=":443"; ma=2592000

hello world

Step 6:  Clean up

Copyright ©1993-2024 Joey E Whelan, All rights reserved.

Node.js on Docker Quickstart



Summary

This post is a basic primer on getting a Node app containerized.

Step 1:  Create the app

Below is a basic HTTP server implementation using Express.
'use strict';

const express = require('express');
const app = express();
const port = process.env.PORT || 8080;

app.get('/', function(req, res) {
 res.send('hello world')
});

app.listen(port, () => {
    console.log(`listening on ${port}`);
});
Below is the resulting package.json containing the app dependencies. This file is created via the 'npm init' command. 'npm install --save' causes the dependencies to be updated for each module used in the app.
{
  "name": "containertest",
  "version": "1.0.0",
  "description": "simple express server",
  "main": "server.js",
  "repository": "none",
  "scripts": {
    "test": "echo \"Error: no test specified\" && exit 1",
    "start": "node server.js"
  },
  "author": "joey whelan",
  "license": "MIT",
  "dependencies": {
    "express": "^4.17.1"
  }
}

Step 2:  Create the Dockerfile and .dockerignore

Below is the Dockerfile to support the above app.  It's written per best practices to cause the app source code to be added as the last layer to enable caching of modules.
FROM node:lts
WORKDIR /usr/src/app
COPY package*.json ./
RUN npm install
COPY . .
EXPOSE 8080
CMD ["npm", "start"]
.dockerignore below.
node_modules
npm-debug.log

Step 3:  Build the Docker image

$ docker build -t containertest .
Sending build context to Docker daemon  19.97kB
Step 1/7 : FROM node:lts
lts: Pulling from library/node
844c33c7e6ea: Pull complete 
ada5d61ae65d: Pull complete 
f8427fdf4292: Pull complete 
f025bafc4ab8: Pull complete 
7a9577c07934: Pull complete 
9b4289f800f5: Pull complete 
55e3fcab47b9: Pull complete 
c7a94e331913: Pull complete 
bb9efc0c132a: Pull complete 
Digest: sha256:88ee7d2a5e18d359b4b5750ecb50a9b238ab467397c306aeb9955f4f11be44ce
Status: Downloaded newer image for node:lts
 ---> 7be6a8478f5f
Step 2/7 : WORKDIR /usr/src/app
 ---> df2833d84c36
Removing intermediate container 11f00574e18a
Step 3/7 : COPY package*.json ./
 ---> a892506a76df
Removing intermediate container 07fc76863a41
Step 4/7 : RUN npm install
 ---> Running in 7abdbc6d6e64
added 50 packages from 37 contributors and audited 126 packages in 1.197s
found 0 vulnerabilities

 ---> 9ab0afa5e750
Removing intermediate container 7abdbc6d6e64
Step 5/7 : COPY . .
 ---> 39620b323b38
Removing intermediate container f2692a5064a0
Step 6/7 : EXPOSE 8080
 ---> Running in 4e758301eaad
 ---> 076d56510119
Removing intermediate container 4e758301eaad
Step 7/7 : CMD npm start
 ---> Running in 80f1a6be42cc
 ---> c9c55b4ddca7
Removing intermediate container 80f1a6be42cc
Successfully built c9c55b4ddca7
Successfully tagged containertest:latest
Resulting images below:
$ docker images
REPOSITORY          TAG                 IMAGE ID            CREATED             SIZE
containertest       latest              c9c55b4ddca7        3 minutes ago       911MB
node                lts                 7be6a8478f5f        2 weeks ago         908MB

Step 4:  Run the container

$ docker run -p 8080:8080 -d containertest
d8d98f3fff8e752c186f99599ca475682790e3a5645d16705a398404ddf9ec74
Resulting running container below:
$ docker ps
CONTAINER ID        IMAGE               COMMAND                  CREATED             STATUS              PORTS                    NAMES
d8d98f3fff8e        containertest       "docker-entrypoint..."   2 minutes ago       Up 2 minutes        0.0.0.0:8080->8080/tcp   thirsty_dijkstra
Execution of HTTP request against the containerized server below:
$ curl -v localhost:8080/
*   Trying 127.0.0.1...
* TCP_NODELAY set
* Connected to localhost (127.0.0.1) port 8080 (#0)
> GET / HTTP/1.1
> Host: localhost:8080
> User-Agent: curl/7.58.0
> Accept: */*
> 
< HTTP/1.1 200 OK
< X-Powered-By: Express
< Content-Type: text/html; charset=utf-8
< Content-Length: 11
< ETag: W/"b-Kq5sNclPz7QV2+lfQIuc6R7oRu0"
< Date: Mon, 09 Dec 2019 16:44:36 GMT
< Connection: keep-alive
< 
* Connection #0 to host localhost left intact
hello world

Step 5:  Stop the container

Stop command results below.  Note it's possible to abbreviate the container ID.
$ docker stop d8
d8
$ docker ps
CONTAINER ID        IMAGE               COMMAND             CREATED             STATUS              PORTS               NAMES

Step 6:  Clean up

Commands below to delete the container and its image.
$ docker ps -a
CONTAINER ID        IMAGE               COMMAND                  CREATED             STATUS                     PORTS               NAMES
d8d98f3fff8e        containertest       "docker-entrypoint..."   17 minutes ago      Exited (0) 4 minutes ago                       thirsty_dijkstra
$ docker rm d8
d8
$ docker ps -a
CONTAINER ID        IMAGE               COMMAND             CREATED             STATUS              PORTS               NAMES
$ docker images
REPOSITORY          TAG                 IMAGE ID            CREATED             SIZE
containertest       latest              c9c55b4ddca7        26 minutes ago      911MB
node                lts                 7be6a8478f5f        2 weeks ago         908MB
$ docker rmi c9
Untagged: containertest:latest
Deleted: sha256:c9c55b4ddca7aac2d12a93c66ee1adf95b5e2b4b09a7b93aac2e0981b45006be
Deleted: sha256:076d565101195290512b90eba0d25f16a3798db0f1cf49aea4d096e926d03250
Deleted: sha256:39620b323b38b824b8f074183eda49339748f2516a70d19718d821a086826234
Deleted: sha256:20407ee7b27893df082e6fa7eddb9608517d53a930beaf426c37ac2453949714
Deleted: sha256:9ab0afa5e750b610d08ed12258972e8d880d8acdd8b3034bd96add8c5daea705
Deleted: sha256:b972df701627963f9fa4dbb2ef1c20148cdddd8a8922aea6c3ba8e2ceca62c27
Deleted: sha256:a892506a76df6ceaaff88b3fe14ee30de477fc9596cb8236aeeee0b3a0106e76
Deleted: sha256:4fab789311ef158be2b924dcdaa1646802900913e07d645f95adb299ee09c506
Deleted: sha256:df2833d84c365c86e3c5218cc997d3ec958e1e4f68eb47cb82483cbd2a14c738
Deleted: sha256:43dd5d9ada9dc352d7fdf5cd3b179cd0855681851eef378a5c82b3ce682bc17e
$ docker images
REPOSITORY          TAG                 IMAGE ID            CREATED             SIZE
node                lts                 7be6a8478f5f        2 weeks ago         908MB

Copyright ©1993-2024 Joey E Whelan, All rights reserved.