<?xml version="1.0" encoding="UTF-8"?><rss version="2.0"
	xmlns:content="http://purl.org/rss/1.0/modules/content/"
	xmlns:wfw="http://wellformedweb.org/CommentAPI/"
	xmlns:dc="http://purl.org/dc/elements/1.1/"
	xmlns:atom="http://www.w3.org/2005/Atom"
	xmlns:sy="http://purl.org/rss/1.0/modules/syndication/"
	xmlns:slash="http://purl.org/rss/1.0/modules/slash/"
	>

<channel>
	<title>AWS &#8211; Few Steps &#8211; ก้าวสั้นๆ แต่ไปเรื่อยๆ</title>
	<atom:link href="https://myifew.com/tag/aws/feed/" rel="self" type="application/rss+xml" />
	<link>https://myifew.com</link>
	<description></description>
	<lastBuildDate>Thu, 09 Sep 2021 03:45:40 +0000</lastBuildDate>
	<language>en-US</language>
	<sy:updatePeriod>
	hourly	</sy:updatePeriod>
	<sy:updateFrequency>
	1	</sy:updateFrequency>
	<generator>https://wordpress.org/?v=6.9.4</generator>

<image>
	<url>https://myifew.com/wp-content/uploads/2018/07/cropped-logo6-ts-32x32.png</url>
	<title>AWS &#8211; Few Steps &#8211; ก้าวสั้นๆ แต่ไปเรื่อยๆ</title>
	<link>https://myifew.com</link>
	<width>32</width>
	<height>32</height>
</image> 
	<item>
		<title>ลองทำ Kubernetes บน Amazon EKS แบบง่ายๆ ด้วย eksctl</title>
		<link>https://myifew.com/5972/kubernetes-on-amazon-eks/</link>
					<comments>https://myifew.com/5972/kubernetes-on-amazon-eks/#respond</comments>
		
		<dc:creator><![CDATA[iFew]]></dc:creator>
		<pubDate>Wed, 08 Sep 2021 17:41:08 +0000</pubDate>
				<category><![CDATA[Technology]]></category>
		<category><![CDATA[AWS]]></category>
		<category><![CDATA[AWS EKS]]></category>
		<category><![CDATA[Kubernetes]]></category>
		<guid isPermaLink="false">https://myifew.com/?p=5972</guid>

					<description><![CDATA[ไปลองเล่น Kubernetes บน Amazon EKS มา ขอบันทึกไว้หน่อย ใช้งานง่ายดีครับ ด้วยเครื่องมือของทาง Amazone ที่ชื่อว่า eksctl]]></description>
										<content:encoded><![CDATA[
<p>ไปลองเล่น Kubernetes บน Amazon EKS มา ขอบันทึกไว้หน่อย ใช้งานง่ายดีครับ ด้วยเครื่องมือของทาง Amazon ที่ชื่อว่า eksctl (ใช้งานร่วมกับ kubectl)</p>



<span id="more-5972"></span>



<h2 class="wp-block-heading">วิธีจัดการ Kubernetes บน Amazon EKS</h2>



<p>ตัว <meta charset="utf-8"><meta charset="utf-8">Amazon EKS (Amazon Elastic Kubernetes Service) มีวิธีการจัดการสองแบบ คือ</p>



<ul class="wp-block-list"><li>eksctl &#8211; ที่เป็น CLI tool ของ Amazon เอง ในการจัดการ Amazon EKS โดยเฉพาะ</li><li>AWS CLI &#8211; ที่เป็น CLI tool ของ Amazon ที่ไว้จัดการ Service ต่างๆ ของ Amazon รวมถึง Amazon EKS ด้วย</li></ul>



<p>จากที่ลองทดสอบเล่นทั้งสองแบบ eksctl ใช้งานง่ายกว่ามาก เรียกว่าเป็น Automated Tool สำหรับ shortcut คำสั่งในการจัดการ EKS เลย แต่ก่อนจะไปต่อ เราควรต้องมีความรู้ในสามเรื่องนี้ก่อน เพื่อจะได้ไม่ไป งง กลางทาง</p>



<ol class="wp-block-list"><li>AWS Service </li><li>Docker</li><li>Kubernetes</li></ol>



<h2 class="wp-block-heading">เริ่มต้นใช้งาน Amazon EKS</h2>



<p>ไม่ว่าจะเลือกวิธีการใดก็ตาม สิ่งที่ต้องทำ มีดังนี้</p>



<ul class="wp-block-list"><li><a rel="noreferrer noopener" href="https://docs.aws.amazon.com/cli/latest/userguide/cli-chap-install.html" data-type="URL" data-id="https://docs.aws.amazon.com/cli/latest/userguide/cli-chap-install.html" target="_blank">ติดตั้ง AWS CLI</a> , <a rel="noreferrer noopener" href="https://docs.aws.amazon.com/cli/latest/userguide/cli-configure-quickstart.html#cli-configure-quickstart-config" data-type="URL" data-id="https://docs.aws.amazon.com/cli/latest/userguide/cli-configure-quickstart.html#cli-configure-quickstart-config" target="_blank">สร้าง Access Key ID และ Configure AWS CLI</a> &#8211; เพื่อให้เครื่องเราสามารถเชื่อมต่อและควบคุม Account AWS ของเราได้</li><li><a rel="noreferrer noopener" href="https://docs.aws.amazon.com/eks/latest/userguide/install-kubectl.html" data-type="URL" data-id="https://docs.aws.amazon.com/eks/latest/userguide/install-kubectl.html" target="_blank">ติดตั้ง kubectl</a> &#8211; เป็นตัว Core เพื่อใช้ควบคุม Kubernetes Clusters </li><li><a rel="noreferrer noopener" href="https://docs.aws.amazon.com/eks/latest/userguide/eksctl.html" data-type="URL" data-id="https://docs.aws.amazon.com/eks/latest/userguide/eksctl.html" target="_blank">ติดตั้ง eksctl</a> &#8211; เพื่อไปใช้ร่วมกับ kubectl อีกที ในการควบคุม Kubernetes Clusters บน Amazon EKS</li><li><a rel="noreferrer noopener" href="https://docs.aws.amazon.com/service-authorization/latest/reference/list_amazonelastickubernetesservice.html" data-type="URL" data-id="https://docs.aws.amazon.com/service-authorization/latest/reference/list_amazonelastickubernetesservice.html" target="_blank">AWS IAM Permissions</a> &#8211; โดย Account AWS เราจะต้องได้รับสิทธิ์ที่เกี่ยวข้องทั้งหมดสำหรับการควบคุม Amazon EKS ทั้งหมด รวมไปถึง ECR, Network, EC2 ด้วยครับ</li></ul>



<h2 class="wp-block-heading">โครงสร้าง Kubernetes และสิ่งที่จะต้องทำบน EKS </h2>



<div class="wp-block-image"><figure class="aligncenter size-large"><img fetchpriority="high" decoding="async" width="1200" height="891" src="https://myifew.com/wp-content/uploads/2021/09/k8s-cluster-diagram-1200x891.png" alt="" class="wp-image-5975" srcset="https://myifew.com/wp-content/uploads/2021/09/k8s-cluster-diagram-1200x891.png 1200w, https://myifew.com/wp-content/uploads/2021/09/k8s-cluster-diagram-1024x760.png 1024w, https://myifew.com/wp-content/uploads/2021/09/k8s-cluster-diagram-768x570.png 768w, https://myifew.com/wp-content/uploads/2021/09/k8s-cluster-diagram-1536x1140.png 1536w, https://myifew.com/wp-content/uploads/2021/09/k8s-cluster-diagram-700x519.png 700w, https://myifew.com/wp-content/uploads/2021/09/k8s-cluster-diagram.png 1776w" sizes="(max-width: 1200px) 100vw, 1200px" /><figcaption>Kubernetes Cluster Diagram</figcaption></figure></div>



<div class="wp-block-image"><figure class="aligncenter size-large"><img decoding="async" width="1200" height="973" src="https://myifew.com/wp-content/uploads/2021/09/k8s-node-overview-1200x973.png" alt="" class="wp-image-5977" srcset="https://myifew.com/wp-content/uploads/2021/09/k8s-node-overview-1200x973.png 1200w, https://myifew.com/wp-content/uploads/2021/09/k8s-node-overview-1024x830.png 1024w, https://myifew.com/wp-content/uploads/2021/09/k8s-node-overview-768x623.png 768w, https://myifew.com/wp-content/uploads/2021/09/k8s-node-overview-1536x1245.png 1536w, https://myifew.com/wp-content/uploads/2021/09/k8s-node-overview-700x568.png 700w, https://myifew.com/wp-content/uploads/2021/09/k8s-node-overview.png 1628w" sizes="(max-width: 1200px) 100vw, 1200px" /><figcaption>Node Overview</figcaption></figure></div>



<p>ในรายละเอียด สามารถอ่านจากบล็อกต่างๆ ได้เลยครับ ผมขอสรุปไวๆ ดังนี้</p>



<ol class="wp-block-list"><li><strong>Kubernetes Cluster </strong>&#8211; จำเป็นต้องสร้างก่อนเลย เพื่อใช้เป็น Cluster ของชุด Server เช่น ควบคุม (Control) กระจาย (Distribution) ตั้งกำหนดการ (Scheduling)</li><li><strong>Control Plane หรือ Master Node</strong> &#8211; ตัวนี้ก็จะเป็นต้องมี แต่ EKS จะสร้างให้อัตโนมัติ โดยอิงจาก Config ที่เรากำหนดในไฟล์ yml หรือใน AWS Console เช่น Min/Max Node, Min/Max Pod ใช้เพื่อควบคุมสิ่งต่างๆ ใน Cluster </li><li><strong>Node</strong> &#8211; ตัวนี้จำเป็นต้องสร้าง เสมือนเครื่อง Server ของเรา สามารถปั๊มเพิ่ม ลด ได้ตามจำนวนที่กำหนด</li><li><strong>Pod</strong> &#8211; ตัวนี้จำเป็นต้องสร้าง คือกลุ่มของ Container ของเรา</li></ol>



<h2 class="wp-block-heading">สร้าง Kubernetes Cluster</h2>



<p>ให้รันคำสั่งตามนี้ครับ</p>



<pre class="wp-block-code"><code><strong>eksctl create cluster --name=test-php-web --region ap-southeast-1 --managed</strong> --nodes=2 --node-type=t2.micro --with-oidc --ssh-access --ssh-public-key ifew_orm_macbook --profile few_orm</code></pre>



<p><meta charset="utf-8">คำอธิบาย</p>



<ul class="wp-block-list"><li><meta charset="utf-8">&#8211;name=test-php-web คือ ชื่อ Cluster</li><li>&#8211;region ap-southeast-1 คือ Zone ทวีป AWS ที่เราจะใช้งาน ในที่นี้คือสิงคโปร์</li><li>&#8211;nodes=2 คือ จำนวน node หรือจำนวนเครื่อง ec2 ที่จะถูกเปิดขึ้นมา (Optional)</li><li>&#8211;node-type=t2.micro คือ type class ของเครื่อง ec2 ที่จะถูกเปิดขึ้นมา (Optional)</li><li>&#8211;managed คือ บอกว่าจะสร้างแบบ Managed nodes ไม่ใช่แบบ Fargate</li><li>&#8211;with-oidc คือ เปิดให้ใช้ OIDC ใน cluster นี้ได้ (Optional)</li><li>&#8211;ssh-access คือ เปิดให้ใช้ SSH Access ใน cluster นี้ได้ (Optional)</li><li>&#8211;ssh-public-key ifew_orm_macbook คือ ระบุ Public Key สำหรับ SSH Access (Optional)</li><li>&#8211;profile few_orm คือ บอกว่าใช้ AWS Profile ไหนในการรันคำสั่งนี้ (Optional)</li></ul>



<h2 class="wp-block-heading">สร้าง Namespace</h2>



<p>ในที่นี้ ผมขอเสริมเรื่องการสร้าง namespace นิดนึง ไว้จำกัดขอบเขตการทำงาน และเพื่อความปลอดภัย กันพลาดพลั้งไปลบอะไรออก</p>



<pre class="wp-block-code"><code><strong>kubectl create namespace myweb</strong></code></pre>



<ul class="wp-block-list"><li>myweb คือ ชื่อของ namespace ที่จะใช้</li></ul>



<p><meta charset="utf-8">จะได้หน้าตาแบบนี้</p>



<figure class="wp-block-image size-full"><img decoding="async" width="846" height="64" src="https://myifew.com/wp-content/uploads/2021/09/eksctl-create-namespace.png" alt="" class="wp-image-5980" srcset="https://myifew.com/wp-content/uploads/2021/09/eksctl-create-namespace.png 846w, https://myifew.com/wp-content/uploads/2021/09/eksctl-create-namespace-768x58.png 768w, https://myifew.com/wp-content/uploads/2021/09/eksctl-create-namespace-700x53.png 700w" sizes="(max-width: 846px) 100vw, 846px" /></figure>



<h2 class="wp-block-heading">สร้าง YML File สำหรับ Deployment</h2>



<p>สร้างไฟล์ yml ขึ้นมา เพื่อ config pod และการ deployment ในที่นี้ผมใช้ไฟล์ชื่อ <strong>eks-myweb-deploy.yml</strong></p>



<pre class="wp-block-code"><code>apiVersion: apps/v1
kind: Deployment
metadata:
  name: myweb-deployment
  namespace: myweb
  labels:
    app: myweb
spec:
  replicas: 2
  selector:
    matchLabels:
      app: myweb
  template:
    metadata:
      labels:
        app: myweb
    spec:
      containers:
      - name: phpfpm
        image: xxxxx.dkr.ecr.ap-southeast-1.amazonaws.com/test:v1
        ports:
        - containerPort: 80
        env:
        - name: MYSQL_DB_HOST
          value: mydb.host.com
        - name: MYSQL_DATABASE
          value: test
        - name: MYSQL_USER
          value: ifew 
        - name: MYSQL_PASSWORD
          value: password1234
        - name: MYSQL_ROOT_PASSWORD
          value: password1234
        - name: TEST
          value: test</code></pre>



<p>จากนั้นทำการรันคำสั่ง</p>



<pre class="wp-block-code"><code><strong>kubectl apply -f eks-myweb-deploy.yml</strong></code></pre>



<p>จะได้หน้าตาแบบนี้</p>



<figure class="wp-block-image size-full"><img loading="lazy" decoding="async" width="930" height="62" src="https://myifew.com/wp-content/uploads/2021/09/eksctl-apply-deploy-1.png" alt="" class="wp-image-5983" srcset="https://myifew.com/wp-content/uploads/2021/09/eksctl-apply-deploy-1.png 930w, https://myifew.com/wp-content/uploads/2021/09/eksctl-apply-deploy-1-768x51.png 768w, https://myifew.com/wp-content/uploads/2021/09/eksctl-apply-deploy-1-700x47.png 700w" sizes="auto, (max-width: 930px) 100vw, 930px" /></figure>



<p>ถ้าอยากรู้ว่าสร้างจริงไหม มีอะไรใน namespace เราแล้วบ้าง ให้ใช้คำสั่งนี้ครับ</p>



<pre class="wp-block-code"><code><meta charset="utf-8"><strong>kubectl get all -n myweb</strong></code></pre>



<p>myweb คือ ระบุชื่อ namespace</p>



<p>จากนั้นจะได้ข้อมูลดังนี้</p>



<figure class="wp-block-image size-full"><img loading="lazy" decoding="async" width="1112" height="308" src="https://myifew.com/wp-content/uploads/2021/09/eksctl-get-all-namespace.png" alt="" class="wp-image-5984" srcset="https://myifew.com/wp-content/uploads/2021/09/eksctl-get-all-namespace.png 1112w, https://myifew.com/wp-content/uploads/2021/09/eksctl-get-all-namespace-1024x284.png 1024w, https://myifew.com/wp-content/uploads/2021/09/eksctl-get-all-namespace-768x213.png 768w, https://myifew.com/wp-content/uploads/2021/09/eksctl-get-all-namespace-700x194.png 700w" sizes="auto, (max-width: 1112px) 100vw, 1112px" /></figure>



<p>ใน config ผมสร้างไว้ 2 pod ดังนั้น และถ้าใช้งานได้ ตรง Status จะขึ้นว่า Running</p>



<p>ลองไปดูใน AWS Console โดยเข้าไปที่เมนู EKS &gt; เลือกชื่อ Cluster ที่เราสร้าง &gt; เลือก Node ที่เราสร้าง &gt; Pods จะแสดงรายการสองกลุ่ม คือ กลุ่ม Node Processes กับกลุ่ม Pod ที่เราสร้างขึ้นมา</p>



<figure class="wp-block-image size-full"><img loading="lazy" decoding="async" width="1146" height="527" src="https://myifew.com/wp-content/uploads/2021/09/eksctl-aws-console-pods.png" alt="" class="wp-image-5985" srcset="https://myifew.com/wp-content/uploads/2021/09/eksctl-aws-console-pods.png 1146w, https://myifew.com/wp-content/uploads/2021/09/eksctl-aws-console-pods-1024x471.png 1024w, https://myifew.com/wp-content/uploads/2021/09/eksctl-aws-console-pods-768x353.png 768w, https://myifew.com/wp-content/uploads/2021/09/eksctl-aws-console-pods-700x322.png 700w" sizes="auto, (max-width: 1146px) 100vw, 1146px" /></figure>



<p>ตอนนี้แอพเราขึ้นไปรันรอไว้แล้ว แต่ยังไม่มีช่องทางเข้านะครับ เราต้องสร้าง Service ขึ้นมาก่อน<meta charset="utf-8"></p>



<h2 class="wp-block-heading">สร้าง YML File สำหรับ Service</h2>



<p><meta charset="utf-8">สร้างไฟล์ yml ขึ้นมา เพื่อ กำหนด Service และให้มีช่องทางเข้าไป Pod ที่เราสร้าง โดยผมจะสร้างแบบมี Load Balance ด้วย และตั้งไฟล์ชื่อ <strong>eks-<meta charset="utf-8"><strong>myweb</strong>-service.yml</strong></p>



<pre class="wp-block-code"><code>apiVersion: v1
kind: Service
metadata:
  name: myweb-service
  namespace: myweb
  labels:
    app: myweb
spec:
  selector:
    app: myweb
  ports:
  - name: http
    port: 80
    targetPort: 80
  type: LoadBalancer</code></pre>



<p><meta charset="utf-8">จากนั้นทำการรันคำสั่ง</p>



<pre class="wp-block-code"><code><meta charset="utf-8"><strong>kubectl apply -f eks-<meta charset="utf-8"><strong>myweb</strong>-service.yml</strong></code></pre>



<p><meta charset="utf-8">จะได้หน้าตาแบบนี้</p>



<figure class="wp-block-image size-full"><img loading="lazy" decoding="async" width="934" height="60" src="https://myifew.com/wp-content/uploads/2021/09/eksctl-apply-service-1.png" alt="" class="wp-image-5986" srcset="https://myifew.com/wp-content/uploads/2021/09/eksctl-apply-service-1.png 934w, https://myifew.com/wp-content/uploads/2021/09/eksctl-apply-service-1-768x49.png 768w, https://myifew.com/wp-content/uploads/2021/09/eksctl-apply-service-1-700x45.png 700w" sizes="auto, (max-width: 934px) 100vw, 934px" /></figure>



<p>ลองดูสิ่งที่อยู่ใน namespace อีกครั้ง</p>



<figure class="wp-block-image size-large"><img loading="lazy" decoding="async" width="1200" height="202" src="https://myifew.com/wp-content/uploads/2021/09/eksctl-get-all-namespace-with-service-1200x202.png" alt="" class="wp-image-5987" srcset="https://myifew.com/wp-content/uploads/2021/09/eksctl-get-all-namespace-with-service-1200x202.png 1200w, https://myifew.com/wp-content/uploads/2021/09/eksctl-get-all-namespace-with-service-1024x172.png 1024w, https://myifew.com/wp-content/uploads/2021/09/eksctl-get-all-namespace-with-service-768x129.png 768w, https://myifew.com/wp-content/uploads/2021/09/eksctl-get-all-namespace-with-service-1536x259.png 1536w, https://myifew.com/wp-content/uploads/2021/09/eksctl-get-all-namespace-with-service-2048x345.png 2048w, https://myifew.com/wp-content/uploads/2021/09/eksctl-get-all-namespace-with-service-700x118.png 700w, https://myifew.com/wp-content/uploads/2021/09/eksctl-get-all-namespace-with-service.png 2364w" sizes="auto, (max-width: 1200px) 100vw, 1200px" /></figure>



<p>จะเห็นว่า มี Service เพิ่มเข้ามาแล้ว โดย Type เป็น LoadBalancer ตามที่ได้ config ใน yml ของเรา <br>ส่วน URL ที่ใช้เข้า App เรา อยู่ตรง External-IP นั้นเอง</p>



<h2 class="wp-block-heading">ลองเข้า App ที่สร้างขึ้นมา</h2>



<p>เปิด Brower และเข้า URL จาก <meta charset="utf-8">External-IP ที่แสดงในขั้นตอนก่อนหน้านี้ ก็สามารถใช้งานได้แล้วครับ</p>



<div class="wp-block-image"><figure class="aligncenter size-large"><img loading="lazy" decoding="async" width="1200" height="643" src="https://myifew.com/wp-content/uploads/2021/09/eks-web-demo-home-1-1200x643.png" alt="" class="wp-image-5991" srcset="https://myifew.com/wp-content/uploads/2021/09/eks-web-demo-home-1-1200x643.png 1200w, https://myifew.com/wp-content/uploads/2021/09/eks-web-demo-home-1-1024x548.png 1024w, https://myifew.com/wp-content/uploads/2021/09/eks-web-demo-home-1-768x411.png 768w, https://myifew.com/wp-content/uploads/2021/09/eks-web-demo-home-1-1536x823.png 1536w, https://myifew.com/wp-content/uploads/2021/09/eks-web-demo-home-1-2048x1097.png 2048w, https://myifew.com/wp-content/uploads/2021/09/eks-web-demo-home-1-700x375.png 700w, https://myifew.com/wp-content/uploads/2021/09/eks-web-demo-home-1.png 2868w" sizes="auto, (max-width: 1200px) 100vw, 1200px" /><figcaption>หน้าแรกของ Demo ที่รันบน nginx + php</figcaption></figure></div>



<div class="wp-block-image"><figure class="aligncenter size-large"><img loading="lazy" decoding="async" width="1200" height="646" src="https://myifew.com/wp-content/uploads/2021/09/eks-web-demo-phpinfo-1-1200x646.png" alt="" class="wp-image-5993" srcset="https://myifew.com/wp-content/uploads/2021/09/eks-web-demo-phpinfo-1-1200x646.png 1200w, https://myifew.com/wp-content/uploads/2021/09/eks-web-demo-phpinfo-1-1024x552.png 1024w, https://myifew.com/wp-content/uploads/2021/09/eks-web-demo-phpinfo-1-768x414.png 768w, https://myifew.com/wp-content/uploads/2021/09/eks-web-demo-phpinfo-1-1536x827.png 1536w, https://myifew.com/wp-content/uploads/2021/09/eks-web-demo-phpinfo-1-2048x1103.png 2048w, https://myifew.com/wp-content/uploads/2021/09/eks-web-demo-phpinfo-1-700x377.png 700w, https://myifew.com/wp-content/uploads/2021/09/eks-web-demo-phpinfo-1.png 2866w" sizes="auto, (max-width: 1200px) 100vw, 1200px" /><figcaption>ลองดูว่าเรียกใช้ PHP และ Extension ได้ครบถ้วนไหม</figcaption></figure></div>



<div class="wp-block-image"><figure class="aligncenter size-large"><img loading="lazy" decoding="async" width="1200" height="320" src="https://myifew.com/wp-content/uploads/2021/09/eks-web-demo-env-1-1200x320.png" alt="" class="wp-image-5992" srcset="https://myifew.com/wp-content/uploads/2021/09/eks-web-demo-env-1-1200x320.png 1200w, https://myifew.com/wp-content/uploads/2021/09/eks-web-demo-env-1-1024x273.png 1024w, https://myifew.com/wp-content/uploads/2021/09/eks-web-demo-env-1-768x205.png 768w, https://myifew.com/wp-content/uploads/2021/09/eks-web-demo-env-1-1536x410.png 1536w, https://myifew.com/wp-content/uploads/2021/09/eks-web-demo-env-1-700x187.png 700w, https://myifew.com/wp-content/uploads/2021/09/eks-web-demo-env-1.png 1964w" sizes="auto, (max-width: 1200px) 100vw, 1200px" /><figcaption>ลองดูว่าส่ง Environment Variable มาครบไหม</figcaption></figure></div>



<h2 class="wp-block-heading">Tip: ใช้งาน HTTPS ผ่าน ACM Certificate</h2>



<p>โดยให้สร้าง ACM certificate ขึ้นมา และใน yml ไฟล์ของ service ให้เพิ่มสองส่วนตามภาพด้านล่าง </p>



<figure class="wp-block-image size-large"><img loading="lazy" decoding="async" width="1200" height="571" src="https://myifew.com/wp-content/uploads/2021/09/eks-ssl-1200x571.png" alt="" class="wp-image-5994" srcset="https://myifew.com/wp-content/uploads/2021/09/eks-ssl-1200x571.png 1200w, https://myifew.com/wp-content/uploads/2021/09/eks-ssl-1024x487.png 1024w, https://myifew.com/wp-content/uploads/2021/09/eks-ssl-768x365.png 768w, https://myifew.com/wp-content/uploads/2021/09/eks-ssl-1536x730.png 1536w, https://myifew.com/wp-content/uploads/2021/09/eks-ssl-700x333.png 700w, https://myifew.com/wp-content/uploads/2021/09/eks-ssl.png 1914w" sizes="auto, (max-width: 1200px) 100vw, 1200px" /></figure>



<p>โดยรายละเอียดวิธีติดตั้ง ให้ดูได้ที่ <a rel="noreferrer noopener" href="https://aws.amazon.com/premiumsupport/knowledge-center/terminate-https-traffic-eks-acm/" target="_blank">https://aws.amazon.com/premiumsupport/knowledge-center/terminate-https-traffic-eks-acm/</a></p>



<h2 class="wp-block-heading">สรุป</h2>



<p>สำหรับ Kubernetes ผมเองเรียกว่าเบสิกมากๆ เป็นมือใหม่เลย ได้ลองเรียนรู้ใช้งานมาสักพัก เลยอยากสรุปไว้ให้ผู้อ่าน ที่กำลังริเริ่มอยากทดลองใช้ Kubernetes มาได้ใช้ด้วยกันครับ ผ่านบริการของ Amazon EKS ก็หวังว่าจะช่วยให้การเริ่มต้นได้ง่ายขึ้น และเอาไปต่อยอดเพื่อใช้งานระดับ Advance ต่อไป</p>



<p>หากใครอยากได้สรุปคำสั่งต่างๆ รวมถึงโค้ดตัวอย่าง ทั้งหมดในโพสต์นี้ <br>สามารถไปดูได้ที่ <a rel="noreferrer noopener" href="https://github.com/ifew/eks-test-php" target="_blank">https://github.com/ifew/eks-test-php</a></p>



<h2 class="wp-block-heading">Reference</h2>



<ul class="wp-block-list"><li><a href="https://docs.aws.amazon.com/eks/latest/userguide/getting-started-eksctl.html">https://docs.aws.amazon.com/eks/latest/userguide/getting-started-eksctl.html</a></li></ul>
]]></content:encoded>
					
					<wfw:commentRss>https://myifew.com/5972/kubernetes-on-amazon-eks/feed/</wfw:commentRss>
			<slash:comments>0</slash:comments>
		
		
			</item>
		<item>
		<title>ทดสอบความเร็ว .NET Core 3.1 (Official Runtimes) บน AWS Lambda</title>
		<link>https://myifew.com/5712/benchmark-netcore-31-aws-lambda/</link>
					<comments>https://myifew.com/5712/benchmark-netcore-31-aws-lambda/#respond</comments>
		
		<dc:creator><![CDATA[iFew]]></dc:creator>
		<pubDate>Sun, 10 May 2020 18:55:07 +0000</pubDate>
				<category><![CDATA[Technology]]></category>
		<category><![CDATA[.Net Core]]></category>
		<category><![CDATA[AWS]]></category>
		<category><![CDATA[AWS Lambda]]></category>
		<category><![CDATA[Performance Test]]></category>
		<category><![CDATA[Serverless]]></category>
		<guid isPermaLink="false">https://myifew.com/?p=5712</guid>

					<description><![CDATA[วันที่ผมรอคอยก็มาถึง เมื่อ AWS ประกาศรองรับ .NET Core 3.1 เป็น Official Runtime ทำให้ Cold Start ลดลงกว่า 50%]]></description>
										<content:encoded><![CDATA[
<p>และแล้ว วันที่ผมรอคอย (รวมถึงชาว .NET Core บน. AWS Lambda) ก็มาถึง เมื่อ <a rel="noreferrer noopener" href="https://aws.amazon.com/blogs/compute/announcing-aws-lambda-supports-for-net-core-3-1/" target="_blank">AWS ประกาศรองรับ .NET Core 3.1 เป็น Official Runtime</a> แล้วจ้า (จริงๆ ประกาศตั้งแต่ 31 มีนาคม 2563 แต่ผมเห็นช้าไปหน่อย แหะๆ) ต้องขอทดสอบสักหน่อยว่าดีขึ้นแค่ไหน แต่เฉลยไว้ก่อนเลยว่า ลดเวลาการเกิด Cold Start จาก .NET Core 2.1 ได้ถึง 50% เลยทีเดียว</p>



<span id="more-5712"></span>



<p>ผมเคยเขียนในบล็อกเรื่อง <a href="https://myifew.com/5565/benchmark-netcore-3-and-readytorun-for-aws-lambda/">ทดสอบความเร็ว .NET Core 3.0 และฟีเจอร์ ReadyToRun (R2R) ในการทำ AWS Lambda</a> ซึ่งตอนนั้น ได้ทำเป็น Custom Runtime และผลลัพธ์ที่ได้ .NET Core 3.0 เร็วกว่า 2.1 นิดเดียว แทบไม่เห็นความต่าง ดีขึ้นแค่เสถียร เพราะมันไปช้าที่ AWS Custom Library ดังนั้นผมจึงแนะนำให้ทุกคนไปใช้ <a rel="noreferrer noopener" href="https://github.com/zaccharles/lambda-native" target="_blank">LambdaNative</a> แทน แต่ก็ยังมีปัญหาว่า <a rel="noreferrer noopener" href="https://github.com/dotnet/corert/issues/7386" target="_blank">Microsoft.DotNet.ILCompiler</a>&nbsp;ไม่รองรับการทำงานที่ซับซ้อน อย่างการเรียกใช้งาน Database ร่วมกับ EFCore หรือ Dapper จึงทำให้ต้องเลี่ยงไปใช้ต่อตรงด้วย ADO.NET  แทน</p>



<h2 class="wp-block-heading">ปัจจัยที่ทำให้ .NET Core 3.1 ทำงานได้ไวใน AWS Lambda</h2>



<p>จากที่ทดสอบเอง, ดู Benchmark, และอ่านๆ จากหลายแหล่ง พบว่ามีหลักๆ มี 2 เรื่อง ที่เป็นฟีเจอร์ใหม่ใน .NET Core 3 ที่ช่วยให้ AWS Lambda ทำงานได้ไวขึ้น คือ </p>



<ol class="wp-block-list"><li><strong>Built-in JSON support</strong> : ซึ่งปกติผู้เขียน C# เวลาต้องแปลง Object, Array ออกไปเป็น JSON จะต้องใช้ Library อย่างเช่น Newtonsoft.Json ซึ่งมันดีนะ แต่มันก็คือการจัดการ String บน .NET แบบ UTF-16 ซึ่งจะเปลืองทรัพยากรมากในการ convert ไปๆมาๆ<br><br>แต่สำหรับ .NET Core 3.0 เป็นต้นมา ได้ built-in JSON Converter เข้าไปเลย และทำงานแบบ UTF-8 encoder ตั้งแต่ต้น จึงไวขึ้น และเมื่อ Core มันมี JSON Converter มาแล้ว ก็ไม่ต้องเรียก 3rd Party Library จึงทำให้ AWS Lambda เราตัวเล็กลงด้วย (ดูการเปรียบเทียบ <a rel="noreferrer noopener" href="https://michaelscodingspot.com/the-battle-of-c-to-json-serializers-in-net-core-3/" target="_blank">The Battle of C# to JSON Serializers in .NET Core 3</a>)</li><li><strong>ReadyToRun</strong> :  เป็นฟีเจอร์ที่ Microsoft ภูมิใจนำเสนอมา ตั้งแต่ปล่อยตัว .NET Core 3.0 Preview 6 แล้ว นั่นคือทำให้ .NET Core application แปลงเป็น Native App ส่งผลให้มีการทำงานที่ไวขึ้น มี startup time ที่ดีขึ้น (AOT – ahead-of-time) เพราะไปลดปริมาณงานที่ต้องแปลง Byte Code เป็น Executable Code ลง (JIT – Just in Time)</li></ol>



<p>ดังนั้น แม้ว่า 3.1 จะเร็วขึ้นมาบ้าง แต่ก็ยังแนะนำให้ทำ AWS Lambda เป็น ReadyToRun นะครับ</p>



<h2 class="wp-block-heading">การ Migrate ไปใช้ .NET Core 3.1 Runtime บน AWS Lambda</h2>



<p>แก้แค่ 3 จุดหลักๆ (ส่วนโค้ดอื่นๆ ที่ไม่รองรับหลังจากทำ 3+1 ข้อนี้ ก็ไปตามแก้กันต่ออีกนะ)</p>



<h4 class="wp-block-heading">1. เปลี่ยน Target Framework ใหม่</h4>



<pre class="wp-block-code"><code>&lt;TargetFramework>netcoreapp3.1&lt;/TargetFramework></code></pre>



<h4 class="wp-block-heading">2. <strong>เปลี่ยนการเรียกใช้ JSON ใหม่</strong></h4>



<p>ใช้ JSON Package ใหม่ชื่อ</p>



<pre class="wp-block-code"><code>Amazon.Lambda.Serialization.SystemTextJson</code></pre>



<p>และระบุใน Class Handler ใหม่เป็น</p>



<pre class="wp-block-code"><code>&#91;assembly: LambdaSerializer(typeof(Amazon.Lambda.Serialization.SystemTextJson.DefaultLambdaJsonSerializer))]</code></pre>



<h4 class="wp-block-heading">3. ใน Deployment Config เปลี่ยน Function Runtime ใหม่ (aws-lambda-tools-defaults.json)</h4>



<pre class="wp-block-code"><code>"framework" : "netcoreapp3.1",
"function-runtime":"dotnetcore3.1",</code></pre>



<h4 class="wp-block-heading">ในกรณี ต้องการทำเป็น ReadyToRun จะต้องเพิ่มใน Csproj file ด้วยดังนี้</h4>



<pre class="wp-block-code"><code>&lt;PublishReadyToRun>true&lt;/PublishReadyToRun>
&lt;TieredCompilation>false&lt;/TieredCompilation>
&lt;TieredCompilationQuickJit>false&lt;/TieredCompilationQuickJit></code></pre>



<p>ที่ต้องปิด Tiered Compilation เพราะจะเกิด Startup Time สูงขึ้น และใช้ Memory เยอะขึ้นโดยใช่เหตุ (อ้างอิง Performance จาก <a href="https://medium.com/@zaccharles/net-core-3-0-aws-lambda-benchmarks-and-recommendations-8fee4dc131b0">.NET Core 3.0 AWS Lambda Benchmarks and Recommendations</a> และหากใครต้องการเข้าใจ <a href="https://docs.microsoft.com/en-us/dotnet/core/whats-new/dotnet-core-3-0#tiered-compilation">What&#8217;s new in .NET Core 3.0</a>)</p>



<p>และถ้าจะใช้ ReadyToRun จะต้อง Build และ Publish เป็น Linux Runtime นะครับ ไม่สามารถ Build Image บน Windows/Mac แล้วเอาขึ้นไปทำงานบน AWS Lambda ได้ เนื่องจากบนนั้นจะเป็น Amazon Linux 2 (ไปดูลิงค์ Demo ของผมที่ด้านล่างได้ ว่าทำอย่างไร)</p>



<h2 class="wp-block-heading">ผลทดสอบ Cold Start ของ AWS Lambda .NET Core 2.1 เมื่อเทียบกับ 3.1 และ 3.1 ReadyToRun</h2>



<p>ผมใช้โค้ดทดสอบตัวเดิมกับที่เคยทดสอบในบล็อก <a href="https://myifew.com/5565/benchmark-netcore-3-and-readytorun-for-aws-lambda/">ทดสอบความเร็ว .NET Core 3.0 และฟีเจอร์ ReadyToRun (R2R) ในการทำ AWS Lambda</a> ซึ่งได้ลองรัน 2.1 ก่อน จากนั้น migrate ไปเป็น 3.1 เพื่อเก็บข้อมูล และสุดท้ายก็ Compile ไปเป็น 3.1 ReadyToRun เพื่อเก็บข้อมูล</p>



<p>โดยการทำงานของโค้ดจะมีการเชื่อมต่อ MySQL Database 5.6 โดยมี table ชื่อ members ที่มี 3 รายการ (ค่าจะผันผวนบ้าง อาจเกิดจากการเรียกใช้ MySQL Database)</p>



<figure class="wp-block-image size-large"><img loading="lazy" decoding="async" width="1602" height="374" src="https://myifew.com/wp-content/uploads/2020/05/compare-aws-lambda-netcore21-31-31r2r.png" alt="" class="wp-image-5715"/></figure>



<p>จะสังเกตได้ว่า แค่แปลงไปใช้ Runtime 3.1 และ Built-in JSON ก็ลดเวลาการเกิด Cold Start ลงไปหนึ่งเท่าตัวเลยทีเดียว </p>



<p>และเมื่อทำเป็น 3.1 ReadyToRun ก็เร็วขึ้นอีกราวๆ 100ms และเมื่อเพิ่ม config เพื่อปิดการใช้ Tiered Compilation ก็ทำให้เร็วขึ้นไปอีก 100-300ms เมื่อใช้ Memory 128MB และ 256MB</p>



<p>แต่มีข้อสังเกตว่า การทำ ReadyToRun มีความเสถียรขึ้น รันครั้งที่ 2 เป็นต้นไป ไม่เจอว่าเกิน 100ms เลย (อย่าลืมว่า AWS Lambda คิดเงินทุกๆ 100ms ดังนั้นเมื่อเกิน มันจะปัดเศษขึ้นไปเป็นอีก 100ms) ส่วนความเร็วที่ผันผวนขึ้นลง 3-5ms ผมคิดว่ามีความต่างน้อยมากๆ ไม่จำเป็นต้องใส่ใจก็ได้ ถ้างานเราไม่ได้ซีเรียสกับเวลาแค่นี้</p>



<p>และเมื่อเราไม่ใช้ LambdaNative กับ <a rel="noreferrer noopener" href="https://github.com/dotnet/corert/issues/7386" target="_blank">Microsoft.DotNet.ILCompiler</a>&nbsp; ตัวปัญหาแล้ว มันจึงทำให้เราใช้ Library ต่างๆ อย่าง Dapper ได้แล้วครับ ไม่ต้องใช้ ADO มา  Mapping Object เองอีกต่อไป แต่ก็ต้องแลกกับเวลาที่มากขึ้นอีก 15-30%</p>



<figure class="wp-block-image size-large"><img loading="lazy" decoding="async" width="698" height="346" src="https://myifew.com/wp-content/uploads/2020/05/compare-aws-lambda-netcore31r2r-dapper.png" alt="" class="wp-image-5716"/></figure>



<h2 class="wp-block-heading">สรุป</h2>



<p>ใครใช้ .NET Core 2.1 บน AWS Lambda อยู่ หรือใช้ LambdaNative หรือใช้ Custom Runtime ตามที่ผมเคยเขียนถึง แนะนำให้เปลี่ยนมาใช้ .NET Core 3.1 ของ AWS Official Runtime ครับ ไวกว่ามาก และถ้าเป็นไปได้ ก็ควรทำให้เป็น ReadyToRun Image ด้วย เพื่อประสิทธิภาพที่ดีขึ้นครับ</p>



<p>ส่วนใครอยากลองดูโค้ดตัวอย่าง ดูได้จากด้านล่างนี้เลย</p>



<p><strong>Code Demo</strong></p>



<ul class="wp-block-list"><li><a rel="noreferrer noopener" href="https://github.com/ifew/aws-lambda-db" target="_blank">AWS Lambda .NET Core 3.1 แบบธรรมดา</a></li><li><a rel="noreferrer noopener" href="https://github.com/ifew/aws-lambda-db-r2r" target="_blank">AWS Lambda .NET Core 3.1 แบบทำเป็น ReadyToRun</a></li><li><a rel="noreferrer noopener" href="https://github.com/ifew/aws-lambda-dapper-r2r" target="_blank">AWS Lambda .NET Core 3.1 แบบทำเป็น ReadyToRun และใช้ Dapper ในการเรียกใช้งาน MySQL Database</a></li></ul>



<p><strong>อ้างอิง</strong></p>



<ul class="wp-block-list"><li><a href="https://aws.amazon.com/blogs/compute/announcing-aws-lambda-supports-for-net-core-3-1/">https://aws.amazon.com/blogs/compute/announcing-aws-lambda-supports-for-net-core-3-1/</a></li><li><a href="https://docs.microsoft.com/en-us/dotnet/core/whats-new/dotnet-core-3-0">https://docs.microsoft.com/en-us/dotnet/core/whats-new/dotnet-core-3-0</a></li><li><a href="https://medium.com/@zaccharles/a-close-look-at-net-core-3-1-on-aws-lambda-9ccec4dd96be">https://medium.com/@zaccharles/a-close-look-at-net-core-3-1-on-aws-lambda-9ccec4dd96be</a></li><li><a href="https://vincepergolizzi.com/programming/2020/04/01/dotnet-core-3.1-lambda.html">https://vincepergolizzi.com/programming/2020/04/01/dotnet-core-3.1-lambda.html</a></li></ul>
]]></content:encoded>
					
					<wfw:commentRss>https://myifew.com/5712/benchmark-netcore-31-aws-lambda/feed/</wfw:commentRss>
			<slash:comments>0</slash:comments>
		
		
			</item>
		<item>
		<title>ลองทำ PHP Serverless ง่ายๆ ด้วย bref</title>
		<link>https://myifew.com/5628/build-php-serverless-by-bref/</link>
					<comments>https://myifew.com/5628/build-php-serverless-by-bref/#respond</comments>
		
		<dc:creator><![CDATA[iFew]]></dc:creator>
		<pubDate>Sat, 07 Mar 2020 16:30:08 +0000</pubDate>
				<category><![CDATA[Technology]]></category>
		<category><![CDATA[AWS]]></category>
		<category><![CDATA[AWS Lambda]]></category>
		<category><![CDATA[Microservices]]></category>
		<category><![CDATA[PHP]]></category>
		<category><![CDATA[Serverless]]></category>
		<guid isPermaLink="false">https://myifew.com/?p=5628</guid>

					<description><![CDATA[ไหนๆ ก็เขียน PHP อยู่แล้ว เลยมาลองทำเป็น Serverless ทดสอบบน AWS Lambda สักหน่อย ถ้าวิธีปกติคือทำเป็น Custom Runtime แต่ขั้นตอนอาจจะยุ่งยากนิดนึง เลยมีคนทำเครื่องมือชื่อ bref ขึ้นมา ใช้ง่ายมากครับ&#8230;]]></description>
										<content:encoded><![CDATA[
<p>ไหนๆ ก็เขียน PHP อยู่แล้ว เลยมาลองทำเป็น Serverless ทดสอบบน AWS Lambda สักหน่อย ถ้าวิธีปกติคือทำเป็น <a rel="noreferrer noopener" aria-label="PHP Custom Runtime (opens in a new tab)" href="https://aws.amazon.com/blogs/apn/aws-lambda-custom-runtime-for-php-a-practical-example/" target="_blank">Custom Runtime</a> แต่ขั้นตอนอาจจะยุ่งยากนิดนึง เลยมีคนทำเครื่องมือชื่อ <a rel="noreferrer noopener" href="https://bref.sh/" target="_blank">bref</a> ขึ้นมา ใช้ง่ายมากครับ ลองมาดูวิธีทำกัน</p>



<span id="more-5628"></span>



<h2 class="wp-block-heading">bref คืออะไร</h2>



<p><a rel="noreferrer noopener" href="https://bref.sh/" target="_blank">bref</a> เป็นเครื่องมือ open source หนึ่งที่ไว้ deploy และ run serverless สำหรับภาษา PHP โดยมันไป integrate กับ <a rel="noreferrer noopener" href="https://serverless.com/" target="_blank">Serverless Framework</a> อีกที (แน่นอน ต้องติดตั้ง Serverless Framework ก่อน) ซึ่งปัจจุบันใช้ได้เฉพาะ AWS Lambda เท่านั้น</p>



<h2 class="wp-block-heading">สิ่งที่ต้องมีในเครื่อง</h2>



<ul class="wp-block-list"><li>PHP 7.2+</li><li>Composer</li><li>AWS account (ต้องมี permission อย่างน้อย: CloudFormation, CloudWatch, Lambda, S3, API Gateway)</li><li>Node.js</li></ul>



<h2 class="wp-block-heading">จัดแจงเตรียมเครื่องมือทำงาน</h2>



<p><strong>ตรียม AWS Credential Key และ Secret Code</strong> (ถ้ามีแล้ว ข้ามได้ครับ)</p>



<figure class="wp-block-image size-large"><img loading="lazy" decoding="async" width="663" height="294" src="https://myifew.com/wp-content/uploads/2020/03/AWS_Security_Credentials9-1.png" alt="" class="wp-image-5629"/></figure>



<p><strong>Install serverless</strong> (ถ้ามีแล้ว ข้ามได้ครับ)</p>



<pre class="wp-block-code"><code>npm install -g serverless</code></pre>



<p><strong>Config serverless with AWS</strong> (ถ้าเคยทำแล้ว ข้ามได้ครับ)</p>



<pre class="wp-block-code"><code>serverless config credentials --provider aws --key &lt;key> --secret &lt;secret></code></pre>



<p><strong>สร้างโฟลเดอร์ที่จะใช้งาน</strong></p>



<pre class="wp-block-code"><code>mkdir php-bref-function</code></pre>



<p><strong>Install bref</strong></p>



<pre class="wp-block-code"><code>composer require bref/bref</code></pre>



<h2 class="wp-block-heading">ทำ PHP Function Serverless ตัวแรก</h2>



<p><strong>Create PHP Lambda</strong></p>



<pre class="wp-block-code"><code>vendor/bin/bref init</code></pre>



<p>เมื่อ Enter จะแสดงข้อความให้เลือกประเภท Lambda</p>



<pre class="wp-block-code"><code>What kind of lambda do you want to create? (you will be able to add more functions later by editing `serverless.yml`) [PHP function]:
  [0] PHP function
  [1] HTTP application
  [2] Console application</code></pre>



<p>ในที่นี้เราจะสร้าง PHP Function สามารถกด Enter ผ่านไป หรือกดคีย์หมายเลข 0 ก็ได้ จะได้ไฟล์ประมาณนี้</p>



<figure class="wp-block-image size-large"><img loading="lazy" decoding="async" width="1752" height="680" src="https://myifew.com/wp-content/uploads/2020/03/php-bref-function-init.png" alt="" class="wp-image-5630"/></figure>



<p><strong>Config PHP Lambda</strong></p>



<p>เมื่อสร้าง Application เสร็จ ให้แก้ไขไฟล์ serverless.yml ดังนี้</p>



<figure class="wp-block-image size-large"><img loading="lazy" decoding="async" width="1258" height="900" src="https://myifew.com/wp-content/uploads/2020/03/php-bref-function-config.png" alt="" class="wp-image-5631"/></figure>



<p>โดย</p>



<p><strong>provider &gt; region</strong> : คือทวีปที่เราใช้ AWS โดยทั่วไปที่ไทยใช้เป็น ap-southeast-1 <br><strong>functions &gt; php-bref-function</strong> : โดย php-bref-function ชื่อ function เราแก้ตามที่ต้องการได้เลย<br><strong>functions &gt; handler</strong> : ช่องทางแรกที่ Lambda จะเรียกทำงาน<br><strong>functions &gt; layers</strong> : เป็น custom runtime แต่ในที่นี้ bref ได้สร้างเป็น layer เพื่อเรียกใช้งาน ไม่เอามาผสมกับ core code ของเรา (ทำความเข้าใจ <a rel="noreferrer noopener" aria-label="AWS Lambda Layer (opens in a new tab)" href="https://myifew.com/5189/working-with-aws-lambda-layers-and-csharp-net-core2/" target="_blank">AWS Lambda Layer คืออะไร</a>)</p>



<blockquote class="wp-block-quote is-layout-flow wp-block-quote-is-layout-flow"><p><strong>อธิบายเพิ่มเติมเกี่ยวกับ Layers</strong><br><strong>สำหรับ PHP functions ให้ระบุเป็น php-74 หรือ php-73 </strong><br>ใช้สำหรับสร้าง PHP function เพื่อประมวลผลเท่านั้น ไม่ใช่  HTTP Application<br><br><strong>สำหรับ HTTP applications ให้ระบุเป็น php-74-fpm หรือ php-73-fpm </strong><br>โดย runtime ตัวนี้จะใช้ PHP-FPM เพื่อทำงานแบบ HTTP applications จะเหมือนกับที่เราเขียน PHP ปกติ คือเขียนเสร็จแล้วต้องเปิดผ่าน PHP hosting หรือพวก Nginx, Apache (สามารถใช้ได้กับ Symfony และ Laravel ได้ด้วยนะ)<br>(Read more at <a href="https://bref.sh/docs/runtimes/#bref-runtimes" target="_blank" rel="noreferrer noopener" aria-label="https://bref.sh/docs/runtimes/#bref-runtimes (opens in a new tab)">https://bref.sh/docs/runtimes/#bref-runtimes</a>)</p></blockquote>



<h2 class="wp-block-heading">ทดสอบ PHP Function Serverless บน localhost</h2>



<p>เมื่อสร้างเสร็จแล้ว ลองทดสอบบนเครื่องตัวเองได้นะครับ ด้วยคำสั่งนี้</p>



<pre class="wp-block-code"><code>serverless invoke local --docker -f &lt;function-name></code></pre>



<p>เปลี่ยน &lt;function-name&gt; ให้เป็นชื่อ function ของเรา ซึ่งที่นี้คือ <strong>php-bref-function</strong> โดยผลลัพธ์ &#8220;Hello World&#8221;</p>



<figure class="wp-block-image size-large"><img loading="lazy" decoding="async" width="1754" height="288" src="https://myifew.com/wp-content/uploads/2020/03/php-bref-function-invocation-localhost.png" alt="" class="wp-image-5632"/></figure>



<p>แต่ถ้าอยากส่ง data payload เข้าไป ให้ใช้คำสั่งนี้</p>



<pre class="wp-block-code"><code>serverless invoke local --docker -f &lt;function-name> --data '{"name": "iFew"}'</code></pre>



<p>ผลลัพธ์ที่ได้คือ &#8220;Hello iFew&#8221;</p>



<figure class="wp-block-image size-large"><img loading="lazy" decoding="async" width="1754" height="292" src="https://myifew.com/wp-content/uploads/2020/03/php-bref-function-invocation-localhost-payload.png" alt="" class="wp-image-5633"/></figure>



<h2 class="wp-block-heading">ทำการ Deploy PHP Function Serverless เพื่อทดสอบบน AWS Lambda</h2>



<p>จะลอง Deploy ขึ้นไปทดสอบบน AWS Lambda ด้วยคำสั่ง</p>



<pre class="wp-block-code"><code>serverless deploy</code></pre>



<p>จะได้ผลลัพธ์ประมาณนี้</p>



<figure class="wp-block-image size-large"><img loading="lazy" decoding="async" width="1380" height="762" src="https://myifew.com/wp-content/uploads/2020/03/php-bref-function-deploy.png" alt="" class="wp-image-5635"/></figure>



<p>สังเกตว่ามีจะมีการ pack file และทำการ upload ขึ้นไปไว้บน S3 เพื่อพักไว้ก่อน และ Deploy ไปที่ Lambda อีกที</p>



<figure class="wp-block-image size-large"><img loading="lazy" decoding="async" width="2644" height="1122" src="https://myifew.com/wp-content/uploads/2020/03/php-bref-function-on-aws-lambda.png" alt="" class="wp-image-5636"/></figure>



<p>ทดสอบอีกรอบว่า บน AWS Lambda ใช้งานได้จริงหรือไม่</p>



<pre class="wp-block-code"><code>serverless invoke -f &lt;function-name></code></pre>



<p>ถ้าใช้งานได้ ก็จะแสดง output ออกมาตามที่โค้ดเราเขียนไว้ </p>



<figure class="wp-block-image size-large"><img loading="lazy" decoding="async" width="1148" height="94" src="https://myifew.com/wp-content/uploads/2020/03/php-bref-function-invocation.png" alt="" class="wp-image-5637"/></figure>



<h2 class="wp-block-heading">ผลการทดสอบความเร็ว ของ PHP Function Serverless ด้วย bref</h2>



<p>ทำงานด้วย Memory 1024MB (ค่ามาตรฐาน) ได้ประมาณ​ 14-15ms</p>



<figure class="wp-block-image size-large"><img loading="lazy" decoding="async" width="2604" height="1268" src="https://myifew.com/wp-content/uploads/2020/03/php-bref-function-memory1024mb.png" alt="" class="wp-image-5638"/></figure>



<p>ทำงานด้วย Memory 256MB ได้ประมาณ​ 75-80ms</p>



<figure class="wp-block-image size-large"><img loading="lazy" decoding="async" width="2590" height="1218" src="https://myifew.com/wp-content/uploads/2020/03/php-bref-function-memory256mb.png" alt="" class="wp-image-5639"/></figure>



<p>ทำงานด้วย Memory 128MB ได้ประมาณ​ 270-290ms</p>



<figure class="wp-block-image size-large"><img loading="lazy" decoding="async" width="2590" height="1234" src="https://myifew.com/wp-content/uploads/2020/03/php-bref-function-memory128mb.png" alt="" class="wp-image-5640"/></figure>



<h2 class="wp-block-heading">ลองทำ PHP HTTP Application Serverless ดูบ้าง</h2>



<p>วิธีการทำแบบเดิมเลยครับ ต่างกันตอนเลือกประเภท Lambda ให้เลือกประเภท [1] HTTP application แทน พร้อมกับแก้ serverless.yml ในหัวข้อ function &gt; layer ให้เป็น php-74-fpm หรือ php-73-fpm แทน</p>



<p>วิธีการทดสอบบน localhost ให้ใช้คำสั่ง ดังนี้</p>



<pre class="wp-block-code"><code>php -S localhost:8000 index.php</code></pre>



<p>จากนั้นเปิด browser ไปที่ http://localhost:8000 จะเจอหน้าตาแบบนี้</p>



<figure class="wp-block-image size-large"><img loading="lazy" decoding="async" width="2866" height="1492" src="https://myifew.com/wp-content/uploads/2020/03/php-bref-http-invocation-localhost.png" alt="" class="wp-image-5641"/></figure>



<p>ซึ่งเมื่อเรา Deploy ไปอยู่บน AWS Lambda ขั้นตอนของ bref จะทำการ upload ไป S3 และ deploy ไป Lambda เหมือนกับกรณี PHP Function แต่เพิ่มเติมคือ จะไปทำการสร้าง API Gateway ให้ด้วย เพื่อเป็นช่องทางการ access เข้าไปใช้งาน ไม่ว่าจะเป็น Website หรือ API (ให้ดูที่ endpoints)</p>



<figure class="wp-block-image size-large"><img loading="lazy" decoding="async" width="1428" height="820" src="https://myifew.com/wp-content/uploads/2020/03/php-bref-http-deploy.png" alt="" class="wp-image-5642"/></figure>



<p>ถ้าลองเปิด url ดูแล้วก็จะได้เหมือนกับที่ลองใน localhost</p>



<figure class="wp-block-image size-large is-resized"><img loading="lazy" decoding="async" src="https://myifew.com/wp-content/uploads/2020/03/php-bref-http-on-aws-lambda.png" alt="" class="wp-image-5643" width="580" height="301"/></figure>



<p>ส่วนผลการทดสอบความเร็ว จะไม่สามารถ invocation ได้นะครับ ต้องเปิดผ่าน API Gateway เท่านั้น</p>



<figure class="wp-block-image size-large"><img loading="lazy" decoding="async" width="2602" height="1248" src="https://myifew.com/wp-content/uploads/2020/03/php-bref-http-cant-invocation.png" alt="" class="wp-image-5644"/></figure>



<p>ผมลองดู SPeed Network ด้วย Chrome แต่ความเร็วที่ได้ มันผ่าน stack ชั้น API Gateway ก่อน เลยทำให้ไม่รู้ว่า AWS Lambda จริงๆทำงานเร็วแค่ไหน แต่ผลลัพธ์ที่ได้ อยู่ราวๆ 80-100ms ถือว่าเร็วเลยหละ</p>



<figure class="wp-block-image size-large"><img loading="lazy" decoding="async" width="2880" height="1238" src="https://myifew.com/wp-content/uploads/2020/03/php-bref-http-test-on-chrome-1.png" alt="" class="wp-image-5650"/></figure>



<h2 class="wp-block-heading">วิธีการลบ PHP Serverless Application บน AWS Lambda</h2>



<p>ผมแนะนำว่าให้ใช้คำสั่งแทนนะครับ เพราะการไปลบบน AWS Lambda อย่างเดียว อาจจะไม่ครบ อาจจะต้องไปตามหาใน S3, CloudFormation, API Gateway และอาจจะหลายสิ่ง จึงแนะนำให้ใช้คำสั่งของ serverless แทน ดังนี้ครับ</p>



<pre class="wp-block-code"><code>serverless remove</code></pre>



<p>อยากลบตัวไหน ให้เข้าไปที่ folder ของ application นั้น มันจะทำการลบตามที่เรา config ในไฟล์ serverless.yml ครับ</p>



<h2 class="wp-block-heading">สรุป</h2>



<p>ถ้าใครอยากทำ Serverless ด้วย PHP และ AWS Lambda ผมแนะนำให้ใช้ <a rel="noreferrer noopener" href="https://bref.sh/" target="_blank">bref</a> เลยครับ ค่อนข้างสะดวก ส่วนเรื่องความเร็วที่ได้ถือว่าประทับใจ เมื่อเทียบกับ .NET Core  (ใน Memory เท่ากัน) แต่ก็ยังช้ากว่า Node.js, Go, Python</p>



<p><strong>ข้อสังเกต</strong></p>



<ol class="wp-block-list"><li>Default Memory คือ 1024MB ระวังให้ดีว่าจะเผลอเสียเงินเยอะโดยใช้เหตุครับ</li><li>Default Timeout คือ 6 วินาที</li><li>ไม่พบ Cold Start จากการ run ครั้งแรก เพราะครั้งต่อๆ ไป ก็ยังทำงานด้วยเวลาพอกัน</li></ol>



<p><strong>ตัวอย่างโค้ด</strong></p>



<ul class="wp-block-list"><li>PHP HTTP Application : <a href="https://github.com/ifew/php-bref-http">https://github.com/ifew/php-bref-http</a></li><li>PHP Function : <a href="https://github.com/ifew/php-bref-function">https://github.com/ifew/php-bref-function</a></li></ul>
]]></content:encoded>
					
					<wfw:commentRss>https://myifew.com/5628/build-php-serverless-by-bref/feed/</wfw:commentRss>
			<slash:comments>0</slash:comments>
		
		
			</item>
		<item>
		<title>ทดสอบความเร็ว .NET Core 3.0  และฟีเจอร์ ReadyToRun (R2R) ในการทำ AWS Lambda</title>
		<link>https://myifew.com/5565/benchmark-netcore-3-and-readytorun-for-aws-lambda/</link>
					<comments>https://myifew.com/5565/benchmark-netcore-3-and-readytorun-for-aws-lambda/#respond</comments>
		
		<dc:creator><![CDATA[iFew]]></dc:creator>
		<pubDate>Tue, 01 Oct 2019 19:20:35 +0000</pubDate>
				<category><![CDATA[Technology]]></category>
		<category><![CDATA[.Net Core]]></category>
		<category><![CDATA[AWS]]></category>
		<category><![CDATA[AWS Lambda]]></category>
		<category><![CDATA[Performance Test]]></category>
		<category><![CDATA[Serverless]]></category>
		<guid isPermaLink="false">https://myifew.com/?p=5565</guid>

					<description><![CDATA[เมื่อ 23 กันยายน ที่ผ่านมา Microsoft ได้ฤกษ์เปิดตัว .NET Core 3.0 (3.0.100) ตัวเต็มให้ได้เล่นกัน หลังจากเป็นข่าวมาร่วมปี ซึ่งมาพร้อมกับฟีเจอร์และพัฒนา Performance จากเวอร์ชัน 2.2 เยอะพอสมควร ใครอยากรู้รายละเอียดสามารถไปตามอ่านได้ที่&#8230;]]></description>
										<content:encoded><![CDATA[
<p>เมื่อ 23 กันยายน ที่ผ่านมา <a rel="noreferrer noopener" aria-label="Microsoft ได้ฤกษ์เปิดตัว .NET Core 3.0 (opens in a new tab)" href="https://devblogs.microsoft.com/dotnet/announcing-net-core-3-0/" target="_blank">Microsoft ได้ฤกษ์เปิดตัว .NET Core 3.0</a> (3.0.100) ตัวเต็มให้ได้เล่นกัน หลังจากเป็นข่าวมาร่วมปี ซึ่งมาพร้อมกับฟีเจอร์และพัฒนา Performance จากเวอร์ชัน 2.2 เยอะพอสมควร ใครอยากรู้รายละเอียดสามารถไปตามอ่านได้ที่ <a rel="noreferrer noopener" aria-label="Announcing .NET Core 3.0 (opens in a new tab)" href="https://devblogs.microsoft.com/dotnet/announcing-net-core-3-0/" target="_blank">Announcing .NET Core 3.0</a></p>



<p>ในครั้งนี้มีฟีเจอร์หนึ่งที่น่าสนใจคือ ReadyToRun (R2R) ซึ่งจริงๆ มันมาตั้งแต่ .NET Core 3 Preview 6 แล้ว นั่นคือทำให้ .NET Core application แปลงเป็น Native App ให้ไวขึ้น ส่งผลกับ startup time ที่ดีขึ้น (AOT &#8211; ahead-of-time) จากแต่เดิมซึ่งมีเพียงแค่ แปลง Byte Code เป็น Executable Code  (JIT &#8211; Just in Time)</p>



<p>ผมนี่ตาลุกวาวทันที ไม่รอช้าที่จะหยิบมันมาทดสอบเป็น AWS Lambda เพื่อดูปัญหา Cold Start ว่าจะดีขึ้นได้มากน้อยแค่ไหน</p>



<span id="more-5565"></span>



<h2 class="wp-block-heading">เริ่มต้นใช้ ReadyToRun (How to Use)</h2>



<p>วิธีการใช้ ReadyToRun  ไม่ยาก แค่ตั้ง Config ในไฟล์ csproj ได้เลย ตัวอย่างที่ผมใช้จะประมาณนี้</p>



<pre class="wp-block-code"><code>&lt;PropertyGroup>
    &lt;OutputType>Exe&lt;/OutputType>
    &lt;TargetFramework>netcoreapp3.0&lt;/TargetFramework>
    &lt;LangVersion>latest&lt;/LangVersion>
    &lt;AWSProjectType>Lambda&lt;/AWSProjectType>
    &lt;PublishReadyToRun>true&lt;/PublishReadyToRun>
    &lt;PublishTrimmed>true&lt;/PublishTrimmed>
    &lt;PublishSingleFile>true&lt;/PublishSingleFile>
&lt;/PropertyGroup></code></pre>



<p>Config R2R ที่ใช้บ่อย จะมี ดังนี้</p>



<ul class="wp-block-list"><li><strong>PublishReadyToRun</strong>&nbsp;คือ บอกให้แอพเรา Publish เป็น ReadyToRun Image</li><li><strong>PublishTrimmed</strong>&nbsp;คือ ตัด assemblies ที่ไม่ได้ใช้ออกไป ซึ่งจะช่วยลดขนาดไฟล์ได้ (โดยจะวิเคราะห์ IL จาก Library ที่ชื่อว่า&nbsp;<a rel="noreferrer noopener" href="https://github.com/mono/linker" target="_blank">IL linker</a>&nbsp;สามารถไปดู Docs เพื่อกำหนดค่าต่างๆได้)</li><li><strong>PublishSingleFile</strong>&nbsp;คือ บอกว่าจะ Publish โดยแพ็คมาให้เป็น Executable File ไฟล์เดียว (ดูข้อมูลเพิ่มเติมได้ที่ <a href="https://github.com/dotnet/designs/blob/master/accepted/single-file/design.md" target="_blank" rel="noreferrer noopener" aria-label="design (opens in a new tab)">design</a>)</li></ul>



<p>และเวลา publish จะต้องระบุ RuntimeIdentifier ด้วย เช่น</p>



<pre class="wp-block-code"><code>dotnet publish -r win-x64 -c Release</code></pre>



<p>ดู Runtime เพิ่มเติมได้ที่ <a rel="noreferrer noopener" aria-label=".NET Core Runtime IDentifier (RID) catalog (opens in a new tab)" href="https://docs.microsoft.com/en-us/dotnet/core/rid-catalog" target="_blank">.NET Core Runtime IDentifier (RID) catalog</a></p>



<h2 class="wp-block-heading">วิธีการทดสอบ (Test Scenarios)</h2>



<p>ผมเคยทดสอบ AWS Lambda เพื่อแก้ปัญหา Cold Start ด้วยหลายๆ วิธี (อ่านเพิ่มเติมได้จาก <a rel="noreferrer noopener" aria-label="9 วิธีจัดการกับ Cold Start ใน Serverless ให้ทำงานได้ไวที่สุด พร้อมตัวอย่าง (opens in a new tab)" href="https://myifew.com/5386/9-ways-to-fighting-with-cold-start-in-serverless/" target="_blank">9 วิธีจัดการกับ Cold Start ใน Serverless ให้ทำงานได้ไวที่สุด พร้อมตัวอย่าง</a>) ครั้งนี้ผมจึงอ้างอิงจากโค้ดชุดเดิม โดยจะทดสอบ 3 Scenarios ดังนี้</p>



<ol class="wp-block-list"><li>ชุดโค้ด .NET Core Dll file ปกติ ที่เปลี่ยน Version จาก AWS Standard Runtime .NET Core 2.1 มาเป็นทำ Custom Runtime ขึ้นมาเองด้วย .NET Core 3.0.100 ผ่านไลบรารี่ <a rel="noreferrer noopener" aria-label="Amazon.Lambda.RuntimeSupport (opens in a new tab)" href="https://aws.amazon.com/blogs/developer/announcing-amazon-lambda-runtimesupport/" target="_blank">Amazon.Lambda.RuntimeSupport</a></li><li>ชุดโค้ด .NET Core Native ที่ใช้ไลบรารี่ <a rel="noreferrer noopener" aria-label="LambdaNative (opens in a new tab)" href="https://github.com/zaccharles/lambda-native" target="_blank">LambdaNative</a> และ <a rel="noreferrer noopener" aria-label="Microsoft.DotNet.ILCompiler (opens in a new tab)" href="https://dotnet.myget.org/feed/dotnet-core/package/nuget/Microsoft.DotNet.ILCompiler" target="_blank">Microsoft.DotNet.ILCompiler</a> แต่เปลี่ยน Version จากเดิม Custom Runtime .NET Core 2.2 มาเป็น Custom Runtime .NET Core 3.0.100</li><li>ชุดโค้ด .NET Core Native ที่ใช้ไลบรารี่ <a rel="noreferrer noopener" href="https://github.com/zaccharles/lambda-native" target="_blank">LambdaNative</a> อย่างเดียว แต่เปลี่ยน Version จากเดิม Custom Runtime .NET Core 2.2 มาเป็น Custom Runtime .NET Core 3.0.100 โดยจะ Publish เป็น ReadyToRun  Image</li></ol>



<p>Database ผมใช้ MySQL 5.5 ที่อยู่บน AWS RDS วงเดียวกันกับ AWS Lambda ที่ผมทดสอบ และจำนวนข้อมูลมีเพียง 3 Rows</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1608" height="1052" src="https://myifew.com/wp-content/uploads/2019/10/lambda-test-data-3rows.png" alt="" class="wp-image-5570"/></figure>



<h2 class="wp-block-heading">ผลการทดสอบ (Benchmark Results)</h2>



<h4 class="wp-block-heading">Scenario 1 &#8211; Normal Compile with Custom Runtime .NET Core 2.2 กับ .NET Core 3.0.100 โดยใช้ ADO ในการเชื่อมต่อกับ MySQL</h4>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1070" height="346" src="https://myifew.com/wp-content/uploads/2019/10/benchmark-crt-ado-netcore22-vs-netcore3.png" alt="" class="wp-image-5571"/></figure>



<p>จะสังเกตว่า .NET Core 3.0 ภาพรวมแล้ว เวลาของ Cold Start จะไวกว่า 2.2 เมื่อใช้ Memory 256Mb และ 512Mb  แต่เมื่อเรียกใช้งานซ้ำๆ .NET Core 3.0 ทำงานได้ไวกว่าทุกกรณี</p>



<p>คราวนี้ลองมาดูเทียบ .NET Core 3.0.100 ด้วยกัน แบบที่ใช้ ReadyToRun กับแบบไม่ใช้</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1106" height="346" src="https://myifew.com/wp-content/uploads/2019/10/benchmark-crt-ado-netcore3-vs-netcore3-readytorun.png" alt="" class="wp-image-5572"/></figure>



<p>ผลคือ เมื่อทำเป็น ReadyToRun จะมีความเร็วของ Cold Start มากกว่าเดิม 50% แบบชนะขาดลอย  ส่วนการเรียกใช้งานแบบซ้ำๆ ตรงนี้ เลขที่ผมได้ ยังเหวี่ยงๆ แต่โดยรวมถือว่าดีกว่า</p>



<p>มาลองเทียบ .NET Core 3.0.100 แบบที่ใช้ ReadyToRun แต่ใช้ Library Connect MySQL ต่างๆ</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1686" height="346" src="https://myifew.com/wp-content/uploads/2019/10/benchmark-crt-ado-dapper-efcore-netcore3-readytorun.png" alt="" class="wp-image-5573"/></figure>



<p>ผลตามคาด คล้ายกับบล็อกเดิมที่ผมเคยทดสอบ คือ ADO เร็วที่สุด รองมาเป็น Dapper และช้าสุดคือ EFCore แต่ทั้งนี้ ผมจะทดสอบใหม่อีกครั้งเมื่อ EFCore 3.0 รองรับการใช้งาน MySQL (ซึ่งตอนที่ผมเขียนบล็อกนี้ รองรับ MySQL แค่ EFCore 2.2 อยู่นะครับ)</p>



<p>และให้สังเกต ว่าตรง memory 128 ผมมีคำว่า (max) อยู่ หมายถึงว่า .NET Core 3.0 กิน memory มากขึ้นจน Lambda ใช้เกิน Memory 128Mb</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="924" height="77" src="https://myifew.com/wp-content/uploads/2019/10/lambda-memory-peak.png" alt="" class="wp-image-5574"/><figcaption>ขึ้นแสดงแบบนี้ หมายถึงใช้ Memory มากกว่าที่จัดสรรให้ ตรงนี้ผมไม่แน่ใจว่ากระทบเงินอย่างไร แต่ที่เห้นชัดคือ ส่งผลทำให้ process ช้าลงพอสมควร น่าจะเกิดจากการต้องใช้และรอคืน memory ออกมาทำงาน จนกว่าจะเสร็จ process ทั้งหมด</figcaption></figure>



<h4 class="wp-block-heading">Scenario 2 &#8211; .NET Core 3.0.100 with LambdaNative and Microsoft.DotNet.ILCompiler โดยใช้ ADO ในการเชื่อมต่อกับ MySQL</h4>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1050" height="378" src="https://myifew.com/wp-content/uploads/2019/10/benchmark-lambdanative-ado-netcore22-vs-netcore3.png" alt="" class="wp-image-5577"/></figure>



<p>ที่ผมเคยเขียนถึง .NET Core 2.2 และ LambdaNative ไว้ใน <a rel="noreferrer noopener" href="https://myifew.com/5386/9-ways-to-fighting-with-cold-start-in-serverless/" target="_blank">9 วิธีจัดการกับ Cold Start ใน Serverless ให้ทำงานได้ไวที่สุด พร้อมตัวอย่าง</a> เป็นวิธีการที่แก้ปัญหา Cold Start ได้ดีที่สุด แต่เมื่อทดสอบกับ .NET Core 3.0 ด้วยแล้วปรากฎว่า ผมลัพธ์ดีขึ้นกว่าแต่เดิม และความเร็วเสถียรพอกันทุกครั้งที่มีการเรียกซ้ำๆ เป็นที่น่าพอใจมาก</p>



<h4 class="wp-block-heading">Scenario 3 &#8211; .NET Core 3.0.100 with LambdaNative and ReadyToRun โดยใช้ ADO ในการเชื่อมต่อกับ MySQL</h4>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1566" height="382" src="https://myifew.com/wp-content/uploads/2019/10/benchmark-lambdanative-ado-netcore3-readytorun.png" alt="" class="wp-image-5579"/></figure>



<p>ลองใช้ LambdaNative ร่วมกับ ReadyToRun ปรากฎว่าทำงานได้ช้ากว่า ใช้ไลบรารี่ Microsoft.DotNet.ILCompiler เสียอีก</p>



<p>การทำงานของ ReadyToRun จะมีการ Extract Bundle Files (ดูข้อมูลเพิ่มเติมได้ที่ <a rel="noreferrer noopener" aria-label="Bundler (opens in a new tab)" href="https://github.com/dotnet/designs/blob/master/accepted/single-file/extract.md" target="_blank">bundler</a>) ก่อนทำการ Execution ผมจึงลองทดสอบแบบทำเป็น Single File ที่เอา Bundle รวมกับโค้ด  (<strong>PublishSingleFile</strong> เป็น true) และแบบ Multiple File ที่เอา Bundle แยกกับโค้ด  (<strong>PublishSingleFile</strong> เป็น false) มาเทียบกัน ปรากฎว่า แบบ Single File ทำงานได้เร็วกว่า (น่าจะมาจาก ahead-of-time ใน .NET Core 3.0 นี้)</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1986" height="380" src="https://myifew.com/wp-content/uploads/2019/10/benchmark-lambdanative-ado-netcore22-vs-netcore3-readytorun.png" alt="" class="wp-image-5578"/></figure>



<p>ถ้าลองเทียบทั้งหมดจาก Scenario 2 และ Scenario 3 พบว่า .NET Core 3 NativeLambda ที่ทำงานกับ Microsoft.DotNet.ILCompiler จะทำงานได้ไวที่สุด</p>



<p>คราวนี้ลองถ้าลองเปลี่ยนตัว Connect MySQL มาเป็น Dapper ดูบ้าง ผลลัพธ์จะต่างกันขนาดไหน</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1988" height="876" src="https://myifew.com/wp-content/uploads/2019/10/benchmark-lambdanative-ado-dapper-netcore3-readytorun.png" alt="" class="wp-image-5580"/></figure>



<p>ในส่วนของ LambdaNative ที่ทำงานกับ <a rel="noreferrer noopener" aria-label="Microsoft.DotNet.ILCompiler ปัจจุบันยังไม่รองรับ Dapper เช่นเดิม (opens in a new tab)" href="https://github.com/dotnet/corert/issues/7386" target="_blank">Microsoft.DotNet.ILCompiler ปัจจุบันยังไม่รองรับ Dapper เช่นเดิม</a> ดังนั้นจึงมีแต่ผลทดสอบของ LambdaNative ที่ทำงานกับ ReadyToRun ซึ่งผลลัพธ์ที่ได้ Cold Start จะช้ากว่า ADO ประมาณ 0.5-2 วินาที เลยทีเดียว</p>



<h2 class="wp-block-heading">สรุป</h2>



<p>จากผลลัพธ์ทั้งหมดที่ลองทดสอบ ฟันธงได้ว่า .NET Core 3.0 ทำงานได้ไวกว่าเวอร์ชันเก่า  ในการนำมาทำ Serverless อย่าง AWS Lambda </p>



<p>แต่ด้วยความที่เป็นของใหม่ ผมเชื่อว่าคงมีอะไรปล่อยออกมาอีกเรื่อยๆ ดังนั้น ขอแนะนำชาว .NET Core ว่ามันคุ้มค่าแก่การ migrate ไปเป็น 3.0 ครับ (ข่าวว่า .NET Core 3.1 จะออกมาในช่วงพฤศจิกายนนี้)</p>



<p>บล็อกนี้ พอเห็น Release ปุ๊บรีบทดสอบและเขียนไว้ทันที ใหม่จนยังไม่มีใครเขียนถึงเลย หากผิดพลาดประการใด สามารถแนะนำ/ติชมได้นะครับ</p>



<h2 class="wp-block-heading">Demo Source Code</h2>



<ul class="wp-block-list"><li><a href="https://github.com/ifew/netcore3-readytorun-efcore">netcore3-readytorun-efcore</a> &#8211; ทดสอบ .NET Core 3 ReadyToRun กับ EFCore</li><li><a href="https://github.com/ifew/netcore3-readytorun-dapper">netcore3-readytorun-dapper</a> &#8211; ทดสอบ .NET Core 3 ReadyToRun กับ Dapper</li><li><a href="https://github.com/ifew/aws-lambda-netcore3-db">aws-lambda-netcore3-db</a> &#8211; ทดสอบทำ Lambda ด้วย .NET Core 3 กับ ADO</li><li><a href="https://github.com/ifew/aws-lambda-netcore3-efcore">aws-lambda-netcore3-efcore</a> &#8211; ทดสอบทำ Lambda ด้วย .NET Core 3 กับ EFCore</li><li><a href="https://github.com/ifew/aws-lambda-netcore3-dapper">aws-lambda-netcore3-dapper</a> &#8211; ทดสอบทำ Lambda ด้วย .NET Core 3 กับ Dapper</li><li><a href="https://github.com/ifew/aws-lambda-lambdanative-db">aws-lambda-lambdanative-db</a> &#8211; ทดสอบทำ LambdaNative ด้วย .NET Core 3 กับ ADO</li><li><a href="https://github.com/ifew/aws-lambda-netcore3-readytorun-db">aws-lambda-netcore3-readytorun-db</a> &#8211; ทดสอบทำ Lambda ด้วย .NET Core 3 ReadyToRun กับ ADO</li><li><a href="https://github.com/ifew/aws-lambda-netcore3-readytorun-efcore">aws-lambda-netcore3-readytorun-efcore</a> &#8211; ทดสอบทำ Lambda ด้วย .NET Core 3 ReadyToRun กับ EFCore</li><li><a href="https://github.com/ifew/aws-lambda-netcore3-readytorun-dapper">aws-lambda-netcore3-readytorun-dapper</a> &#8211; ทดสอบทำ Lambda ด้วย .NET Core 3 ReadyToRun กับ Dapper</li></ul>
]]></content:encoded>
					
					<wfw:commentRss>https://myifew.com/5565/benchmark-netcore-3-and-readytorun-for-aws-lambda/feed/</wfw:commentRss>
			<slash:comments>0</slash:comments>
		
		
			</item>
		<item>
		<title>9 วิธีจัดการกับ Cold Start ใน Serverless ให้ทำงานได้ไวที่สุด พร้อมตัวอย่าง</title>
		<link>https://myifew.com/5386/9-ways-to-fighting-with-cold-start-in-serverless/</link>
					<comments>https://myifew.com/5386/9-ways-to-fighting-with-cold-start-in-serverless/#respond</comments>
		
		<dc:creator><![CDATA[iFew]]></dc:creator>
		<pubDate>Thu, 30 May 2019 12:50:32 +0000</pubDate>
				<category><![CDATA[Technology]]></category>
		<category><![CDATA[AWS]]></category>
		<category><![CDATA[AWS Lambda]]></category>
		<category><![CDATA[Cold Start]]></category>
		<category><![CDATA[Microservices]]></category>
		<category><![CDATA[Performance Test]]></category>
		<category><![CDATA[Serverless]]></category>
		<guid isPermaLink="false">https://myifew.com/?p=5386</guid>

					<description><![CDATA[Update 6 Nov 2019 &#8211; ทาง AWS ได้แก้ไขปัญหาเรื่อง Cold Start เนื่องจากใช้ VPC แล้ว ซึ่งมีผลกับ Region ดังนี้ US West&#8230;]]></description>
										<content:encoded><![CDATA[
<p><strong>Update 6 Nov 2019</strong> &#8211; ทาง AWS ได้แก้ไขปัญหาเรื่อง Cold Start เนื่องจากใช้ VPC แล้ว ซึ่งมีผลกับ Region ดังนี้ US West (N. California), EU (Ireland), EU (Paris), Asia Pacific (Mumbai), Asia Pacific (Seoul), Asia Pacific (Singapore), Asia Pacific (Sydney), and South America (San Paulo)    &#8211; ที่มา <a rel="noreferrer noopener" href="https://aws.amazon.com/th/blogs/compute/announcing-improved-vpc-networking-for-aws-lambda-functions/" target="_blank">aws.amazon.com</a> </p>



<p><strong>Update 27 Sep 2019</strong> &#8211;  ทาง AWS ได้แก้ไขปัญหาเรื่อง Cold Start เนื่องจากใช้ VPC แล้ว ซึ่งมีผลกับ Region ดังนี้  US East (Ohio), EU (Frankfurt), and Asia Pacific (Tokyo)  &#8211; ที่มา <a rel="noreferrer noopener" aria-label="aws.amazon.com (opens in a new tab)" href="https://aws.amazon.com/th/blogs/compute/announcing-improved-vpc-networking-for-aws-lambda-functions/" target="_blank">aws.amazon.com</a></p>



<p style="text-align:center">&#8230;</p>



<p>ผมเชื่อว่าใครก็ตามที่ตัดสินใจทำ Serverless มักเจอปัญหาที่แก้ไม่ตกคือ เรื่อง Cold Start ซึ่งบล็อกนี้ผมขอแชร์ประสบการณ์ที่ได้แก้ปัญหาทั้งหมด 9 วิธี เผื่อจะเป็นประโยชน์กันนะครับ  (ขออ้างอิงจากผู้ให้บริการ AWS Lambda และภาษาที่ใช้พัฒนา C# .NET Core)</p>



<span id="more-5386"></span>



<h2 class="wp-block-heading">ทำความเข้าใจ Cold Start คืออะไร</h2>



<p>ขอเกริ่นถึงสั้นๆ ของการทำงาน Serverless คือ เมื่อมีการร้องขอ (Request) ไปที่  Instances (เข้าใจง่ายๆ คือ Server ตัวหนึ่ง) เพื่อให้มันทำงาน มันจะใช้เวลาช่วงหนึ่งเพื่อทำการ Initial Instances จากนั้นค่อยเรียกโค้ดที่เราเขียน ขึ้นมาทำงานตามคำสั่ง </p>



<p>เมื่อมีการเรียกซ้ำอีก มันก็จะทำต่อเนื่องได้ทันที ไม่ต้องเสียเวลา Initial Instances ใหม่ แต่เมื่อทำงานเสร็จและปล่อยไว้สักระยะหนึ่ง เจ้า Instances ตัวนั้นก็จะเข้าสู่โหมด Idle อีก (หรือบางคนเรียกว่า หลับ, sleep) ดังนั้นเมื่อมีการร้องขอ (Request) ใช้งานอีกครั้ง จะเกิดกระบวนการ Initial Instances แบบเดิม ก่อนที่จะทำงานโค้ดของเรา</p>



<p>การ Initial Server ในครั้งแรก หรือครั้งหลังจากโหมด Idle นี่เอง เราจะเรียกว่า Cold Start</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1200" height="578" src="https://myifew.com/wp-content/uploads/2019/02/01_cold_start_aws_client-1200x578.jpeg" alt="" class="wp-image-5286" srcset="https://myifew.com/wp-content/uploads/2019/02/01_cold_start_aws_client-1200x578.jpeg 1200w, https://myifew.com/wp-content/uploads/2019/02/01_cold_start_aws_client-1024x493.jpeg 1024w, https://myifew.com/wp-content/uploads/2019/02/01_cold_start_aws_client-768x370.jpeg 768w, https://myifew.com/wp-content/uploads/2019/02/01_cold_start_aws_client-700x337.jpeg 700w, https://myifew.com/wp-content/uploads/2019/02/01_cold_start_aws_client-600x289.jpeg 600w, https://myifew.com/wp-content/uploads/2019/02/01_cold_start_aws_client.jpeg 1266w" sizes="auto, (max-width: 1200px) 100vw, 1200px" /><figcaption>Initial Instances หรือ Cold Start คือช่วงตั้งแต่ Download the code, Container Setup, Initialization and start of the code</figcaption></figure>



<p>และมากกว่านั้น ภาษาที่ต้องมีการ Compile จะมีขั้นตอน 1-3 นานกว่าภาษา Script หลายเท่า (ผมลองพ่น Hello World โดยใช้ Node ไม่ถึง 1 วินาที, เทียบกับ .Net Core ใช้เวลา 4-5 วินาที)</p>



<p>และมีคำถามบ่อยๆ ว่าจะเข้าสู่โหมด Idle ภายในกี่นาที คำตอบคือ ไม่มีใครสามารถตอบได้ชัดเจน บางแหล่งก็บอก 30-45 นาที บางแหล่งก็บอก 15 นาที แต่สำหรับผมที่เล่นเอง จะเจอแถวๆ 5 นาทีเป็นต้นไป เรียกได้ว่าลุกไปเข้าห้องน้ำแป๊บนึง พอกลับมา มันก็ช้าอีกแล้ว (ผมใช้ AWS Lambda ที่ ap-southeast-1)  </p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1624" height="860" src="https://myifew.com/wp-content/uploads/2019/05/estimate-time-to-sleep.png" alt="" class="wp-image-5407"/><figcaption>ความเป็นไปได้ที่จะเกิด Cold Start หลังมีการเรียกใช้ไปแล้ว x นาที (รูปอ้างอิงจาก <a rel="noreferrer noopener" href="https://mikhail.io/serverless/coldstarts/aws/" target="_blank">mikhail.io</a>)</figcaption></figure>



<p>อ่านพื้นฐานเรื่อง Serverless และ AWS Lambda ฉบับเต็มได้ที่ </p>



<ul class="wp-block-list"><li><a href="https://myifew.com/5166/understand-serverless-with-aws-lambda-for-newbie/">มาทำความรู้จักกับ Serverless และ AWS Lambda ฉบับคนคิดจะใช้</a></li></ul>



<h2 class="wp-block-heading">9 วิธีเพื่อใช้ลดเวลา Cold Start</h2>



<p>คราวนี้ Cold Start ได้เป็นเสี้ยนหนามของผมมาก สุดท้ายแล้ว ผมพบว่ามี 9 วิธี ที่จะลดเวลา Cold Start ได้ ผู้อ่านสามารถเลือกไปทำ หรือจะทำทั้งหมดเลยก็ได้ ลองดูกันว่ามีวิธีใดบ้าง</p>



<ol class="wp-block-list"><li>เพิ่ม Memory Allocated</li><li>Lean and Clean Code</li><li>ลด และ/หรือ เลือกใช้ Dependency ที่จำเป็นและเหมาะสม</li><li>Warm ตัว Server ของเราตลอดเวลา</li><li>Warm ตัว Server ของเราตามเวลาที่เหมาะสม</li><li>ใช้ Custom Runtimes และเลือกใช้ Runtimes ที่ทำงานได้เร็ว</li><li>ใช้ Custom Runtimes และทำเป็น Native App</li><li>หลีกเลี่ยงการใช้ VPC ถ้าไม่จำเป็น</li><li>เลือกภาษาเพื่อใช้ทำ Serverless ที่เหมาะสม</li></ol>



<h2 class="wp-block-heading">1. เพิ่ม Memory Allocated</h2>



<p>วิธีง่ายที่สุดคือใช้เงินจอง Memory ให้กับ AWS Lambda Instances มากๆ (Memory Allocated) ซึ่งมันก็เห็นผลได้เร็วและชัดเจนซะด้วย </p>



<p>กรณีแบบนี้ผมคิดว่าเหมาะกับการคำนวณ (Compute) ที่ไม่ได้เกิดขึ้นบ่อย แต่ต้องทำงานได้ไว และไม่ต้องการแก้ไขให้ยุ่งยาก &#8212; หรือพูดสั้นๆ คือ ยอมจ่ายค่า Memory ที่แพงขึ้น แลกกับ Duration Time ที่ลดลง แต่นานๆที เรียกใช้</p>



<p>กรณีนี้ผมทดลองด้วย C# .NET Core 2.1 ดึงข้อมูลจาก MySQL จำนวน 3 รายการ</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="2542" height="1214" src="https://myifew.com/wp-content/uploads/2019/05/memory-allocated-memberlist-128.png" alt="" class="wp-image-5387"/><figcaption>Memory Allocated 128 MB &#8211; Duration Time 10,169 ms</figcaption></figure>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1473" height="234" src="https://myifew.com/wp-content/uploads/2019/05/memory-allocated-memberlist-256.png" alt="" class="wp-image-5388"/><figcaption>Memory Allocated 256 MB &#8211; Duration Time 4999 ms</figcaption></figure>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1457" height="237" src="https://myifew.com/wp-content/uploads/2019/05/memory-allocated-memberlist-512.png" alt="" class="wp-image-5389"/><figcaption>Memory Allocated 512 MB &#8211; Duration Time 2,426 ms</figcaption></figure>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1450" height="233" src="https://myifew.com/wp-content/uploads/2019/05/memory-allocated-memberlist-1024.png" alt="" class="wp-image-5390"/><figcaption>Memory Allocated 1,024 MB &#8211; Duration Time 1,153 ms</figcaption></figure>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1834" height="758" src="https://myifew.com/wp-content/uploads/2019/05/memory-allocated-compare.png" alt="" class="wp-image-5391"/><figcaption>เทียบโค้ดชุดเดียวกัน แต่ใช้ Memory Allocate ต่างกันกัน</figcaption></figure>



<p>สังเกตได้ว่า ยิ่งเพิ่ม Memory Allocated มากเท่าไร ก็ยิ่งลดเวลา Cold Start ได้มากเป็นเท่าตัว</p>



<p>แต่วิธีการนี้อาจจะไม่ดี 100% กับทุกกรณี เช่น ทดสอบแสดงข้อความ &#8220;Hello World&#8221; ด้วยภาษา Javascript  ตามรูปด้านล่าง</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1347" height="643" src="https://myifew.com/wp-content/uploads/2019/05/aws-coldstart-js-by-memory.png" alt="" class="wp-image-5411"/><figcaption>Javascript แสดงข้อความ Hello Wold ที่จัดสรร Memory ต่างกัน แต่ผลออกมาแทบไม่ต่างกัน <br>(รูปอ้างอิงจาก <a href="https://mikhail.io/2018/08/serverless-cold-start-war/">mikhail.io</a>)</figcaption></figure>



<p>สังเกตว่า แม้เพิ่ม Memory Allocated ไปมากเท่าใด เวลาของ Cold Start ก็ไม่ต่างกันมากนัก และ Memory สูงที่สุด ก็ไม่ได้บ่งบอกว่าจะเร็วที่สุดด้วย ดังนั้นเราจึงควรหาวิธีอื่นเพื่อลด Cold Start แทน</p>



<p>ถ้าให้ผมแนะนำ กรณีเพิ่ม Memory Allocated นี้ เราไม่ควรจะคาดเดา ซึ่งผมเคยเขียนวิธีการทดสอบไว้แล้ว สามารถอ่านวิธีการได้ที่ <a rel="noreferrer noopener" aria-label="จัดสรรทรัพยากร AWS Lambda อย่างไร ให้ราคาถูกและเร็วที่สุด (opens in a new tab)" href="https://myifew.com/5339/how-to-optimize-memory-on-aws-lambda-for-costs-and-fastest/" target="_blank">จัดสรรทรัพยากร AWS Lambda อย่างไร ให้ราคาถูกและเร็วที่สุด</a></p>



<h2 class="wp-block-heading" id="mce_24">2. Lean and Clean Code</h2>



<p>วิธีการนี้อาศัยทักษะทางเขียนโปรแกรมหน่อย คือการทำให้ Application เรากระชับที่สุด อะไรไม่ใช้ก็ตัดทิ้ง อะไรที่จำเป็นน้อย ทำให้ไม่จำเป็นเลยได้หรือไม่ อะไรเปลี่ยนวิธีได้ก็เปลี่ยน เช่น</p>



<p>การเรียกใช้ Config จาก JSON File วิธีนี้จะทำให้เราต้องเรียกใช้ Package Dependency จัดการไฟล์ JSON ต่างๆ ซึ่งเราสามารถย้ายไปใช้ Environment Variable แทนได้  ซึ่งถูกต้องตามหลัก <a href="https://myifew.com/5080/the-twelve-factors/">The Twelve Factors</a> ด้วย</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1894" height="408" src="https://myifew.com/wp-content/uploads/2019/05/lean-clean-config-in-environment-1.png" alt="" class="wp-image-5393"/><figcaption>Environment Variable ใน AWS Lambda</figcaption></figure>



<p>หรือการเรียกใช้ Package Dependency เกินความจำเป็น หรือไม่ได้ใช้แล้ว แต่ลืมลบ กรณีนี้เครื่องมือดีๆ สามารถช่วยเราได้ เช่น Visual Studio ที่จะทำการแจ้งเตือนให้เราลบตัวไม่ใช้ออก</p>



<p>ทั้งนี้ รวมไปถึงพวก Class, Function ต่างๆ ที่เขียนไว้ แต่ไม่ได้ใช้ ก็ต้องลบออกด้วย</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1184" height="332" src="https://myifew.com/wp-content/uploads/2019/05/lean-clean-delete-not-using-used.png" alt="" class="wp-image-5420"/><figcaption>Package อะไรไม่ใช้ ก็ลบออกไป</figcaption></figure>



<p>ยิ่งเราทำให้ขนาดโค้ดเราเล็กลงได้เท่าไร ก็ทำให้ไฟล์ที่แพ็คขึ้นสู่ AWS Lambda (Deployment Package File) เล็กลงตามเท่านั้น มันจะส่งผลเรื่อง Cold Start ที่ลดเวลาลงไปด้วย </p>



<p>ลองดูได้จากสถติของเว็บ <a rel="noreferrer noopener" href="https://mikhail.io/serverless/coldstarts/aws/" target="_blank">mikhail.io</a> ที่เขาใช้ Javascript ในการแสดงคำว่า &#8220;Hello World&#8221; แต่ทดลองเพิ่ม NPM packages เข้าไปใน Deployment Package File ด้วย </p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1624" height="860" src="https://myifew.com/wp-content/uploads/2019/05/compare-cold-start-deployment-size.png" alt="" class="wp-image-5409"/><figcaption>เทียบเวลาการเกิด Cold Start เมื่อ Deployment Package (แบบ zip แล้ว) มีขนาดไฟล์ที่ใหญ่ขึ้น (รูปอ้างอิงจาก <a rel="noreferrer noopener" href="https://mikhail.io/serverless/coldstarts/aws/" target="_blank">mikhail.io</a> ) <br></figcaption></figure>



<p>ผลกระทบต่อมา ถ้าเราทำให้ Deployment Package File ใหญ่ขึ้น หมายถึงการทำให้ AWS Lambda Instances ใหญ่ขึ้นตามด้วย ถ้ามาถึงขั้นนี้แล้ว แม้จะเพิ่ม Memory เข้าไปช่วยเท่าไร ก็ไม่เห็นผลนะครับ เพราะ Cold Start ยังนานเหมือนเดิม แถมเสียเงินค่า Memory เพิ่มขึ้นด้วย</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1622" height="862" src="https://myifew.com/wp-content/uploads/2019/05/compare-cold-start-instance-size.png" alt="" class="wp-image-5410"/><figcaption>เมื่อ Instances Lambda เราใหญ่ การเพิ่ม Memory เข้าไป ดูจะไม่ข่วยทำให้เวลาในการ Cold Start ลดขึ้น แถมเสียเงินไปโดยเปล่าประโยชน์  (รูปอ้างอิงจาก <a rel="noreferrer noopener" href="https://mikhail.io/serverless/coldstarts/aws/" target="_blank">mikhail.io</a> ) </figcaption></figure>



<h2 class="wp-block-heading">3. ลด และ/หรือ เลือกใช้ Dependency ที่จำเป็นและเหมาะสม</h2>



<p>ข้อนี้จำเป็นต้อง Debug ให้เห็นรายละเอียดก่อน เพื่อประเมินว่าอะไรคือจุดที่ทำให้โค้ดเราทำงานช้า หรืออาจจะไปดูจาก Benchmark ที่มีผู้ทดสอบเปรียบเทียบ Package ต่างๆ เพื่อที่เราจะได้เลือกใช้ได้เหมาะสมตั้งแต่เริ่มพัฒนางาน</p>



<p>ปกติผมเขียน .NET Core ผมจะใช้ Entity Framework (EF) Core หรือ EF Core เพราะมันเหมือนชุดมาตรฐานที่ทำงานได้เข้ากันดี และมีเครื่องมือค่อนข้างครบ แต่ปรากฎว่า EF Core นี่เอง ทำให้ใช้เวลา Cold Start มากกว่าเดิมอย่างแสนสาหัส!! </p>



<p>ผมฝืนใช้มันอยู่นาน และไปหาทางแก้วิธีอื่นแทน จนกลับมาเอะใจตอนที่ Compile ว่า EF Core มันตัวใหญ่นี่หว่า ผมเลยลองเปลี่ยนไปใช้ ADO.NET หรือ Micro ORM อย่าง Dapper แทน ซึ่งผลปรากฎว่า เร็วขึ้นอีก 100% เลยทีเดียว </p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="2028" height="756" src="https://myifew.com/wp-content/uploads/2019/05/change-dependency-compare-128mb.png" alt="" class="wp-image-5401"/><figcaption>ผลเปรียบเทียบในการใช้ ADO.NET, Dapper, EF Core เพื่อเรียกข้อมูลจาก Database</figcaption></figure>



<p>เรื่องนี้คงไม่ขอเขียนเยอะ กราฟมันบอกชัดเจนอยู่แล้ว หากใครสนใจลองเอาไปเปรียบเทียบ สามารถดาวน์โหลดโค้ดของผมได้จากที่นี่ (ต้องใช้ .NET Core 3.0, MySQL)</p>



<ul class="wp-block-list"><li>ADO.NET &#8211; <a href="https://github.com/ifew/aws-lambda-db">https://github.com/ifew/aws-lambda-db</a></li><li>Dapper &#8211; <a href="https://github.com/ifew/aws-lambda-dapper">https://github.com/ifew/aws-lambda-dapper</a></li><li>Entity Framework (EF) Core &#8211; <a href="https://github.com/ifew/aws-lambda-efcore">https://github.com/ifew/aws-lambda-efcore</a></li></ul>



<p>กรณี Benchmark เทียบ Database Framework ดูเพิ่มเติมได้ที่</p>



<ul class="wp-block-list"><li><a rel="noreferrer noopener" aria-label="Dapper vs Entity Framework vs ADO.NET Performance Benchmarking (opens in a new tab)" href="https://exceptionnotfound.net/dapper-vs-entity-framework-vs-ado-net-performance-benchmarking/" target="_blank">Dapper vs Entity Framework vs ADO.NET Performance Benchmarking</a></li><li><a rel="noreferrer noopener" aria-label="Fetch performance of various .NET ORM / Data-access frameworks (opens in a new tab)" href="https://weblogs.asp.net/fbouma/fetch-performance-of-various-net-orm-data-access-frameworks" target="_blank">Fetch performance of various .NET ORM / Data-access frameworks</a></li><li><a href="https://koukia.ca/entity-framework-core-2-0-vs-dapper-net-performance-benchmark-querying-sql-azure-tables-7696e8e3ed28" target="_blank" rel="noreferrer noopener" aria-label="Entity Framework Core 2.0 vs. Dapper performance benchmark, querying SQL Azure tables (opens in a new tab)">Entity Framework Core 2.0 vs. Dapper performance benchmark, querying SQL Azure tables</a></li></ul>



<h2 class="wp-block-heading">4. Warm ตัว Server ของเราตลอดเวลา</h2>



<p>หากเรามีความจำเป็นต้องใช้ Lambda อยู่บ่อยๆ มีผู้เข้าใช้กระจัดกระจายแบบกำหนดเวลาไม่ได้ รู้แต่เข้าเมื่อไร ต้องเร็ว เช่น การแสดงรายการสินค้าของหน้าเว็บไซต์ E-Commerce เป็นต้น ถ้าสุดท้ายแล้วเราไม่สามารถแก้ไขโค้ดได้อีก หรือไม่อยากเพิ่ม Memory อีก เราก็ต้องทำให้ Server ของเราตื่นอยู่เสมอนั่นเอง</p>



<p>วิธีการคือ การตั้งเวลาให้มี Cronjob ไปเรียก (Ping) Server ของเราอยู่เรื่อยๆ อย่างเช่น AWS Lambda ที่ผมเจอ จะ Idle ทุกๆ 5 นาทีไปแล้ว ผมก็ต้องหา Cronjob ไปทำการ Invoke Lambda ประมาณนาทีที่ 5 เสมอ </p>



<p>วิธีการแบบนี้จะเปลือง Request มาก เพราะแปลว่า 1 วัน ผมจะต้องปลุกมัน 288 ครั้ง ถ้าครั้งหนึ่งต้องใช้เวลาประมวลผล (Duration Time) 2 วินาที และจอง Memory 512MB คิดดูว่าเดือนหนึ่งผมจะต้องเสียเงินไปเท่าไร</p>



<p>ซึ่งหากใครสนใจลองทำ วิธีนี้ค้นเจอได้ตามเว็บทั่วไป ส่วนมากจะใช้ CloudWatch ในการทำ ซึ่งผมไม่รู้ว่าทำไมถึงแนะนำให้ทำกันแบบนี้ แต่มันก็ไม่ผิดนะ ถ้าเหมาะสมกับรูปแบบงาน คือ ถ้าตัว Server เราถูกเรียกบ่อยทั้งวัน กอปรกับใช้เวลาประมวลผลไม่มาก ใช้ Memory ไม่สูง ก็อาจจะเหมาะสม</p>



<p>สามารถดูเพิ่มเติมได้ที่</p>



<ul class="wp-block-list"><li><a rel="noreferrer noopener" aria-label="How to Keep Your Lambda Functions Warm (opens in a new tab)" href="https://read.acloud.guru/how-to-keep-your-lambda-functions-warm-9d7e1aa6e2f0" target="_blank">How to Keep Your Lambda Functions Warm</a></li><li><a rel="noreferrer noopener" aria-label="Tutorial: Schedule AWS Lambda Functions Using CloudWatch Events (opens in a new tab)" href="https://docs.aws.amazon.com/AmazonCloudWatch/latest/events/RunLambdaSchedule.html" target="_blank">Tutorial: Schedule AWS Lambda Functions Using CloudWatch Events</a></li><li><a rel="noreferrer noopener" aria-label="Lambda Warmer: Optimize AWS Lambda Function Cold Starts (opens in a new tab)" href="https://www.jeremydaly.com/lambda-warmer-optimize-aws-lambda-function-cold-starts/" target="_blank">Lambda Warmer: Optimize AWS Lambda Function Cold Starts</a></li><li><a href="https://github.com/jeremydaly/lambda-warmer">Lambda Warmer﻿</a></li></ul>



<h2 class="wp-block-heading">5. Warm ตัว Server ของเราตามเวลาที่เหมาะสม</h2>



<p>วิธีการนี้เหมือนกับข้อ 4 คือมี Conjob ไปเรียก (Ping) Server แต่ต่างกันที่ กำหนดเวลาในการเรียกให้ถี่มากน้อย ขึ้นกับการใช้งานจริง</p>



<p>เช่น 08:00-22:00 คนเข้าเว็บไซต์เยอะ ก็ให้เรียกทุก 5 นาที เพื่อให้ Instances ไม่เข้าโหมด Idle ส่วนเวลาตั้งแต่ 22:01-07:59 ให้เรียกทุก 10 นาที แทน เป็นต้น</p>



<p>หรือใคร Advance หน่อย ก็ทำสคริ้ปอัตโนมัติเพื่อ Edit Config Lambda ในช่วงเวลา 22:01-07:59 เพื่อปรับ  Memory ขึ้นไป แต่ไม่ต้องทำการ Ping ซึ่งมันก็จะไปใช้สูตรแบบข้อ 1. เพิ่ม Memory Allocated แทนการใช้แบบข้อ 4. เพื่อ Warm ตัว Server ตลอดเวลา</p>



<p>สามารถดูเพิ่มเติมได้ที่ </p>



<ul class="wp-block-list"><li><a rel="noreferrer noopener" aria-label="AWS CLI - Modify the version-specific settings of a Lambda function  (opens in a new tab)" href="https://docs.aws.amazon.com/cli/latest/reference/lambda/update-function-configuration.html" target="_blank">AWS CLI &#8211; Modify the version-specific settings of a Lambda function </a></li><li><a rel="noreferrer noopener" aria-label="How to trigger aws lambda at different schedule every day (opens in a new tab)" href="https://stackoverflow.com/questions/53570401/how-to-trigger-aws-lambda-at-different-schedule-every-day" target="_blank">How to trigger aws lambda at different schedule every day</a></li></ul>



<h2 class="wp-block-heading">6. ใช้ Custom Runtimes และเลือกใช้ Runtimes ที่ทำงานได้เร็ว</h2>



<p>กรณีของผู้ให้บริการอย่าง Amazon ได้เพิ่มฟีเจอร์มาให้กับ AWS Lambda คือ การ  <a rel="noreferrer noopener" aria-label="Custom Runtimes (opens in a new tab)" href="https://docs.aws.amazon.com/lambda/latest/dg/runtimes-custom.html" target="_blank">Custom Runtimes</a> ได้เอง โดยไม่จำเป็นต้องใช้ตัวมาตรฐานที่ให้มา</p>



<p>ตัวอย่างเช่น .NET Core Runtime ที่ทาง Amazon มีมาให้คือ version 1.0 และ 2.1 เท่านั้น</p>



<p><strong>.NET Runtimes (ดูเพิ่มเติม ได้ที่ <a rel="noreferrer noopener" aria-label="AWS Lambda Runtimes (opens in a new tab)" href="https://docs.aws.amazon.com/lambda/latest/dg/lambda-runtimes.html" target="_blank">AWS Lambda Runtimes</a>)</strong></p>



<table class="wp-block-table"><tbody><tr><th>Name</th><th>Identifier</th><th>Languages</th><th>Operating System</th></tr><tr><td>.NET Core 2.1</td><td><code>dotnetcore2.1</code></td><td>C#PowerShell Core 6.0</td><td>Amazon Linux</td></tr><tr><td>.NET Core 1.0</td><td><code>dotnetcore1.0</code></td><td>C#</td><td>Amazon Linux</td></tr></tbody></table>



<p>แต่ปัจจุบันมี 2.2 และ 3.0 แล้ว ซึ่งถ้าเราพบว่า 3.0 ทำงานได้ดีกว่า หรือไวกว่า เราก็สามารถสร้าง Runtimes ขึ้นมาเองได้ หรืออาจจะไปสร้างเป็น <a href="https://aws.amazon.com/th/blogs/apn/aws-lambda-custom-runtime-for-php-a-practical-example/" target="_blank" rel="noreferrer noopener" aria-label="PHP Runtimes (opens in a new tab)">PHP Runtimes</a> ก็ยังได้ ถ้าเราอยากจะใช้ แต่ Amazon ไม่ได้ทำรองรับ</p>



<p>ในประเด็นนี้ ผมพยายามทดสอบหลายครั้ง ผมพบว่าใน .NET Core เมื่อทำ Custom Runtimes แล้ว ไม่ได้ช่วยทำให้ไวขึ้นแต่อย่างใด แถมช้าลงด้วย มีดีอย่างเดียวคือ ได้ใช้ฟีเจอร์ใหม่ๆ ในเวอร์ชั่นใหม่ (แต่ในภาษาอื่นๆ ถ้าใครทำดูแล้วได้ผลที่ดีขึ้น ลองมาแชร์กันได้ครับ)</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="2534" height="1090" src="https://myifew.com/wp-content/uploads/2019/05/crt-netcore21.png" alt="" class="wp-image-5398"/><figcaption>.NET Core 2.1 Runtimes (ตัวมาตรฐานของ Amazon)</figcaption></figure>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="2542" height="1092" src="https://myifew.com/wp-content/uploads/2019/05/crt-netcore30.png" alt="" class="wp-image-5399"/><figcaption>.NET Core 3.0 preview4 Runtimes (ตัว Custom Runtimes ที่ผมทำขึ้นมา)</figcaption></figure>



<p>สังเกตว่า ตัวมาตรฐาน .NET Core 2.1 Runtimes ทำงานได้ไวกว่าที่ผม Custom Runtimes เอง ที่เวอร์ชั่น 3.0 (แม้ว่าจริงๆแล้วผล <a href="https://devblogs.microsoft.com/dotnet/performance-improvements-in-net-core-3-0/" target="_blank" rel="noreferrer noopener" aria-label="Benchmark ของ .NET Core 3.0 (opens in a new tab)">Benchmark ของ .NET Core 3.0</a> จะทำงานได้ไวกว่า 2.1 ในสภาพแวดล้อมที่ไม่ใช่ AWS Lambda ก็ตาม)</p>



<p>ใครอยากทดสอบ Custom Runtimes ลองดูโค้ดตัวอย่างที่ผมทำไว้ได้</p>



<ul class="wp-block-list"><li>.NET Core 3.0  &#8211; <a rel="noreferrer noopener" aria-label="https://github.com/ifew/aws-lambda-crt-example (opens in a new tab)" href="https://github.com/ifew/aws-lambda-crt-example" target="_blank">https://github.com/ifew/aws-lambda-crt-example</a></li><li>.NET Core 3.0 with ADO.NET &#8211; <a rel="noreferrer noopener" aria-label="https://github.com/ifew/aws-lambda-crt-db (opens in a new tab)" href="https://github.com/ifew/aws-lambda-crt-db" target="_blank">https://github.com/ifew/aws-lambda-crt-db</a></li><li>.NET Core 3.0 with Dapper &#8211; <a rel="noreferrer noopener" aria-label="https://github.com/ifew/aws-lambda-crt-db-dapper (opens in a new tab)" href="https://github.com/ifew/aws-lambda-crt-db-dapper" target="_blank">https://github.com/ifew/aws-lambda-crt-db-dapper</a></li></ul>



<p>เรื่อง Custom Runtimes สามารถศึกษาเพิ่มเติมได้ที่นี่ครับ</p>



<ul class="wp-block-list"><li><a href="https://docs.aws.amazon.com/lambda/latest/dg/runtimes-custom.html">Custom AWS Lambda Runtimes</a></li><li><a href="https://medium.com/@zaccharles/benchmarking-lambdas-new-custom-runtime-for-net-core-43ea79b5a35a" target="_blank" rel="noreferrer noopener" aria-label="Benchmarking Lambda’s New Custom Runtime for .NET Core (opens in a new tab)">Benchmarking Lambda’s New Custom Runtime for .NET Core</a></li></ul>



<h2 class="wp-block-heading" id="mce_94">7. ใช้ Custom Runtimes และทำเป็น Native App</h2>



<p>แม้ว่าการทำ Custom Runtimes ผมจะไม่พอใจกับผลลัพธ์ที่เกิดขึ้น (กรณี C# .NET Core) แต่เราสามารถเอาวิธีการเดียวกันมาเรียกใช้ Binary Executable File แทนได้นะ ด้วยการทำโค้ดของเราให้เป็น Native Application ซะเลย </p>



<p>การแก้ไขปัญหาด้วยวิธีนี้ ผมภูมิใจนำเสนอมากๆๆๆ เรียกได้ว่า เป็นไม้ตาย ที่แก้ปัญหา Cold Start ได้อยู่หมัดจากทุกๆ วิธีที่ผมทำมาเลยทีเดียว แต่ก็ค่อนข้างทำยากและมีข้อจำกัดเยอะ</p>



<p>มาลองดูสถิติที่ผมทำบน C# .NET Core กัน ว่าผลแตกต่างแค่ไหน ซึ่งผมเทียบโดยใช้ Database Framework ต่างกัน 3 ตัว ที่ทำงานต่างกันบน Runtimes แบบปกติ (.NET Core 2.1), Custom Runtimes (.NET Core 3.0) และ Native (.NET Core  3.0 + CoreRT)</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="2106" height="310" src="https://myifew.com/wp-content/uploads/2019/05/nativeapp-compare.png" alt="" class="wp-image-5402"/></figure>



<p>จากตาราง ตัว Lambda Native (ช่องสีเขียว) ชนะใสๆ แบบขาดรอย เพราะ Cold Start แค่ 1.3 วินาที ในขณะที่ตัวอื่นๆ ทำงานที่ 10-33 วินาที (เทียบกันที่ ADO.NET)</p>



<p>สังเกตว่าทำไมผมมี Lambda Native แค่ ADO.NET อย่างเดียว </p>



<p>เนื่องจาก Lambda Native ใช้ Runtimes ชื่อว่า CoreRT ซึ่งมันเป็น .NET Core runtime อีกตัวที่พัฒนามาเพื่อ AOT (ahead of time compilation) โดยเฉพาะ กล่าวคือ มันจะแปลงภาษาจาก High-Level Programming เช่น C#, Java ไปเป็นภาษาเครื่อง (Machine code) กันเลยทีเดียว</p>



<p>แต่ๆๆ เจ้าตัว CoreRT ยังไม่สมบูรณ์นัก และมีผู้ใช้งานไม่มาก หาข้อมูลได้ยาก จึงทำให้การพัฒนาได้ยากตาม รวมไปถึงหลายสิ่งยังไม่สามารถใช้งานได้ เช่น <a rel="noreferrer noopener" aria-label="เมื่อมีการเอา Dapper มาใช้ แต่โดนแจ้ง Error ว่า &quot;Operation is not supported on this platform&quot; (opens in a new tab)" href="https://github.com/dotnet/corert/issues/7386" target="_blank">เมื่อมีการเอา Dapper มาใช้ แต่โดนแจ้ง Error ว่า &#8220;Operation is not supported on this platform&#8221;</a> ด้วยเหตุผลนี้เอง ผมถึงบอกว่า ทำไมยังมีข้อจำกัดอยู่ และทำได้ยาก</p>



<p>ดูตัวอย่างที่ผมทำได้ที่นี่</p>



<ul class="wp-block-list"><li>Hello World &#8211; <a href="https://github.com/ifew/aws-lambda-lambdanative">https://github.com/ifew/aws-lambda-lambdanative</a></li><li>ADO.NET &#8211; <a href="https://github.com/ifew/aws-lambda-lambdanative-db">https://github.com/ifew/aws-lambda-lambdanative-db</a></li></ul>



<p>สามารถดูข้อมูลการทำ Lambda Native เพิ่มเติมได้ที่</p>



<ul class="wp-block-list"><li>LambdaNative &#8211; <a href="https://github.com/zaccharles/lambda-native">https://github.com/zaccharles/lambda-native</a></li><li>CoreRT &#8211; <a href="https://github.com/dotnet/corert">https://github.com/dotnet/corert</a></li></ul>



<h2 class="wp-block-heading">8. หลีกเลี่ยงการใช้ VPC ถ้าไม่จำเป็น</h2>



<p><strong>Update 6 Nov 2019</strong> &#8211; ทาง AWS ได้แก้ไขปัญหาเรื่อง Cold Start เนื่องจากใช้ VPC แล้ว ซึ่งมีผลกับ Region ดังนี้ US West (N. California), EU (Ireland), EU (Paris), Asia Pacific (Mumbai), Asia Pacific (Seoul), Asia Pacific (Singapore), Asia Pacific (Sydney), and South America (San Paulo)    &#8211; ที่มา <a rel="noreferrer noopener" href="https://aws.amazon.com/th/blogs/compute/announcing-improved-vpc-networking-for-aws-lambda-functions/" target="_blank">aws.amazon.com</a> </p>



<p><strong>Update 27 Sep 2019</strong> &#8211;  ทาง AWS ได้แก้ไขปัญหาเรื่อง Cold Start เนื่องจากใช้ VPC แล้ว ซึ่งมีผลกับ Region ดังนี้  US East (Ohio), EU (Frankfurt), and Asia Pacific (Tokyo)  &#8211; ที่มา <a rel="noreferrer noopener" aria-label="aws.amazon.com (opens in a new tab)" href="https://aws.amazon.com/th/blogs/compute/announcing-improved-vpc-networking-for-aws-lambda-functions/" target="_blank">aws.amazon.com</a></p>



<p style="text-align:center">&#8230;</p>



<p>คำแนะนำจาก <a rel="noreferrer noopener" aria-label="theburningmonk.com (opens in a new tab)" href="https://theburningmonk.com/2018/01/im-afraid-youre-thinking-about-aws-lambda-cold-starts-all-wrong/" target="_blank">theburningmonk.com</a> บอกว่า ควรหลีกเลี่ยง ถ้าไม่จำเป็น เพราะการใช้ VPC มันจะกำหนดให้ Lambda ต้องสร้าง ENIs ขึ้นมา (Elastic Network Interface) เพื่อใช้คุยกันภายใน VPC และในเว็บไซต์บอกว่า กระบวนการนี้มันจะทำให้เพิ่มเวลา Cold Start อีกประมาณ  10 วินาที!!</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1632" height="864" src="https://myifew.com/wp-content/uploads/2019/05/vpc-novpc-cold-start.png" alt="" class="wp-image-5406"/><figcaption>เทียบเวลาการเปิดใช้งาน AWS Lambda ตัวเดียวกัน แบบไม่มี VPC และมี VPC (รูปอ้างอิงจาก <a href="https://mikhail.io/serverless/coldstarts/aws/" target="_blank" rel="noreferrer noopener" aria-label="mikhail.io (opens in a new tab)">mikhail.io</a>)</figcaption></figure>



<p>และในเว็บไซต์ <a href="https://epsagon.com/blog/aws-lambda-programming-language-comparison/">epsagon.com</a> ได้ให้ข้อมูลเพิ่มเติมว่า การกำหนด Memory ที่มากขึ้น ใน AWS Lambda ของภาษา C# , Java มันจะทำการสร้าง ENIs เพิ่มจำนวนขึ้นด้วย แม้ว่า AWS Lambda จะมีกระบวนการที่นำ ENIs เก่ามาใช้ใหม่ได้ก็ตาม โดยสังเกตได้จากสูตรที่ชี้แจงไว้ใน AWS Document</p>



<p>ดังนั้น สรุปได้คือ ถ้าเราใช้ VPS และยิ่งเพิ่ม Memory Allocate เข้าไปในตัว AWS Lambda มันก็จะยิ่งเพิ่ม ENIs ให้เหมาะสมกัน และจะส่งผลกับ Cold Start ที่ใช้เวลามากขึ้นตามไปด้วย</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1586" height="828" src="https://myifew.com/wp-content/uploads/2019/05/memory-related-enis.png" alt="" class="wp-image-5404"/><figcaption>การเพิ่ม Memory เพื่อรองรับ Capacity ที่มากขึ้น เท่ากับ  AWS Lambda จะต้องเพิ่มจำนวนการสร้าง ENIs ตามไปด้วย (รูปอ้างอิงจาก <a href="https://docs.aws.amazon.com/lambda/latest/dg/vpc.html">https://docs.aws.amazon.com/lambda/latest/dg/vpc.html</a>)<br></figcaption></figure>



<p>ปล. มีข่าวดีแว่วๆมา อ้างอิงจาก <a href="https://twitter.com/jeremy_daly/status/1068272580556087296">@jeremy_daly</a> ที่เข้าร่วมงาน AWS re:Invent ว่า AWS มีแผนจะแก้ปัญหาเรื่องนี้ภายในปี 2019 นี่แหละครับ</p>



<h2 class="wp-block-heading">9. เลือกภาษาเพื่อใช้ทำ Serverless ที่เหมาะสม</h2>



<p>ข้อสุดท้าย ถ้าไม่ผูกติดกับภาษาในการพัฒนา Serverless แนะนำให้ใช้ตระกูล Scripting Language แทน Compiled language จะช่วยแก้ปัญหา Cold Start ได้อย่างมาก</p>



<p>ลองดูจากสถิติของเว็บไซต์ <a rel="noreferrer noopener" href="https://mikhail.io/serverless/coldstarts/aws/" target="_blank">mikhail.io</a></p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1624" height="864" src="https://myifew.com/wp-content/uploads/2019/05/cold-start-per-language.png" alt="" class="wp-image-5408"/><figcaption>เทียบเวลาการเกิด Cold Start ในแต่ละภาษา (รูปอ้างอิงจาก <a rel="noreferrer noopener" href="https://mikhail.io/serverless/coldstarts/aws/" target="_blank">mikhail.io</a>)</figcaption></figure>



<p>พวก JavaScript, Python, Go, Java, และ Ruby จะทำงานเสร็จแถวๆ 500ms หรือไม่เกิน 800ms แต่สำหรับ C# จะใช้เวลาตั้งแต่ 0.8 จนถึง 5 วินาที เลยทีเดียว</p>



<p>ตัวอย่างสถิติอีกสักแห่ง จากเว็บไซต์ <a href="https://read.acloud.guru/does-coding-language-memory-or-package-size-affect-cold-starts-of-aws-lambda-a15e26d12c76">read.acloud.guru</a></p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1796" height="1068" src="https://myifew.com/wp-content/uploads/2019/05/compare-cold-start-by-language.png" alt="" class="wp-image-5412"/><figcaption>เวลาเกิด Cold Start โดยเปรียบเทียบภาษาที่ใช้กับการจัดสรร Memory (รูปอ้างอิงจาก <a href="https://read.acloud.guru/does-coding-language-memory-or-package-size-affect-cold-starts-of-aws-lambda-a15e26d12c76">read.acloud.guru</a>)</figcaption></figure>



<p>สถิติจากเว็บไซต์นี้ ได้ผลใกล้เคียงกับที่แรกคือ พวก JavaScript, Python ยังคงทำงานได้เร็วแถวๆ 500ms ส่วน Java และ C# ใช้เวลา Cold Start นานมาก</p>



<p>และให้ข้อมูลเพิ่มเติมนิดนึงครับว่า แม้จะเปลี่ยนผู้ให้บริการ ก็ได้ผลไม่ต่างกัน ถ้าหากใครคิดว่า ใช้ C# บน Azure Function น่าจะเร็วกว่า AWS Lambda นั้น คิดผิดนะคร้าบบบ..</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1209" height="837" src="https://myifew.com/wp-content/uploads/2019/05/coldstart-per-language-vendor.png" alt="" class="wp-image-5413"/><figcaption>เทียบเวลาการเกิด Cold Start ในแต่ละภาษา และแต่ละผู้ให้บริการ (รูปอ้างอิงจาก <a rel="noreferrer noopener" href="https://mikhail.io/serverless/coldstarts/aws/" target="_blank">mikhail.io</a>)</figcaption></figure>



<p>ให้สังเกตว่า C# บน Azure Function ทำงานได้ช้ากว่า AWS Lambda ประมาณ 500ms &#8211; 1 วินาที เลยทีเดียว</p>



<p>อ่านข้อมูลการเปรียบเทียบเพิ่มเติมได้ที่นี่</p>



<ul class="wp-block-list"><li><a rel="noreferrer noopener" aria-label="Comparing AWS Lambda performance of Node.js, Python, Java, C# and Go (opens in a new tab)" href="https://read.acloud.guru/comparing-aws-lambda-performance-of-node-js-python-java-c-and-go-29c1163c2581" target="_blank">Comparing AWS Lambda performance of Node.js, Python, Java, C# and Go</a></li><li><a href="https://read.acloud.guru/does-coding-language-memory-or-package-size-affect-cold-starts-of-aws-lambda-a15e26d12c76" target="_blank" rel="noreferrer noopener" aria-label="How does language, memory and package size affect cold starts of AWS Lambda? (opens in a new tab)">How does language, memory and package size affect cold starts of AWS Lambda?</a></li></ul>



<h2 class="wp-block-heading">สรุป</h2>



<p>ตอนนี้เราก็คงต้องต่อสู่กับ Cold Start กันต่อไป เพราะเราไม่สามารถแก้ไขอะไรมันได้มากนัก ส่วนผู้ให้บริการเองอย่าง Amazon, Azure, Google Cloud ก็รู้ปัญหานี้เช่นกัน และพยายามปรับปรุงมันให้ดีขึ้นอยู่เรื่อยๆ แต่ที่สำคัญ ก่อนที่เราจะตัดสินใจใช้ Serverless เราต้องคิดให้ดีก่อนว่าเหมาะสมกับงานของเราแล้วหรือยัง</p>



<p>&#8212;</p>



<p>ที่มาภาพหน้าปก <a href="http://www.drunktiki.com/2016/08/17/yoda-pug/yoda-pug-jpg/">http://www.drunktiki.com/2016/08/17/yoda-pug/yoda-pug-jpg/</a></p>
]]></content:encoded>
					
					<wfw:commentRss>https://myifew.com/5386/9-ways-to-fighting-with-cold-start-in-serverless/feed/</wfw:commentRss>
			<slash:comments>0</slash:comments>
		
		
			</item>
		<item>
		<title>จัดสรรทรัพยากร AWS Lambda อย่างไร ให้ราคาถูกและเร็วที่สุด</title>
		<link>https://myifew.com/5339/how-to-optimize-memory-on-aws-lambda-for-costs-and-fastest/</link>
					<comments>https://myifew.com/5339/how-to-optimize-memory-on-aws-lambda-for-costs-and-fastest/#respond</comments>
		
		<dc:creator><![CDATA[iFew]]></dc:creator>
		<pubDate>Thu, 11 Apr 2019 12:08:16 +0000</pubDate>
				<category><![CDATA[Technology]]></category>
		<category><![CDATA[AWS]]></category>
		<category><![CDATA[AWS Lambda]]></category>
		<category><![CDATA[Microservices]]></category>
		<category><![CDATA[Performance Test]]></category>
		<category><![CDATA[Serverless]]></category>
		<category><![CDATA[Testing Tools]]></category>
		<guid isPermaLink="false">https://myifew.com/?p=5339</guid>

					<description><![CDATA[เป็นคำถามที่ทุกคนอยากรู้ ว่าจะตั้งค่า Memory ใน AWS Lambda Function อย่างไรให้ราคาถูกสุด แต่ได้ความเร็วที่สุด แบบไม่ใช้ทรัพยากรเว่อร์เกินไป]]></description>
										<content:encoded><![CDATA[
<p>มาจัดเรื่อง AWS Lambda อีกสักโพสต์ (ก่อนที่ผมจะสลับไปเขียนเรื่องท่องเที่ยวบ้าง ฮ่าๆ) ซึ่งครั้งนี้ขอเขียนถึงวิธีการเลือกใช้ทรัพยากร อย่างเช่น Memory ให้เหมาะสมที่สุด ทำงานได้เร็วด้วย และต้องคุ้มค่ากับการจ่ายเงินด้วย </p>



<span id="more-5339"></span>



<p>ใน Slide ของ <a rel="noreferrer noopener" aria-label="Become a Serverless Black Belt (opens in a new tab)" href="https://www.slideshare.net/AmazonWebServices/become-a-serverless-black-belt-optimizing-your-serverless-applications-aws-online-tech-talks" target="_blank">Become a Serverless Black Belt</a> ได้พูดถึงเรื่องการจัดสรรทรัพยากรไว้หลายหน้า และในหน้าปิดท้ายของตอน ได้บอกไว้ว่า </p>



<blockquote class="wp-block-quote is-layout-flow wp-block-quote-is-layout-flow"><p>Dont&#8217;s Guestimate</p></blockquote>



<p>หมายถึง เรื่องพวกนี้ &#8220;อย่าไปเดา&#8221; นะหนู เพราะเราไม่สามารถรู้ได้เลยว่า โค้ดที่เราเขียน ใช้ Memory เท่าไร ดังนั้นจะต้องจัดสรรให้มันเท่าไรถึงจะเพียงพอ ซึ่งถ้าจัดให้มากไป ก็เสียเงินจับจองเมมโมรี่ (Allocated Memory) โดยเปล่าประโยชน์, และถ้าจัดให้น้อยไป ก็ใช้เวลาคำนวณนานขึ้น (Computed Time) ไม่ว่าจะทางใด ก็มีผลกับค่าใช้จ่ายที่เกิดขึ้นทั้งคู่.. ความสนุก Serverless มันอยู่ตรงนี้แหละ!!</p>



<h2 class="wp-block-heading">เลิกเดา แล้วมาคุยกันด้วยข้อมูลดีกว่า</h2>



<p>ใน Slide เรื่องเดียวกัน ได้อ้างถึงเครื่องมือตัวหนึ่ง ชื่อ <a rel="noreferrer noopener" aria-label="AWS Lambda Power Tuning (opens in a new tab)" href="https://github.com/alexcasalboni/aws-lambda-power-tuning" target="_blank">AWS Lambda Power Tuning</a> ที่พัฒนาโดย Alex Casalboni และเขาได้เล่าที่มาในบล็อก <a rel="noreferrer noopener" aria-label="AWS Lambda Power Tuning with AWS Step Functions (opens in a new tab)" href="https://serverless.com/blog/aws-lambda-power-tuning/" target="_blank">AWS Lambda Power Tuning with AWS Step Functions</a> ว่า Serverless Developer ทุกคนมักหลับหูหลับตาเพื่อจัดสรรทรัพยากร หรือไม่ก็ใช้วิธีการ Manual Test เพื่อทดสอบดูทีละแบบ ซึ่งมันก็เสียเวลาและไม่ได้ผลลัพธ์ที่ดีแน่นอน</p>



<p>พี่แกเลยบอกว่า ถ้าเช่นนั้นเรามาใช้วิธี Data-Driven ดีกว่า! แกเลยเริ่มจากทำ<a rel="noreferrer noopener" aria-label="โพลสำรวจใน Twitter (opens in a new tab)" href="https://twitter.com/alex_casalboni/status/854059283465383937" target="_blank">โพลสำรวจใน Twitter</a> ว่าสิ่งที่สังเกตเห็นมันเป็นจริงไหม</p>



<figure class="wp-block-embed-twitter wp-block-embed is-type-rich is-provider-twitter"><div class="wp-block-embed__wrapper">
<blockquote class="twitter-tweet" data-width="550" data-dnt="true"><p lang="en" dir="ltr">How do you choose the optimal RAM configuration for your AWS Lambda Functions? I&#39;ll publish results &amp; my OSS project in a week! <a href="https://twitter.com/hashtag/serverless?src=hash&amp;ref_src=twsrc%5Etfw">#serverless</a></p>&mdash; Alex Casalboni (@alex_casalboni) <a href="https://twitter.com/alex_casalboni/status/854059283465383937?ref_src=twsrc%5Etfw">April 17, 2017</a></blockquote><script async src="https://platform.twitter.com/widgets.js" charset="utf-8"></script>
</div></figure>



<p>ซึ่งกว่า 50% ในโพล เคยใช้ Memory เท่าไรก็ตั้งไว้แบบเดิมเท่านั้น ไม่ได้สนใจที่จะ Tuning อะไร และครึ่งหนึ่งนิยมตั้งค่าไว้ที่ 128MB ทุก Function ด้วยเหตุผลที่ว่า เพราะมันราคาถูกที่สุดน่ะสิ! ไม่ได้สนใจเลยว่าจะทำงานได้ไวไหม</p>



<p>แต่ไม่ว่าจะเลือกตอบข้อไหน ก็ยังไม่ใช่สิ่งที่ควรทำในการตั้งค่า Memory ให้กับ Serverless อย่าง AWS Lambda</p>



<p>จากนั้นเขาก็เขียนเล่าถึงเหตุผลว่าทำไม Function ทำงานได้ช้า และคอนเซปของการแก้ปัญหาของ (ซึ่งถ้าใครอยากอ่านฉบับเต็ม ไปอ่านได้ที่ <a rel="noreferrer noopener" href="https://serverless.com/blog/aws-lambda-power-tuning/" target="_blank">AWS Lambda Power Tuning with AWS Step Functions) </a>จนสรุปได้ว่า การ Tuning Memory ทีละลำดับ พร้อมกับทดสอบเรียกใช้ (Invocating) ด้วยจำนวนที่ต้องการ ​กับ Lambda Function ทีละตัว คือหนทางที่เหมาะสม และต้องเป็นการทดสอบแบบ Automate ทั้งหมดด้วย เพื่อความรวดเร็ว  จึงเป็นที่มาของเครื่องมือ AWS Lambda Power Tuning ที่ผมกำลังจะเขียนถึง</p>



<h2 class="wp-block-heading">รู้จักกับ AWS Lambda Power Tuning</h2>



<p>เจ้าเครื่องมือ AWS Lambda Power Tuning นี้ ถูกพัฒนาด้วย Node.js โดยจะมี Lambda Function จำนวน 4 ตัว และเราต้องสร้าง Role กับ StepFunction เพื่อใช้มัน</p>



<p>แต่ไม่ต้องตกใจ ว่าจะยุ่งยาก เพราะโค้ดบน Github นาย Alex ได้มีไฟล์ Serverless.yml หรือเป็น AWS CloudFormation Script ไว้ให้เราใช้ Provision Infrastructure เรียบร้อย แต่เราจะต้องรันคำสั่งด้วยเครื่องมือชื่อ Serverless</p>



<p>การทำงานของมันคือ มันจะไปสร้าง Version ใน Lambda Function และกำหนด Memory ทีละช่วงที่เราต้องการ และทดสอบทีละ Version โดยจะหาค่าที่เหมาะสมที่สุดว่า ต้องใช้ Memory เท่าไร ที่สามารถ Compute ได้เร็วสุด และจ่ายเงินถูกที่สุด </p>



<p>ถ้าทำเอง น่าจะใช้เวลามากๆ เลยทีเดียว กว่าจะทดสอบ Lambda ได้ทีละตัว</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="648" height="956" src="https://myifew.com/wp-content/uploads/2019/04/lambda-power-tunning-working.png" alt="" class="wp-image-5351"/></figure>



<h2 class="wp-block-heading" id="mce_25">ตัวอย่างผลลัพธ์ของ AWS Lambda Power Tuning</h2>



<p>เช่น ผมมี Lambda Function เพื่อแสดงรายการสินค้า 1 ตัว ผมต้องการรู้ว่า Memory ในช่วง 128 MB &#8211; 1024 MB จำนวนเท่าไรจะเหมาะสมที่สุด..</p>



<p>ตัว AWS Lambda Power Tuning ก็จะเรียก Function ของมันมาเรียงเป็น Step ทำงานให้เราดูเป็นกราฟ</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="2142" height="1234" src="https://myifew.com/wp-content/uploads/2019/04/example-lambda-power-tunning.png" alt="" class="wp-image-5349"/></figure>



<p>เมื่อทำการทดสอบเสร็จ จะสรุปที่ Output ด้านขวามือ เพื่อบอกว่า ควรใช้ Power (Memory) เท่าไร ถึงจะคุ้มค่าที่สุด ด้วย Cost และ Duration Time เท่าไร</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="2642" height="1250" src="https://myifew.com/wp-content/uploads/2019/04/example-lambda-power-tunning-monitoring-execution-1.png" alt="" class="wp-image-5361"/></figure>



<p>ถ้าใครอยากเห็นรายละเอียดการทดสอบแต่ละ Memory ให้เลื่อนลงมาดูด้านล่าง จะมี Output บอกอีกส่วนหนึ่งแบบละเอียด</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="914" height="1158" src="https://myifew.com/wp-content/uploads/2019/04/example-lambda-power-tunning-logs-detail.png" alt="" class="wp-image-5369"/></figure>



<p>คราวนี้ ทดสอบครั้งแรก ผมไม่ค่อยเชื่อผลเท่าไร จึงลองทดสอบทั้งหมด 4 รอบ, แต่ละรอบและแต่ละ Memory ผมจำลองให้มีการร้องขอ (Invocation) จำนวน 10 ครั้ง จำนวน 3 รอบ และ 100 ครั้ง จำนวน 1 รอบ ผลที่ได้ออกมา คือ</p>



<p><strong>Execution #1 &#8211; Invocation 10</strong></p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="898" height="404" src="https://myifew.com/wp-content/uploads/2019/04/example-compare-lambda-power-tunning-list1.png" alt="" class="wp-image-5346"/></figure>



<p><strong>Execution #2 &#8211; Invocation 10</strong></p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="862" height="408" src="https://myifew.com/wp-content/uploads/2019/04/example-compare-lambda-power-tunning-list3.png" alt="" class="wp-image-5348"/></figure>



<p><strong>Execution #3 &#8211; Invocation 10</strong></p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="860" height="408" src="https://myifew.com/wp-content/uploads/2019/04/example-compare-lambda-power-tunning-list2.png" alt="" class="wp-image-5347"/></figure>



<p><strong>Execution #4 &#8211; Invocation 100</strong></p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="902" height="406" src="https://myifew.com/wp-content/uploads/2019/04/example-compare-lambda-power-tunning-list4.png" alt="" class="wp-image-5350"/></figure>



<p>จากผลการทดสอบ แปลว่า Function แสดงรายการสินค้า ถ้าใช้ Memory ที่ 320 MB จะเร็วและถูกที่สุด โดยเราไม่จำเป็นต้องใช้ Memory ถึง 1024 MB ก็ได้ เพราะผลที่ได้อาจจะไม่ต่างกันมาก ไม่ว่าจะมีการเรียกใช้ 10 ครั้งหรือ 100 ครั้ง ต่อเนื่องก็ตาม </p>



<p>(อย่าลืมว่า ค่าใช้จ่ายการ Compute จะปัดเศษขึ้น เป็นหน่วย 100ms นะครับ ไม่ว่าจะ 40.2ms , 80.8ms ก็ถูกปัดไปคิดเงินเป็น 100ms เท่ากัน)</p>



<p>ปล. จุดสังเกต คือ Memory 256 MB จะทำงานได้ช้ากว่า และแพงกว่า ทั้ง 4 รอบเลย ตรงนี้ผมต้องไปตรวจสอบในระดับ Source Code, Logs, หรือใช้ AWS X-Ray ดูต่อว่าทำไม</p>



<h2 class="wp-block-heading" id="mce_57">วิธีการติดตั้ง AWS Lambda Power Tuning ผ่าน Serverless</h2>



<p class="has-background has-very-light-gray-background-color">** หากใครไม่สะดวกติดตั้งผ่าน Serverless สามารถทำด้วยการ Clone git แล้วรันสคริ้ป ซึ่งสามารถดูได้ที่ <a href="https://github.com/alexcasalboni/aws-lambda-power-tuning">https://github.com/alexcasalboni/aws-lambda-power-tuning</a> **</p>



<p>ขายของมานาน มาดูวิธีติดตั้งและใช้งานกัน ซึ่งในที่นี้ผมยกตัวอย่างของ MacOS, Linux  ส่วนชาว Windows ให้ใช้ PowerShell ดูนะครับ</p>



<h4 class="wp-block-heading">ความต้องการก่อนติดตั้งใช้งาน</h4>



<ol class="wp-block-list"><li>AWS Account จะต้องมีสิทธิ์ใช้งาน ดังนี้<ul><li>cloudformation:*</li><li>states:*</li><li>lambda:*</li><li>iam:CreateRole</li></ul></li><li>เครื่องที่ติดตั้งจะต้องมี npm (<a rel="noreferrer noopener" aria-label="วิธีติดตั้ง (opens in a new tab)" href="https://docs.npmjs.com/downloading-and-installing-node-js-and-npm" target="_blank">วิธีติดตั้ง npm</a>)</li></ol>



<h4 class="wp-block-heading">ติดตั้ง Serverless Framework และใส่ AWS credentials</h4>



<pre class="wp-block-code"><code>$ npm install serverless -g
$ serverless config credentials --provider aws --key XXX --secret YYY</code></pre>



<h4 class="wp-block-heading">ติดตั้ง AWS Lambda Power Tuning</h4>



<pre class="wp-block-code"><code>$ serverless install -u https://github.com/alexcasalboni/aws-lambda-power-tuning</code></pre>



<h4 class="wp-block-heading">ทำการติดตั้ง Package Dependency</h4>



<pre class="wp-block-code"><code>$ cd aws-lambda-power-tuning
$ npm install</code></pre>



<h4 class="wp-block-heading">สร้าง Config ของ State Machine ในไฟล์ serverless.yml</h4>



<pre class="wp-block-code"><code>$ npm run generate -- -A ACCOUNT_ID [-R eu-west-1] [-P 128,256,512,1024]

เช่น

$ npm run generate -- -A 182000046000 -R ap-southeast-1 -P 128,192,256,320,448,512,704,1024</code></pre>



<h4 class="wp-block-heading">ทำการ Deploy AWS Lambda Power Tuning</h4>



<pre class="wp-block-code"><code>$ serverless deploy</code></pre>



<h2 class="wp-block-heading" id="mce_66">วิธีการใช้งาน AWS Lambda Power Tuning</h2>



<p>เมื่อติดตั้งเสร็จแล้ว ลองเช็คที่ AWS Lambda Function เรา จะเจอฟังก์ชั่นเพิ่มขึ้นมา ดังนี้</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="2582" height="340" src="https://myifew.com/wp-content/uploads/2019/04/lambda-power-tunning-lambda-function.png" alt="" class="wp-image-5354"/></figure>



<p>จากนั้นให้ไปดูที่เซอร์วิส <a rel="noreferrer noopener" aria-label="Step Function (opens in a new tab)" href="https://console.aws.amazon.com/states/home" target="_blank">Step Function</a> จะพบ State Machine</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="2790" height="714" src="https://myifew.com/wp-content/uploads/2019/04/lambda-power-tuning-state-machine.png" alt="" class="wp-image-5355"/></figure>



<p>ให้กดที่ Start Excution</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="2130" height="968" src="https://myifew.com/wp-content/uploads/2019/04/lambda-power-tuning-state-machine-execute.png" alt="" class="wp-image-5356"/></figure>



<p>กรอกข้อมูล State Machine Input สำหรับจะใช้ทำงาน โดยมีรูปแบบตามนี้</p>



<pre class="wp-block-code"><code>{
    "lambdaARN": "your-lambda-function-arn",
    "num": 10
}</code></pre>



<p>เช่น</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="2502" height="870" src="https://myifew.com/wp-content/uploads/2019/04/lambda-power-tuning-state-machine-execute-input-1.png" alt="" class="wp-image-5358"/></figure>



<p>รอสักครู่ จะมี Status บอกเราว่า Successed พร้อมบอกรายละเอียดการทำงาน และผลลัพธ์ ซึ่งถ้ามันขึ้นว่า Failed ให้เราไปดูที่ Exception ได้ </p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="2642" height="1250" src="https://myifew.com/wp-content/uploads/2019/04/example-lambda-power-tunning-monitoring-execution-1.png" alt="" class="wp-image-5361"/></figure>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="2606" height="1256" src="https://myifew.com/wp-content/uploads/2019/04/example-lambda-power-tunning-monitoring-execution-detail.png" alt="" class="wp-image-5360"/></figure>



<h2 class="wp-block-heading">การกำหนดค่า State Machine Input</h2>



<p>เราสามารถกำหนด parameters ของ input ได้ ดังนี้:</p>



<ul class="wp-block-list"><li><strong>lambdaARN</strong> (จำเป็น): ARN ของ Lambda Function ที่ต้องการทดสอบ</li><li><strong>num</strong> (จำเป็น): หมายเลขจำนวนที่ต้องการทดสอบเรียกใช้งาน (invocations) ต่อครั้งและต่อ Memory ที่เรากำหนด (อย่างน้อย 5, แนะนำ: 10 &#8211; 100)</li><li><strong>payload</strong> (string/object): Input สำหรับใช้ใน Lambda Function ที่ต้องการทดสอบ (ใส่เป็นข้อมูล JSON)</li><li><strong>parallelInvocation</strong> (ค่าตั้งต้น: false): ถ้าเราใส่เป็น true, จะมีการเรียก Execute พร้อมๆ กัน ตามจำนวนที่เราระบุใน num</li></ul>



<h2 class="wp-block-heading">สรุป</h2>



<p>ไม่ยากเลยใช่ไหมครับ สามารถเอาไปใช้ตรวจสอบ AWS Lambda Fucntion ของเราเองได้อย่างรวดเร็วและสะดวก ไม่ต้องทำมือเองท่ีละตัว หรือใช้วิธีการเดา เพื่อความรวดเร็ว และคุ้มค่าที่สุด</p>



<p><strong>อ้างอิง</strong></p>



<ul class="wp-block-list"><li><a href="https://serverless.com/blog/aws-lambda-power-tuning/">https://serverless.com/blog/aws-lambda-power-tuning/</a></li><li><a rel="noreferrer noopener" aria-label="https://github.com/alexcasalboni/aws-lambda-power-tuning (opens in a new tab)" href="https://github.com/alexcasalboni/aws-lambda-power-tuning" target="_blank">https://github.com/alexcasalboni/aws-lambda-power-tuning</a></li><li><a rel="noreferrer noopener" aria-label="https://www.slideshare.net/AmazonWebServices/become-a-serverless-black-belt-optimizing-your-serverless-applications-aws-online-tech-talks (opens in a new tab)" href="https://www.slideshare.net/AmazonWebServices/become-a-serverless-black-belt-optimizing-your-serverless-applications-aws-online-tech-talks" target="_blank">https://www.slideshare.net/AmazonWebServices/become-a-serverless-black-belt-optimizing-your-serverless-applications-aws-online-tech-talks</a></li></ul>
]]></content:encoded>
					
					<wfw:commentRss>https://myifew.com/5339/how-to-optimize-memory-on-aws-lambda-for-costs-and-fastest/feed/</wfw:commentRss>
			<slash:comments>0</slash:comments>
		
		
			</item>
		<item>
		<title>ลองทำ Serverless WebSocket ด้วย AWS Lambda และ AWS API Gateway</title>
		<link>https://myifew.com/5321/real-time-applications-with-serverless-websockets-by-aws-lambda-and-api-gateway/</link>
					<comments>https://myifew.com/5321/real-time-applications-with-serverless-websockets-by-aws-lambda-and-api-gateway/#respond</comments>
		
		<dc:creator><![CDATA[iFew]]></dc:creator>
		<pubDate>Wed, 27 Mar 2019 03:20:18 +0000</pubDate>
				<category><![CDATA[Technology]]></category>
		<category><![CDATA[API Gateway]]></category>
		<category><![CDATA[AWS]]></category>
		<category><![CDATA[AWS Lambda]]></category>
		<category><![CDATA[Microservices]]></category>
		<category><![CDATA[Serverless]]></category>
		<category><![CDATA[Websockets]]></category>
		<guid isPermaLink="false">https://myifew.com/?p=5321</guid>

					<description><![CDATA[แนวคิดการทำ WebSocket ด้วย Serverless ที่มันดูค่อนข้างขัดแย้งกัน แต่มันก็ทำได้นะ ด้วย AWS Lambda และ AWS API Gateway]]></description>
										<content:encoded><![CDATA[
<p>อยากอัพเดทข้อมูลแบบ Real-time, แสดงข้อความ Notification ทันทีเมื่อมีการแจ้งข่าวให้สมาชิก, ทำระบบแชทที่พูดคุยกันได้ตลอดเวลา ฯลฯ ระบบแบบนี้เรามักทำเป็น WebSocket ซึ่งเป็น Protocol หนึ่งที่จะเปิดท่อ Connection ไว้เพื่อให้สื่อสารระหว่าง Server  และ Client ได้ตลอดเวลา ซึ่งรายละเอียดมากกว่านี้ หรือทำอย่างไร ลองไปหาจาก Google ได้เยอะแยะ</p>



<p>แต่ที่ชวนฉงนคือ การทำงานของ Serverless เป็นแบบ เกิดขึ้น และทำ Process แล้วก็ดับไป มันจะทำเป็น WebSocket  ได้ด้วยหรอ เพราะต้องเปิดแช่ Connection เพื่อสื่อสารกันตลอดเวลา </p>



<p>โพสต์นี้จะแชร์แนวคิดให้อ่านกัน ว่ามันเป็นไปได้จริงๆ ด้วยบริการของ Amazon API Gateway และ AWS Lambda ส่วนโค้ดตัวอย่าง ไปดูได้ที่ Github (<a href="https://github.com/ifew/aws-lambda-websocket">aws-lambda-websocket</a>) ผมได้เลย</p>



<span id="more-5321"></span>



<h2 class="wp-block-heading">อธิบายการทำงาน</h2>



<p>การทำ Websocket ด้วย Lambda มีจุดสำคัญอยู่ 2 จุด คือ </p>



<h4 class="wp-block-heading"><strong>1.Amazon API Gateway</strong> </h4>



<p>ซึ่งทาง Amazon เอง เพิ่งประกาศฟีเจอร์ WebSocket นี้เมื่อ 3 เดือนที่แล้วนี่เอง (<a rel="noreferrer noopener" aria-label="18 ธันวาคม 2561 (opens in a new tab)" href="https://aws.amazon.com/th/blogs/compute/announcing-websocket-apis-in-amazon-api-gateway/" target="_blank">18 ธันวาคม 2561</a>) โดยสามารถใช้งานกับบริการต่างๆของ Amazon ได้ เช่น AWS Lambda, Amazon Kinesis, หรืออื่นๆที่เชื่อมต่อด้วย HTTP endpoint </p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="928" height="571" src="https://myifew.com/wp-content/uploads/2019/03/api-gw-create-websockets.png" alt="" class="wp-image-5323"/><figcaption>หน้าตาของ Amazon API Gateway ตอนที่จะสร้าง WebSocket API</figcaption></figure>



<p>ซึ่งใน Amazon API Gateway ส่วนของ Websocket API จะมีการ Config Resource Type ต่างจากแบบ REST คือ </p>



<ol class="wp-block-list"><li><strong>route</strong> ที่เอาไว้เป็นช่องทางจัดการคำขอต่างๆที่มาจากผู้เรียกใช้ (route selection expression)</li><li><strong>routeKey</strong> ที่เป็นช่องทางของคำขอ โดยแบ่งได้ 3 routeKey หลัก คือ<ul><li><strong>$default</strong> &#8211; ใช้สำหรับคำขอที่ไม่เข้าเงื่อนไขใดๆเลย จากผู้รองขอ เช่น การส่งข้อมูล Error ให้กับผู้ร้องขอ</li><li><strong>$connect</strong> &#8211; ใช้สำหรับเมื่อมีผู้ร้องขอเพื่อขอใช้งาน และตรวจพบว่าเป็นการเรียกใช้ Websockets นี้ครั้งแรก (การที่เคยขอใช้ และ Connection ของ WebSocket หมดอายุไป ก็ต้องเรียก Connect ใหม่เช่นกัน)</li><li><strong>$disconnect</strong> &#8211; ใช้สำหรับบอกว่าผู้ใช้งานมีการ Disconnect จาก API ไปแล้ว ก็จะทำการตัด Connection ของ WebSocket ออกไปด้วย</li></ul></li></ol>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1219" height="636" src="https://myifew.com/wp-content/uploads/2019/03/api-gw-integrate-lambda.png" alt="" class="wp-image-5326"/></figure>



<h4 class="wp-block-heading"><strong>2.การเก็บ Connection ID ลง ฐานข้อมูล </strong></h4>



<p>โดยปกติถ้าเราเขียน WebSocket ทั่วไป มันจะทำการเก็บ Connection ID ไว้ใน Server แต่เมื่อเราทำบน Serverless อย่าง AWS Lambda ที่ธรรรมชาติของมันพร้อมจะปิดสวิตส์ตัวเองเสมอเมือ่ไม่ได้ใช้งานนานๆ เราจึงต้องเพิ่มการเก็บ Connection ID ลงในฐานข้อมูลแทน เพื่ออ้างอิงไว้ใช้ เมื่อ Lambda เราตื่นขึ้นมาทำงานต่อ</p>



<h2 class="wp-block-heading">ตัวอย่างการทำงาน AWS Lambda กับ WebSocket </h2>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="747" height="429" src="https://myifew.com/wp-content/uploads/2019/03/websockets-arch.png" alt="" class="wp-image-5325"/><figcaption>รูปจาก https://aws.amazon.com/th/blogs/compute/announcing-websocket-apis-in-amazon-api-gateway/</figcaption></figure>



<p>ในรูปตัวอย่าง อธิบายการทำงานของระบบ Chat ที่ใช้ Lambda และ WebSocket API โดยมีขั้นตอนคือ</p>



<ul class="wp-block-list"><li>ผู้ใช้ เข้ามาใช้งาน Chat Room ซึ่งระบบก็จะทำการ connect ไปที่ WebSocket API </li><li>ระบบ WebSocket API จะทำการสร้าง Connection ID และส่ง message กลับไปหาผู้ร้องขอตาม URL ที่มีการเรียกใช้มาตอนแรก (URL Callback) เพื่อบอกว่า  ทำการเชื่อมต่อสำเร็จ (Connected) พร้อมทั้งเก็บ Connection ID ไว้ในฐานข้อมูลด้วย</li><li>เมื่อผู้ใช้ส่งข้อความ Chat ไปให้ WebSocket API มันจะส่ง message นั้นไปที่ Reosource ที่เรากำหนดไว้ (ซึ่งในที่นี้คือ Lambda Function สำหรับการ Chat) โดยไม่ต้อง connect ใหม่อีกรอบ</li><li>message ที่ถูกส่งมา จะถูกกระจายส่งไปหา Connection ทุกคนที่มี Connection ID อยู่ในรายการฐานข้อมูล</li><li>สุดท้าย เมื่อพบว่าผู้ใช้ออกจากห้อง Chat Room ไปแล้ว ตัว WebSocket API จะ Disconnect ผู้ใช้คนนั้นอัตโนมัติ</li></ul>



<p>ผมทดลองโดยเปิด Connection ไว้ สามหน้าต่าง ซึ่งจะมี Connection ID ต่างกัน ดังนั้น เมื่อผมส่ง message ออกจากหน้าต่างไหนก็ตาม มันจะถูกเผยแพร่ไปหาหน้าต่างอื่นๆ ด้วย ดังนั้น จึงเหมาะกับงานประเภททำงานแบบ Realtime และ Broadcast เช่น Chat, Notification, Bidding หรือแม้แต่ระบบโหวตและนับคะแนนเสียงเลือกตั้ง </p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="2716" height="1648" src="https://myifew.com/wp-content/uploads/2019/03/Screen-Shot-2562-03-27-at-09.48.57.png" alt="" class="wp-image-5331"/><figcaption>ตัวอย่างการทำงานเพื่อคุยข้าม Connection กัน<br></figcaption></figure>



<p>ตัวอย่างโค้ด ดาวน์โหลดได้ที่ <a href="https://github.com/ifew/aws-lambda-websocket">https://github.com/ifew/aws-lambda-websocket</a></p>



<h2 class="wp-block-heading">ค่าใช้จ่าย สำหรับ AWS Lambda WebSocket </h2>



<p>ค่าใช้จ่ายในส่วนของ AWS Lambda จะเหมือนเดิม ตามที่เคยเขียนเล่าไว้ในบล็อก <a href="https://myifew.com/5166/understand-serverless-with-aws-lambda-for-newbie/">มาทำความรู้จักกับ Serverless และ AWS Lambda ฉบับคนคิดจะใช้</a> </p>



<p>จะมีที่แตกต่างคือการคิดเงินในฝั่งของ Amazon API Gateway ที่คิดตาม request และ จำนวนนาทีที่มีการเชื่อมต่ออยู่ สามารถดูได้จากตารางด้านล่าง</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="881" height="511" src="https://myifew.com/wp-content/uploads/2019/03/websockets-pricing.png" alt="" class="wp-image-5327"/></figure>



<p>ถามว่าแพงไหม ผมก็ไม่รู้นะ อยู่ที่การใช้งานของแต่ละระบบ แต่จำนวน request 1พันล้านแรก ราคา $1.15 กับ connection 1 ล้านนาที ราคา $0.288 คงเป็นตัวเลขที่ถูกแสนถูก มากๆ เท่ากับข้าวแกงจานนึง </p>



<p>แต่ตัวแปรสำคัญคงอยู่ที่ Lambda Compute ที่อาจจะแพง ฮ่าๆ เพราะ การทำงาน 1 ชุด จะมี Lambda อย่างน้อย 3 ตัวแน่นอน คือ Connect, Disconnect, Default ส่วน Lambda จะมีค่าใช้จ่ายสูงแค่ไหน ก็กลับไปดูจำนวน request และ compute time (วิธีคิดอ่านได้จากบล็อก <a href="https://myifew.com/5166/understand-serverless-with-aws-lambda-for-newbie/">มาทำความรู้จักกับ Serverless และ AWS Lambda ฉบับคนคิดจะใช้</a> )</p>



<h2 class="wp-block-heading">ข้อจำกัด</h2>



<ul class="wp-block-list"><li>รองรับผู้ใช้งานได้ 10,000 requests ต่อวินาที (RPS) ต่อ region (สามารถขอเพิ่มจากทาง Amazon ได้)</li><li>ระบบจะทำการเพิ่มให้ 500 requests ในทุกๆวินาที ต่อ region</li><li>ส่งข้อมูลผ่าน WebSocket (frame size) ได้ไม่เกิน 32 KB ถ้าต้องใช้มากกว่านั้น ต้องตัดข้อมูลเป็นเฟรมๆเพื่อส่ง, ซึ่งถ้าส่งเกิน ระบบจะทำการ Disconnect อัตโนมัติ</li><li>ระยะเวลาสำหรับ Connection อยู่ได้ 2 ชั่วโมง</li><li>เข้าสู่โหมด Idle Connection Timeout ที่เวลา 10 นาที กรณีไม่มีการส่งข้อมูลอะไรผ่าน WebSocket API</li><li>สามารถทำ AWS Lambda authorizers ต่อ API ได้ 10 ตัว (สามารถขอเพิ่มจากทาง Amazon ได้)</li><li>สร้าง Routes ต่อ API ได้ไม่เกิน 300 ตัว (สามารถขอเพิ่มจากทาง Amazon ได้) </li><li>ทำการเชื่อมต่อ API กับ Resource ต่างๆ เช่น AWS Lambda (Integrations) ได้ไม่เกิน 300 ตัว (สามารถขอเพิ่มจากทาง Amazon ได้)</li><li>สามารถมี Stages ต่อ API ได้ไม่เกิน 10 ตัว (สามารถขอเพิ่มจากทาง Amazon ได้)</li></ul>



<h2 class="wp-block-heading">สรุป</h2>



<p>น่าสนใจมากๆ สำหรับการทำ WebSocket ด้วย Serverless ดูเป็นเรื่องน่าสนุกดีสำหรับผม  เหมือนได้เล่นของใหม่ ที่พยายามพลิกแพลง Serverless มาทำ แต่ถามว่าดีหรือไม่ เหมาะสมไหม ตอนนี้มันยังใหม่มาก หาเคสตัวอย่างใหญ่ๆ ไม่เจอ (หรืออาจจะมีแต่ไม่มีใครเปิดเผย) ดังนั้นถ้าให้บอกข้อดีที่นึกออกตอนนี้ ก็คงเรื่องของราคาค่า Server ที่มันถูกลง ตาม benefit เดิมของ Serverless นั่นแหละ, ส่วนข้อเสียที่ผมพบตอนนี้ รู้สึกทดสอบยาก ณ ตอนเขียนบล็อกนี้ยังนึกการทำ Integration Test ไม่ออกเลย นอกจากการ Deploy ขึ้นไปทดสอบบน AWS เอง, หากใครคิดออก หรือมีตัวอย่างเจ๋งๆ ก็แชร์กันได้นะครับ</p>



<p><strong>อ้างอิง</strong></p>



<ul class="wp-block-list"><li><a href="https://aws.amazon.com/th/blogs/compute/announcing-websocket-apis-in-amazon-api-gateway/">https://aws.amazon.com/th/blogs/compute/announcing-websocket-apis-in-amazon-api-gateway/</a></li><li>รูปปกจาก <a href="https://blog.stanko.io/do-you-really-need-websockets-343aed40aa9b?gi=77181225da63">https://blog.stanko.io/do-you-really-need-websockets-343aed40aa9b?gi=77181225da63</a></li></ul>
]]></content:encoded>
					
					<wfw:commentRss>https://myifew.com/5321/real-time-applications-with-serverless-websockets-by-aws-lambda-and-api-gateway/feed/</wfw:commentRss>
			<slash:comments>0</slash:comments>
		
		
			</item>
		<item>
		<title>มาทำความรู้จักกับ Serverless และ AWS Lambda ฉบับคนคิดจะใช้</title>
		<link>https://myifew.com/5166/understand-serverless-with-aws-lambda-for-newbie/</link>
					<comments>https://myifew.com/5166/understand-serverless-with-aws-lambda-for-newbie/#respond</comments>
		
		<dc:creator><![CDATA[iFew]]></dc:creator>
		<pubDate>Wed, 27 Feb 2019 11:56:43 +0000</pubDate>
				<category><![CDATA[Technology]]></category>
		<category><![CDATA[AWS]]></category>
		<category><![CDATA[AWS Lambda]]></category>
		<category><![CDATA[Microservices]]></category>
		<category><![CDATA[Serverless]]></category>
		<guid isPermaLink="false">https://myifew.com/?p=5166</guid>

					<description><![CDATA[อธิบาย ความเป็นมาของ Serverless และ AWS Lambda คืออะไร ดีอย่างไร เพื่อให้คนไม่รู้จักและคนที่ไม่ใช่ IT ได้เข้าใจง่ายๆ ]]></description>
										<content:encoded><![CDATA[
<p>ผมได้ยินกิตติศัพท์ <strong>AWS Lambda</strong> มานานมาก แต่ไม่เคยเข้าไปดูจริงจัง รู้แค่มันเป็น Service ที่มีไว้ทำ Process สักอย่าง เมื่อเสร็จแล้วก็ทำลายตัวเองทิ้งไป คิดเงินตามการเรียกใช้และระยะเวลาที่ใช้ทำงานเท่านั้น หรือที่เรียกกันว่า <strong>Serverless</strong> </p>



<p>ช่วงนี้ได้มีโอกาสเล่นมันแบบจริงๆจังๆ เลยรู้สึกว่า คนต้นคิดเรื่อง Serverless เจ๋งดี เลยอยากมาเขียนสรุปให้อ่านกันสำหรับคนที่ไม่รู้อะไรเลย ผ่านตัวอย่างบริการของ AWS Lambda</p>



<span id="more-5166"></span>



<h2 class="wp-block-heading">Serverless คืออะไร?</h2>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1200" height="675" src="https://myifew.com/wp-content/uploads/2019/02/evolution-of-serverless-1200x675.png" alt="" class="wp-image-5285" srcset="https://myifew.com/wp-content/uploads/2019/02/evolution-of-serverless-1200x675.png 1200w, https://myifew.com/wp-content/uploads/2019/02/evolution-of-serverless-1024x576.png 1024w, https://myifew.com/wp-content/uploads/2019/02/evolution-of-serverless-768x432.png 768w, https://myifew.com/wp-content/uploads/2019/02/evolution-of-serverless-700x394.png 700w, https://myifew.com/wp-content/uploads/2019/02/evolution-of-serverless-600x338.png 600w, https://myifew.com/wp-content/uploads/2019/02/evolution-of-serverless.png 1510w" sizes="auto, (max-width: 1200px) 100vw, 1200px" /><figcaption>ภาพจาก: <a href="https://www.youtube.com/watch?v=3vaqTIIcfvY">https://www.youtube.com/watch?v=3vaqTIIcfvY</a></figcaption></figure>



<p>Serverless ถ้าแปลตรงตัวก็คงเรียกว่า &#8220;ไม่มี Server&#8221; กล่าวคือ เมื่อมีการเรียกใช้งานอะไรสักอย่าง มันจะสร้าง Server ชั่วคราวขึ้นมาเพื่อประมวลผล และเมื่อส่งผลลัพธ์ให้เราเสร็จ มันก็จะหยุดทำงาน หรือทำลายตัวเองทิ้งไป </p>



<blockquote class="wp-block-quote is-layout-flow wp-block-quote-is-layout-flow"><p><em>ปล. จริงๆ มันก็มี Server นั่นเอง ผมอยากแปลมันว่า &#8220;Server ชั่วคราว&#8221; มากกว่า แต่เพื่อความเข้าใจตรงกันขอใช้คำแปลแบบท่านอื่นๆแล้วกัน เพราะส่วนตัวคิดว่า คำว่า &#8220;Serverless ทำงานแบบไร้ Server&#8221; มันเป็นข้อความทางการตลาด ให้คนสนใจ แต่ค่อนข้างสร้างความสับสนว่าเป็นไปได้ด้วยหรอ?!</em></p><cite>iFew</cite></blockquote>



<p>ถ้าไล่ประวัติตามรูปด้านบน เริ่มต้นมาจากสมัย Internet มาแรกๆ ที่เราต้องตั้งเครื่อง Server ทีละตัว หรือหลายๆเครื่องตามปริมาณที่รองรับคนได้ ต้องดูแลเองทุกอย่าง </p>



<p>จนถึงวันหนึ่งที่อุปกรณ์ต่างๆ เริ่มมีราคาถูกลงและสเปกแรงขึ้น เราจึงประหยัดต้นทุนด้วยการจำลอง Server หลายๆตัว ลงในเครื่อง Server ตัวเดียว หรือที่เรียกกันว่า VM (Virtual Machine) </p>



<p>และเมื่อ Internet เร็วและแรงขึ้น ก็ได้เกิดบริการที่เรียกว่า Cloud Computer ซึ่งใน Cloud เองก็ใช้สถาปัตยกรรมหลายรูปแบบแล้วแต่ผู้ให้บริการ จนมาถึงยุคปัจจุบัน ที่กำลังอยู่ในช่วงของความนิยมใช้ Container อย่างเข่น Docker, Kubernates และก็กำลังก้าวเข้าสู่ยุค Serverless ที่ผมกำลังจะเล่าถึง</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1200" height="911" src="https://myifew.com/wp-content/uploads/2019/02/infoq-trend-report-feb2019-1200x911.jpg" alt="" class="wp-image-5303" srcset="https://myifew.com/wp-content/uploads/2019/02/infoq-trend-report-feb2019-1200x911.jpg 1200w, https://myifew.com/wp-content/uploads/2019/02/infoq-trend-report-feb2019-1024x777.jpg 1024w, https://myifew.com/wp-content/uploads/2019/02/infoq-trend-report-feb2019-768x583.jpg 768w, https://myifew.com/wp-content/uploads/2019/02/infoq-trend-report-feb2019-700x531.jpg 700w, https://myifew.com/wp-content/uploads/2019/02/infoq-trend-report-feb2019-600x455.jpg 600w, https://myifew.com/wp-content/uploads/2019/02/infoq-trend-report-feb2019.jpg 1800w" sizes="auto, (max-width: 1200px) 100vw, 1200px" /><figcaption>ภาพจาก: https://www.infoq.com/articles/devops-cloud-trends-2019</figcaption></figure>



<p>สังเกตไหมครับว่า อะไรที่เปลี่ยนไป?&#8230;</p>



<p>สิ่งที่เปลี่ยนไปคือ สิ่งที่เราใช้ประมวลผลมันเล็กลงเรื่อยๆ จากเครื่อง Server ย่อยลงมาเหลือแค่ระดับ VM มาเป็น Container รวมถึงการจัดการเรื่องเหล่านี้ก็ถูกผลักไปให้ทางผู้ให้บริการดูแลมากขึ้น จากที่ต้องดูแลเรื่อง Hardware + Network + Middleware + Application  ก็เหลือเพียงแค่โฟกัสเรื่อง Application อย่างเดียว (ส่วน SaaS มันคือการไปใช้ระบบของคนอื่นเลย ไม่ต้องทำเอง ไม่ต้องดูแล)</p>



<p>ซึ่ง ผมไม่รู้ว่าใครเริ่มก่อนกัน ระหว่าง Hardware กับ Software แต่มันสอดคล้องกับแนวโน้มการพัฒนา Software ที่นิยมย่อยชิ้นส่วนให้เล็กลง แบบที่เราคุ้นชื่อกันตอนนี้ เช่น Microservice ที่เกิดมาเพื่อลดข้อผิดพลาดที่เป็นผลพวงจากการทำงานที่เกี่ยวข้องกัน รวมถึงจัดการทรัพยากรให้เหมาะสม และก็ลากไปเกี่ยวโยงอีกว่า เพื่อรองรับกับวิธีบริหารจัดการในแบบ Interative and Incremental อย่าง Scrum, DSDM, XP เป็นต้น</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1200" height="551" src="https://myifew.com/wp-content/uploads/2019/02/1_dAnZk19kGszKTvAgag31sQ-1200x551.jpeg" alt="" class="wp-image-5284" srcset="https://myifew.com/wp-content/uploads/2019/02/1_dAnZk19kGszKTvAgag31sQ-1200x551.jpeg 1200w, https://myifew.com/wp-content/uploads/2019/02/1_dAnZk19kGszKTvAgag31sQ-1024x470.jpeg 1024w, https://myifew.com/wp-content/uploads/2019/02/1_dAnZk19kGszKTvAgag31sQ-768x352.jpeg 768w, https://myifew.com/wp-content/uploads/2019/02/1_dAnZk19kGszKTvAgag31sQ-700x321.jpeg 700w, https://myifew.com/wp-content/uploads/2019/02/1_dAnZk19kGszKTvAgag31sQ-600x275.jpeg 600w, https://myifew.com/wp-content/uploads/2019/02/1_dAnZk19kGszKTvAgag31sQ.jpeg 1600w" sizes="auto, (max-width: 1200px) 100vw, 1200px" /><figcaption>ภาพจาก: <br><a href="https://medium.com/@sdorzak/why-serverless-is-the-new-black-e4ff9e9947e0">https://medium.com/@sdorzak/why-serverless-is-the-new-black-e4ff9e9947e0</a> <br></figcaption></figure>



<p>และเมื่อมันไม่มี Server ให้เราจัดการและดูแล ก็จะเป็นหน้าที่ของผู้ให้บริการ Serverless ต่างๆ ที่จะทำแทนเรา โดยมันจะมีการขยายตัว (Scale out) หดตัว (Scale down) เพื่อจัดทรัพยากรย์ต่างๆ ให้รองรับปริมาณการทำงานตามความเหมาะสมแบบอัตโนมัติ และคิดค่าใช้จ่ายตามการใช้งานจริง</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1200" height="667" src="https://myifew.com/wp-content/uploads/2019/02/lambda-memory-timeout-1200x667.png" alt="" class="wp-image-5299" srcset="https://myifew.com/wp-content/uploads/2019/02/lambda-memory-timeout-1200x667.png 1200w, https://myifew.com/wp-content/uploads/2019/02/lambda-memory-timeout-1024x569.png 1024w, https://myifew.com/wp-content/uploads/2019/02/lambda-memory-timeout-768x427.png 768w, https://myifew.com/wp-content/uploads/2019/02/lambda-memory-timeout-700x389.png 700w, https://myifew.com/wp-content/uploads/2019/02/lambda-memory-timeout-600x333.png 600w, https://myifew.com/wp-content/uploads/2019/02/lambda-memory-timeout.png 1292w" sizes="auto, (max-width: 1200px) 100vw, 1200px" /><figcaption>ใน AWS Lambda เราจัดสรรทรัพยากรได้แค่ Memory กับกำหนด Timeout ได้ เพราะสองเรื่องนี้จะมีผลกับค่าใช้จ่ายโดยตรง</figcaption></figure>



<h2 class="wp-block-heading">AWS L<g class="gr_ gr_5 gr-alert gr_spell gr_inline_cards gr_run_anim ContextualSpelling ins-del multiReplace" id="5" data-gr-id="5">ambda</g> คืออะไร?</h2>



<p>เป็น Serverless Platform ที่สร้างโดย Amazon ตั้งแต่ November 2014 โดยมีรูปแบบเป็น FaaS (Function-as-a-Service) หรือว่าง่ายๆ คือ Lambda 1 ตัวที่ถูกสร้างชั่วคราวเพื่อมาใช้งาน จะมีเพียง 1 Function หรือหนึ่งการทำงานเท่านั้น (ประมาณว่า Server 1 ตัว เก็บแค่ 1 Function)</p>



<p>และมันได้ออกแบบมาเป็น EDA (Event-driven architecture) คือ ต้องมี Event Trigger ไปเรียกมันขึ้นมาใช้งาน ไม่สามารถทำงานได้ด้วยตัวเอง</p>



<p>ตอนนี้รองรับภาษายอดนิยม อย่าง Node.js, Python, Java, Go, Ruby และ C# (.NET Core) และถ้าใครใช้ PHP แม้จะยังไม่รองรับ แต่ก็สามารถทำ <a href="https://aws.amazon.com/blogs/apn/aws-lambda-custom-runtime-for-php-a-practical-example/" target="_blank" rel="noreferrer noopener" aria-label="PHP Custom Runtime (opens in a new tab)">PHP Custom Runtime</a> เองได้</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1200" height="555" src="https://myifew.com/wp-content/uploads/2019/02/working-with-lambda-1200x555.png" alt="" class="wp-image-5293" srcset="https://myifew.com/wp-content/uploads/2019/02/working-with-lambda-1200x555.png 1200w, https://myifew.com/wp-content/uploads/2019/02/working-with-lambda-1024x474.png 1024w, https://myifew.com/wp-content/uploads/2019/02/working-with-lambda-768x355.png 768w, https://myifew.com/wp-content/uploads/2019/02/working-with-lambda-700x324.png 700w, https://myifew.com/wp-content/uploads/2019/02/working-with-lambda-600x278.png 600w, https://myifew.com/wp-content/uploads/2019/02/working-with-lambda.png 1522w" sizes="auto, (max-width: 1200px) 100vw, 1200px" /><figcaption>ภาพจาก: <br><a href="https://www.slideshare.net/AmazonWebServices/deep-dive-on-aws-lambda">https://www.slideshare.net/AmazonWebServices/deep-dive-on-aws-lambda</a></figcaption></figure>



<p>รายละเอียดอื่นๆ ไปดูเพิ่มเติมได้ที่ <a rel="noreferrer noopener" aria-label="https://aws.amazon.com/lambda/ (opens in a new tab)" href="https://aws.amazon.com/lambda/" target="_blank">https://aws.amazon.com/lambda/</a> ครับ</p>



<h2 class="wp-block-heading">Serverless ทำงานอย่างไร?</h2>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1200" height="526" src="https://myifew.com/wp-content/uploads/2019/02/serverless-request-1200x526.png" alt="" class="wp-image-5289" srcset="https://myifew.com/wp-content/uploads/2019/02/serverless-request-1200x526.png 1200w, https://myifew.com/wp-content/uploads/2019/02/serverless-request-1024x448.png 1024w, https://myifew.com/wp-content/uploads/2019/02/serverless-request-768x336.png 768w, https://myifew.com/wp-content/uploads/2019/02/serverless-request-700x307.png 700w, https://myifew.com/wp-content/uploads/2019/02/serverless-request-600x263.png 600w, https://myifew.com/wp-content/uploads/2019/02/serverless-request.png 1498w" sizes="auto, (max-width: 1200px) 100vw, 1200px" /><figcaption>ภาพจาก: <br><a href="https://medium.com/@chrishantha/ballerina-services-in-serverless-world-b54c5e7382a0">https://medium.com/@chrishantha/ballerina-services-in-serverless-world-b54c5e7382a0</a> <br></figcaption></figure>



<p>ตามที่เกริ่นไปก่อนหน้า เมื่อมันไม่มี Server เปิดทิ้งไว้ มันจึงต้องมีอะไรบางอย่างไปสะกิดเพื่อปลุกมันให้ตื่น (หรือที่เรียกว่า Event Trigger) โดยส่วนมากแล้วก็ถูกนำไปใช้กับ API Gateway เพื่อทำเป็น API Backend </p>



<p>โดยเมื่อมีใครเรียกใช้มัน มันก็จะสร้าง Server (หรือเรียกว่า Instant) ขึ้นมาเพื่อประมวลผลอะไรบางอย่าง และส่งผลลัพธ์ให้เรา และเมื่อไม่ได้ใช้บริการมันในระยะเวลาหนึ่ง มันจะกลับไปหลับเหมือนเดิม (หรือเรียกว่า Idle, Sleep) เพื่อรอการเรียกอีกครั้ง ซึ่งในขณะที่มันหลับอยู่ก็จะไม่คิดค่าบริการกับเรา</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1200" height="668" src="https://myifew.com/wp-content/uploads/2019/02/serverless-lifecycle-1200x668.png" alt="" class="wp-image-5287" srcset="https://myifew.com/wp-content/uploads/2019/02/serverless-lifecycle-1200x668.png 1200w, https://myifew.com/wp-content/uploads/2019/02/serverless-lifecycle-1024x570.png 1024w, https://myifew.com/wp-content/uploads/2019/02/serverless-lifecycle-768x427.png 768w, https://myifew.com/wp-content/uploads/2019/02/serverless-lifecycle-700x390.png 700w, https://myifew.com/wp-content/uploads/2019/02/serverless-lifecycle-600x334.png 600w, https://myifew.com/wp-content/uploads/2019/02/serverless-lifecycle.png 1258w" sizes="auto, (max-width: 1200px) 100vw, 1200px" /><figcaption>ภาพจาก: myifew.com &#8211; อธิบายการทำงาน Lambda Life Cycle<br></figcaption></figure>



<p>ลงในรายละเอียดอีกนิด และสำคัญมากๆ คือ ตอนมัน Initial เพื่อสร้าง Server ขึ้นมา มันจะโหลดโค้ดของเรา และทำการติดตั้งนั่นโน่นนี่ และรันโค้ดของเราเพื่อมาทำงาน </p>



<p>ที่มันสร้างและติดตั้งต่างๆนี่เอง เรียกว่าช่วง Cold Start  ผมให้ข้อสังเกตว่า ตรงนี้เป็นปัญหาหนึ่งที่ทำให้กินเวลาการประมวลผลเพิ่มขึ้น แต่จะเกิดกับครั้งแรกที่รันเท่านั้น พอรันซ้ำอีกครั้ง ในเวลาไล่เลี่ยกัน (หรือก่อนที่มันจะหลับ) มันจะ execute code อย่างเดียว ก็จะทำให้เร็วตามที่โค้ดเราทำงานจริงๆ </p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1194" height="672" src="https://myifew.com/wp-content/uploads/2019/02/serverless-lifecycle-condition.png" alt="" class="wp-image-5288" srcset="https://myifew.com/wp-content/uploads/2019/02/serverless-lifecycle-condition.png 1194w, https://myifew.com/wp-content/uploads/2019/02/serverless-lifecycle-condition-1024x576.png 1024w, https://myifew.com/wp-content/uploads/2019/02/serverless-lifecycle-condition-768x432.png 768w, https://myifew.com/wp-content/uploads/2019/02/serverless-lifecycle-condition-700x394.png 700w, https://myifew.com/wp-content/uploads/2019/02/serverless-lifecycle-condition-600x338.png 600w" sizes="auto, (max-width: 1194px) 100vw, 1194px" /><figcaption>ภาพจาก: <br><a href="https://www.slideshare.net/AmazonWebServices/srv310designing-microservices-with-serverless">https://www.slideshare.net/AmazonWebServices/srv310designing-microservices-with-serverless</a> <br></figcaption></figure>



<p>ถามว่า Cold Start กินเวลานานขนาดไหน ต้องบอกว่าเป็นหลักวินาที ยิ่งโค้ดใหญ่มาก จัดสรร Memory ให้น้อย ก็ยิ่งใช้เวลานาน ตรงนี้เป็นจุดหนึ่งที่ต้องคิดเยอะๆ เพราะมีผลกับผู้ใช้งานคนแรก และค่าใช้จ่ายที่ตามมา</p>



<blockquote class="wp-block-quote is-layout-flow wp-block-quote-is-layout-flow"><p><em>ปล. ผมโดนถามบ่อยๆว่า นานแค่ไหนที่ Server จะเข้าสู่โหมด Idle? จากที่เจอเอง ใช้ Server Singapore ผมลุกไปเข้าห้องน้ำ 5-10 นาที กลับมาก็รู้สึกช้าแล้วครับ แต่กับที่ไปหาอ่านของฝรั่ง น่าจะใช้ Server ทวีปอื่น บางคนบอก 15 นาที บางคนบอก 45 นาที ผมจึงเชื่อว่าทวีปน่าจะมีผลกับเวลา Idle</em></p><cite>iFew</cite></blockquote>



<h2 class="wp-block-heading">การทำงานแบบ Continuous Scaling</h2>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1200" height="629" src="https://myifew.com/wp-content/uploads/2019/02/continuous-scaling-1200x629.png" alt="" class="wp-image-5290" srcset="https://myifew.com/wp-content/uploads/2019/02/continuous-scaling-1200x629.png 1200w, https://myifew.com/wp-content/uploads/2019/02/continuous-scaling-1024x537.png 1024w, https://myifew.com/wp-content/uploads/2019/02/continuous-scaling-768x403.png 768w, https://myifew.com/wp-content/uploads/2019/02/continuous-scaling-700x367.png 700w, https://myifew.com/wp-content/uploads/2019/02/continuous-scaling-600x315.png 600w, https://myifew.com/wp-content/uploads/2019/02/continuous-scaling.png 1648w" sizes="auto, (max-width: 1200px) 100vw, 1200px" /><figcaption>ภาพจาก: myifew.com</figcaption></figure>



<p>สั่งเกตว่าใช้คำว่า Continuous Scaling ไม่ใช่ Auto Scaling แบบที่เคยทำกับ Cloud ทั่วไป แล้วมันทำงานอย่างไรล่ะ?</p>



<p>อ้างอิงจากเอกสาร <a rel="noreferrer noopener" aria-label="Understanding Scaling Behavior (opens in a new tab)" href="https://docs.aws.amazon.com/lambda/latest/dg/scaling.html" target="_blank">Understanding Scaling Behavior</a> ของ Amazon ได้บอกไว้ว่า AWS Lambda มีการ Initial ครั้งแรกก็จะรองรับ Concurrency ได้จำนวนหนึ่ง แต่จะแตกต่างกันตามแต่ละทวีปที่เราใช้บริการ </p>



<p>เช่น ผมเลือกใช้ของ Singapore ดังนั้นการสร้าง Server ขึ้นมาครั้งแรก มันจะรองรับ Concurrency ที่ 500 และถ้ามีปริมาณผู้ใช้มากขึ้นเรื่อยๆ มันจะทำการเพิ่ม Concurrency ให้ครั้งละ 500 ในทุกๆ 1 นาที จนกว่าจะชนเพดาน Limit ที่เรามี (ซึ่งในภาพตัวอย่าง คือ รองรับได้ที่ 1,000 Concurrency พร้อมกัน ซึ่งทุกคนจะได้ค่ามาตรฐานเป็นเท่านี้ ยกเว้นไปอีเมลขอปริมาณเพิ่มจากทาง Amazon)</p>



<p>โดย Concurrency ทั้งหมดที่เรามีนั้น  Amazon จะกระจายให้กับทุก Function ตามการใช้งานจริง เช่น Function A มีการเรียกใช้ 125 Concurrency จะทำให้ Function อื่นๆ ทั้งหมดแย่งกันใช้จาก 875 Concurrency ที่เหลือ ดังนั้น Amazon จึงมีฟีเจอร์ให้เราจอง Concurrency ให้กับ Function ได้ว่า ต้องมีเหลือให้ใช้เสมอ ในจำนวนเท่าไร</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1200" height="534" src="https://myifew.com/wp-content/uploads/2019/02/control-concurrency-1200x534.png" alt="" class="wp-image-5292" srcset="https://myifew.com/wp-content/uploads/2019/02/control-concurrency-1200x534.png 1200w, https://myifew.com/wp-content/uploads/2019/02/control-concurrency-1024x456.png 1024w, https://myifew.com/wp-content/uploads/2019/02/control-concurrency-768x342.png 768w, https://myifew.com/wp-content/uploads/2019/02/control-concurrency-700x312.png 700w, https://myifew.com/wp-content/uploads/2019/02/control-concurrency-600x267.png 600w, https://myifew.com/wp-content/uploads/2019/02/control-concurrency.png 1666w" sizes="auto, (max-width: 1200px) 100vw, 1200px" /><figcaption>ภาพจาก: myifew.com</figcaption></figure>



<p>รวมไปถึง หากพบว่ามีการประมวลผลแปลกๆ ทำงานนานผิดปกติ หรือพบว่ามีการ Reuqest มากผิดปกติ ซึ่งจะส่งผลกับค่าใช้จ่ายที่มากขึ้น เราสามารถปิดการทำงานชั่วคราวได้ด้วยการลด  Concurrency ให้เหลือศูนย์ (Throttle)</p>



<h2 class="wp-block-heading" id="mce_30">Serverless เอาไปใช้งานใดได้บ้าง?</h2>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1200" height="554" src="https://myifew.com/wp-content/uploads/2019/02/serverless-use-case-1200x554.png" alt="" class="wp-image-5296" srcset="https://myifew.com/wp-content/uploads/2019/02/serverless-use-case-1200x554.png 1200w, https://myifew.com/wp-content/uploads/2019/02/serverless-use-case-1024x473.png 1024w, https://myifew.com/wp-content/uploads/2019/02/serverless-use-case-768x355.png 768w, https://myifew.com/wp-content/uploads/2019/02/serverless-use-case-700x323.png 700w, https://myifew.com/wp-content/uploads/2019/02/serverless-use-case-600x277.png 600w, https://myifew.com/wp-content/uploads/2019/02/serverless-use-case.png 1494w" sizes="auto, (max-width: 1200px) 100vw, 1200px" /><figcaption>ภาพจาก <br><a href="https://www.slideshare.net/AmazonWebServices/deep-dive-on-aws-lambda">https://www.slideshare.net/AmazonWebServices/deep-dive-on-aws-lambda</a> <br></figcaption></figure>



<p>อ้างอิงจากรูป แบ่งตามประเภทแล้ว มันสามารถใช้งานได้ในหลากหลายรูปแบบ แต่เราต้องเข้าใจธรรมชาติมันก่อนนะว่า มันเกิดขึ้น ตั้งอยู่ ดับไป คิดเงินตามการใช้งานจริง และต้องมี Event Trigger ไปปลุกให้มันตื่น</p>



<p>จากความคิดเห็นส่วนตัวของผม ถ้าระบบที่เราจะนำไปใช้ มีความจำเป็นต้องใช้งานตลอดเวลา บางทีการมี Server ปกติ เปิดทิ้งไว้ อาจจะมีค่าใช้จ่ายที่ถูกกว่า และจัดการได้ง่ายกว่า หรือจะพิจารณา API บางตัวไปใช้เป็น Serverless ก็เพียงพอ ไม่ต้องทำทั้งระบบ ทั้งนี้ ไม่มีใครบอกได้ว่าควรใช้ส่วนไหนบ้าง อยู่ที่ผู้เข้าใจระบบเท่านั้น</p>



<h2 class="wp-block-heading">ข้อดี/ข้อเสียของ Serverless</h2>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1200" height="664" src="https://myifew.com/wp-content/uploads/2019/02/serverless-advantage-1200x664.png" alt="" class="wp-image-5294" srcset="https://myifew.com/wp-content/uploads/2019/02/serverless-advantage-1200x664.png 1200w, https://myifew.com/wp-content/uploads/2019/02/serverless-advantage-1024x566.png 1024w, https://myifew.com/wp-content/uploads/2019/02/serverless-advantage-768x425.png 768w, https://myifew.com/wp-content/uploads/2019/02/serverless-advantage-700x387.png 700w, https://myifew.com/wp-content/uploads/2019/02/serverless-advantage-600x332.png 600w, https://myifew.com/wp-content/uploads/2019/02/serverless-advantage.png 1222w" sizes="auto, (max-width: 1200px) 100vw, 1200px" /><figcaption>ภาพจาก <br><a href="https://www.slideshare.net/AmazonWebServices/deep-dive-on-aws-lambda">https://www.slideshare.net/AmazonWebServices/deep-dive-on-aws-lambda </a><br></figcaption></figure>



<p>หลังจากเข้าใจแล้วว่าคืออะไร ลองมาพิจารณาข้อดีข้อเสียและชั่งน้ำหนักก่อนจะตัดสินใจใช้งานดูนะครับ</p>



<ul class="wp-block-list"><li><strong>ข้อดี</strong><ul><li>ลดค่าใช้จ่าย เพราะจ่ายเท่าที่ประมวลผลจริง ไม่มีเปิด Server ทิ้งไว้</li><li>ไม่มี Server ให้ต้องดูแลและจัดการ</li><li>Scale&nbsp;ได้ต่อเนื่องตามการใช้งานจริง และอัตโนมัติ</li></ul></li><li><strong>ข้อเสีย</strong><ul><li>เรื่องของการ Initial Server ครั้งแรก (Cold&nbsp;Start) จะใช้เวลานิดหนึ่ง</li><li>ทรัพยากรที่จำกัด (Memory 128MB-3GB, Limit Concurrency 1,000/region/account)</li><li>แนวคิดการออกแบบซอร์ฟแวร์ที่จะยากขึ้น ว่าทำอย่างไรให้หั่นการทำงานใหญ่ๆ เป็นการทำงานย่อยๆ</li><li>การทดสอบใน Local Machine ยาก จำเป็นต้องมี Unit Test (และก็เขียน Unit Test ยากเช่นกัน)</li><li>Monitoring และ&nbsp;debugging ค่อนข้างยาก (ทำได้แต่เสียเงินเพิ่ม อย่าง AWS Lambda ต้องเปิดบริการอย่าง AWS Cloudwatch,  AWS X-Ray ในการดูเพิ่มเติม)</li><li>OS vulnerabilities ถูกดูแลโดยผู้ให้บริการ Serverless, เราอัพเดท&nbsp;หรือแก้ไขอะไรเองไม่ได้</li></ul></li></ul>



<h2 class="wp-block-heading">แถม: Serverless สามารถลดค่าใช้จ่ายได้จริงหรือ?</h2>



<p>เป็นข้อดีและก็เป็นข้อสงสัยว่า Serverless สามารถลดค่าใช้จ่ายได้จริงหรือ ซึ่งผมคิดว่าอยู่ที่การพิจารณาเอาไปใช้ ว่าทดแทนส่วนไหนแล้วลดค่าใช้จ่ายได้ ซึ่งไม่มีใครตัดสินใจให้ได้ แต่ผมจะขออธิบายโครงสร้างราคาและตัวอย่าง ของ AWS Lambda ไว้เป็นแนวทาง</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1200" height="450" src="https://myifew.com/wp-content/uploads/2019/02/aws-lambda-pricing-1200x450.png" alt="" class="wp-image-5297" srcset="https://myifew.com/wp-content/uploads/2019/02/aws-lambda-pricing-1200x450.png 1200w, https://myifew.com/wp-content/uploads/2019/02/aws-lambda-pricing-1024x384.png 1024w, https://myifew.com/wp-content/uploads/2019/02/aws-lambda-pricing-768x288.png 768w, https://myifew.com/wp-content/uploads/2019/02/aws-lambda-pricing-700x262.png 700w, https://myifew.com/wp-content/uploads/2019/02/aws-lambda-pricing-600x225.png 600w, https://myifew.com/wp-content/uploads/2019/02/aws-lambda-pricing.png 1964w" sizes="auto, (max-width: 1200px) 100vw, 1200px" /><figcaption>ที่มา https://aws.amazon.com/lambda/pricing/</figcaption></figure>



<p>โดย AWS Lambda มีการคิดเงินจากสองส่วน คือ</p>



<ul class="wp-block-list"><li><strong>คิดตามจำนวนการเรียกใช้ (Requests)</strong><ul><li>ให้ฟรี 1 ล้าน Requests ทุกเดือน</li><li>1 ล้านต่อไป คิดที่ $0.20 (~6 บาท)</li></ul></li><li><strong>คิดตามระยะเวลาประมวลผล (Duration Time)</strong><ul><li>ให้ฟรีประมวลผล 400,000 GB-Seconds* ทุกเดือน</li><li>ทุกๆ 1 GB-Seconds คิดที่ $0.00001667 (~0.00055011 บาท)</li><li>ทุกครั้งที่มีประมวลผล จะปัดเศษขึ้นที่หลัก 100ms (เช่น 15237ms = 15300ms, 97.12ms = 100ms, 0.43ms = 100ms) </li></ul></li></ul>



<blockquote class="wp-block-quote is-layout-flow wp-block-quote-is-layout-flow"><p><em>ปล. GB-Seconds ย่อมาจาก Gigabyte second หมายความว่า การประมวลผลโดยใช้ Memory จำนวน 1 Gigabyte ใน 1 วินาที (เช่น เราจัดสรรให้ฟังก์ชั่นเราทำงาน 3GB แต่มีการประมวลผล 2 วินาที ก็เท่ากับการรันครั้งนี้ใช้งานไป 6 GB-Seconds)</em></p><cite>iFew</cite></blockquote>



<p>เรื่อง Requests น่าจะเข้าใจได้ แต่เรื่อง Duration Time กับ Memory อาจจะยังงงๆ ให้ดูภาพด้านล่างนี้ครับ</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1200" height="400" src="https://myifew.com/wp-content/uploads/2019/02/lambda-duration-memory-1200x400.png" alt="" class="wp-image-5298" srcset="https://myifew.com/wp-content/uploads/2019/02/lambda-duration-memory-1200x400.png 1200w, https://myifew.com/wp-content/uploads/2019/02/lambda-duration-memory-1024x341.png 1024w, https://myifew.com/wp-content/uploads/2019/02/lambda-duration-memory-768x256.png 768w, https://myifew.com/wp-content/uploads/2019/02/lambda-duration-memory-700x233.png 700w, https://myifew.com/wp-content/uploads/2019/02/lambda-duration-memory-600x200.png 600w, https://myifew.com/wp-content/uploads/2019/02/lambda-duration-memory.png 1488w" sizes="auto, (max-width: 1200px) 100vw, 1200px" /><figcaption>ภาพจาก: myifew.com</figcaption></figure>



<p>ในหน้าของ AWS Lambda เองจะมีข้อมูลบอกแบบนี้ ว่า Duration Time ของการประมวลผลจริงๆ เท่าไร (Actual) และคิดเงินที่เท่าไร (Billed) จากนั้นเอา Memory ที่เราจัดสรรให้กับ Server เรา มาคำนวณเพื่อหาว่า GB-Second ของการทำงานครั้งนี้เป็นเท่าไร</p>



<h3 class="wp-block-heading">โจทย์ตัวอย่าง</h3>



<p>ถ้ามีฟังก์ชั่นชื่อ get_product() โดยเราจัดสรร Memory ให้ 128MB มีการเรียกใช้งาน 30 ล้านครั้งใน 1 เดือน ซึ่งแต่ละครั้ง ใช้เวลาประมวลผล 200ms </p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1200" height="381" src="https://myifew.com/wp-content/uploads/2019/02/lambda-example-pricing-1200x381.png" alt="" class="wp-image-5301" srcset="https://myifew.com/wp-content/uploads/2019/02/lambda-example-pricing-1200x381.png 1200w, https://myifew.com/wp-content/uploads/2019/02/lambda-example-pricing-1024x325.png 1024w, https://myifew.com/wp-content/uploads/2019/02/lambda-example-pricing-768x244.png 768w, https://myifew.com/wp-content/uploads/2019/02/lambda-example-pricing-700x222.png 700w, https://myifew.com/wp-content/uploads/2019/02/lambda-example-pricing-600x190.png 600w, https://myifew.com/wp-content/uploads/2019/02/lambda-example-pricing.png 1594w" sizes="auto, (max-width: 1200px) 100vw, 1200px" /><figcaption>ภาพจาก: myifew.com</figcaption></figure>



<p>ในเดือนนั้น เราต้องจ่ายค่าประมวลผล get_product() ทั้งหมด $11.63 โดยคิดเป็นค่า Requests $5.80 และค่าประมวลผล (Compute) $5.83</p>



<h3 class="wp-block-heading">วิธีการคิด&nbsp;ค่า&nbsp;Requests</h3>



<p><strong>สูตรคือ</strong> Total requests – Free tier request = Billable requests<br><strong>แทนค่า</strong> 30M requests – 1M free tier requests = 29M Monthly billable requests<br><strong>สรุป</strong> Monthly request charges = 29M * $0.2/M = $5.80</p>



<h3 class="wp-block-heading" id="mce_70">วิธีการคิด&nbsp;ค่า&nbsp;Compute (Duration Time)</h3>



<p><strong>สูตรคือ</strong> Total compute (seconds) = 30,000,000 * 0.2sec = 6,000,000 seconds<br><strong>แปลงเป็น GB-Second</strong> Total compute (GB-s) = 6,000,000sec * 128MB/1024 = 750,000 GB-s<br><strong>หา GB-Second ที่ต้องจ่าย</strong> 750,000 GB-s – 400,000 free tier GB-s = 350,000 GB-s<br><strong>สรุป</strong> Monthly compute charges = 350,000 * $0.00001667 = $5.83</p>



<h2 class="wp-block-heading">สรุป</h2>



<p>หลังจากที่ผมได้เล่น AWS lambda แล้ว ผมชอบคอนเซ็ปมันนะ รู้สึกว่ามันกำลังพาเราก้าวเข้าสู่ยุคสมัยใหม่ (ขนาดนั้นเลยจริงๆ) แล้วมันก็มาพร้อมกับเรื่องที่เราต้องเรียนรู้อีกมากมาย เพราะจากความรู้สึกส่วนตัวของผม ผมรู้สึกเห็นภาพของ Unit Test และการ Design Architectuer อย่าง  Microservice ได้ชัดเจนมาก เพราะด้วย Platform มันบีบให้เราต้องทำ เพราะถ้าเราไม่ทำ มันมีค่าใช้จ่ายโดยตรงที่เราต้องเสีย อย่างจำนวน Request และ Duration Time ชัดเจน (ก่อนหน้านั้น เป็นค่าใช้จ่ายแฝงของการบริหารโปรเจ็คที่เราเห็นภาพไม่ค่อยชัด อย่างเข่นการไม่มี Test)</p>



<p><strong>Slide</strong></p>



<ul class="wp-block-list"><li><a href="https://www.slideshare.net/few/introduce-aws-lambda-for-newbie-and-nonit" target="_blank" rel="noreferrer noopener" aria-label="Introduce AWS Lambda for newbie and Non-IT (opens in a new tab)">Introduce AWS Lambda for newbie and Non-IT</a></li></ul>



<p><strong>อ้างอิง</strong></p>



<ul class="wp-block-list"><li>รูปปกจาก http://alena-vysotskaya.ru/stone-wall-wallpapers/img644822B2784</li><li>รูปอื่นๆ จาก ลิงค์ที่อ้างอิงใต้รูป</li></ul>
]]></content:encoded>
					
					<wfw:commentRss>https://myifew.com/5166/understand-serverless-with-aws-lambda-for-newbie/feed/</wfw:commentRss>
			<slash:comments>0</slash:comments>
		
		
			</item>
		<item>
		<title>ทดสอบ AWS Lambda Function บนเครื่องตัวเอง ด้วย Lambda Docker</title>
		<link>https://myifew.com/5257/how-to-test-aws-lambda-on-local-machine-via-lambda-docker/</link>
					<comments>https://myifew.com/5257/how-to-test-aws-lambda-on-local-machine-via-lambda-docker/#respond</comments>
		
		<dc:creator><![CDATA[iFew]]></dc:creator>
		<pubDate>Thu, 21 Feb 2019 16:25:07 +0000</pubDate>
				<category><![CDATA[Technology]]></category>
		<category><![CDATA[.Net Core]]></category>
		<category><![CDATA[AWS]]></category>
		<category><![CDATA[AWS Lambda]]></category>
		<category><![CDATA[Serverless]]></category>
		<category><![CDATA[Testing Tools]]></category>
		<guid isPermaLink="false">https://myifew.com/?p=5257</guid>

					<description><![CDATA[จากที่เคยนำเสนอเครื่องมือตัวหนึ่งไปแล้วในบล็อก ทดสอบ .NET AWS Lambda ด้วย AWS .NET Mock Lambda Test Tool แต่ยังไม่หนำใจ เพราะผมพบว่ามันมีปัญหากับโค้ดที่เขียนแบบ Dependency Injection ซึ่งลองหาวิธี Debug&#8230;]]></description>
										<content:encoded><![CDATA[
<p>จากที่เคยนำเสนอเครื่องมือตัวหนึ่งไปแล้วในบล็อก <a href="https://myifew.com/5223/test-netcore-aws-lambda-by-aws-net-mock-lambda-test-tool/">ทดสอบ .NET AWS Lambda ด้วย AWS .NET Mock Lambda Test Tool</a> แต่ยังไม่หนำใจ เพราะผมพบว่ามันมีปัญหากับโค้ดที่เขียนแบบ Dependency Injection ซึ่งลองหาวิธี Debug แต่ก็ยังไม่เห็นอะไร เลยตัดใจลองหาเครื่องมือตัวอื่นดู ก็เจอ <a rel="noreferrer noopener" aria-label="AWS SAM (AWS Serverless Application Model) (opens in a new tab)" href="https://aws.amazon.com/serverless/sam/" target="_blank">AWS SAM (AWS Serverless Application Model)</a> </p>



<p>แต่ก็กระนั้นอีก ในหน้าเว็บโปรเจ็คบอกทำงานกับ .NET Core 2.1 ได้ แต่พอลองทำดู กลับบอกว่า ไม่รองรับ .NET Core Runtime!! (ใครแก้ปัญหาสองข้อบนของผมได้ มาแลกเปลี่ยนความรู้กันหน่อยครับ จักหายคาใจ) ก็เลยต้องแกะมัน พบว่า มันไปใช้ Lambda Docker เพื่อสร้าง Environment จำลองขึ้นมาสำหรับทดสอบ.. ปั๊ดโธ่ว แบบนี้ก็เสร็จโก๋!!  จึงเป็นที่มาของบล็อกนี้..</p>



<span id="more-5257"></span>



<p style="text-align:center" class="has-background has-very-light-gray-background-color"><strong>คำแนะนำ: ผู้อ่านบล็อกนี้ควรใช้งาน AWS Lambda และ Docker เบื้องต้นแล้ว,&nbsp;ตัวอย่างใช้เป็น&nbsp;.NET&nbsp;Core 2.1&nbsp;แต่สามารถนำวิธีใช้กับภาษาอื่นๆได้</strong></p>



<h2 class="wp-block-heading"> Lambda Docker คืออะไร? ดีไหม?</h2>



<p><g class="gr_ gr_24 gr-alert gr_gramm gr_inline_cards gr_disable_anim_appear Grammar only-ins doubleReplace replaceWithoutSep" id="24" data-gr-id="24">เป็น</g>&nbsp;Docker ที่จำลอง Environment ไว้เพื่อทดสอบ (Sandbox) ของ&nbsp;<a href="https://aws.amazon.com/lambda/">AWS Lambda</a>&nbsp;ซึ่งเคลมตัวเองว่า ใกล้เคียงกับของจริงมากๆ</p>



<p>ถามว่าใกล้เคียงขนาดไหน? คำตอบคือใกล้มากจริงๆ ตามที่เขาโม้ไว้ เพราะภายใน image ของมันประกอบไปด้วย software, libraries, โครงสร้างของไฟล์ และ permissions, environment variables, context objects หรือแม้แต่  user สำหรับใช้เรียก process ก็ยังใช้ชื่อเดียวกัน</p>



<p>ด้วยความเหมือนเป๊ะ มันจึงเหมาะที่จะเอา Lambda Function มาทดสอบบน Docker Environment ที่มีข้อจำกัดคล้ายกับ AWS Lambda ได้เกือบ 100%, และที่สำคัญ มันปรับ Memory, Timeout และใช้ Layer ได้ด้วยนะ </p>



<p>มีข้อดีแล้วก็ต้องมีข้อเสีย ซึ่งตอนนี้ผมพบจุดเดียว คือ เรื่องของเวลาในการคำนวณ ซึ่งเอาไปอ้างอิงไม่ได้ เพราะไม่มี Cold Start และมันเป็น CPU บนเครื่องเรา, ถ้าจำทดสอบเพื่อไปคำนวณค่าใช้จ่าย ควรใช้เครื่องมืออีกตัวชื่อว่า <a rel="noreferrer noopener" aria-label="AWS Lambda Power Tuning (opens in a new tab)" href="https://github.com/alexcasalboni/aws-lambda-power-tuning" target="_blank">AWS Lambda Power Tuning</a> (ไว้จะเขียนถึงอีกที)</p>



<h2 class="wp-block-heading">สิ่งที่ต้องติดตั้งก่อนใช้ Lambda Docker </h2>



<p>ก่อนไปต่อ ควรติดตั้ง <a rel="noreferrer noopener" aria-label="Docker (opens in a new tab)" href="https://www.docker.com/get-started" target="_blank">Docker</a> ซะก่อนนะ (เช็คให้เรียบร้อยว่าไม่ติด Security Policy ทำงานได้ดี สามารถ Pull Image และ Run Container ได้)</p>



<h2 class="wp-block-heading">เริ่มต้นใช้งาน Lambda Docker</h2>



<p>รูปแบบของคำสั่งจะเป็นดังนี้</p>



<pre class="wp-block-code"><code>docker run [--rm] -v &lt;code_dir>:/var/task [-v &lt;layer_dir>:/opt] lambci/lambda:&lt;runtime> [&lt;handler>] [&lt;event>]</code></pre>



<p>การทำงานของมันคือ จะสร้าง Docker Container ขึ้นมา และเอาโค้ดของเราไปรันทดสอบ ด้วยตัวแปรต่างๆที่เราระบุไว้ใน Command แค่นั้นเอง</p>



<ul class="wp-block-list"><li><strong>&#8211;rm</strong> คือเพื่อให้สร้าง Container มาทดสอบเพียงครั้งเดียว และทำลายตัวเองทิ้ง (ไม่ใส่ก็ได้ รันทำงานซ้ำไปได้เรื่อยๆ)</li><li><strong> -v &lt;code_dir&gt;:/var/task</strong> คือ ชี้ที่อยู่ของโค้ดเรา ไปให้ตรงกับโฟลเดอร์ใน Container เพื่อใช้งาน</li><li> <strong>-v &lt;layer_dir&gt;:/opt</strong> คือ ชี้ที่อยู่ของ layer  โค้ด ไปให้ตรงกับโฟลเดอร์ใน Container เพื่อใช้งาน (ถ้าไม่รู้จัก หรือไม่ได้ใช้ ก็ไม่ต้องใส่)</li><li><strong>&lt;runtime&gt;</strong> &nbsp;คือ Runtime Language ที่เราใช้เขียน Function เช่น dotnetcore2.1, nodejs4.3, python3.7, java8, go1.x</li><li><strong>&lt;handler&gt;</strong> คือ Fucntion Handler </li><li><strong>&lt;event&gt;</strong> คือ Request Input นั่นหละ</li></ul>



<h2 class="wp-block-heading">ทดสอบรันคำสั่งง่ายๆ</h2>



<p>ตัวอย่าง Function ของผมชื่อ aws_lambda_function โดยรับ string เข้าไป จากนั้นจะพ่นออกมาว่า &#8220;Hello, &lt;ชื่อ&gt;&#8221; โดยมีรูปแบบคำสั่งคือ</p>



<pre class="wp-block-code"><code>docker run --rm -v "$PWD":/var/task lambci/lambda:dotnetcore2.1 aws-lambda-function::aws_lambda_function.Function::FunctionHandler '"iFew"'</code></pre>



<p>เนื่องด้วยการทำงานมันจะ run code ที่ผ่านการ complie แล้ว, ซึ่งถ้าให้แน่ใจ ว่าไม่ได้รันอยู่บนโค้ดชุดเดิม ให้เราทำคำสั่ง compile ก่อนสักครั้ง จากนั้นไปที่โฟลเดอร์โค้ดที่ Compiled และทำคำสั่งด้านบนอีกที (ผมชี้ที่อยู่โค้ดไปที่ $PWD หมายถึง ให้ใช้โค้ดในโฟลเดอร์ปัจจุบันที่ผมอยู่)</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1200" height="132" src="https://myifew.com/wp-content/uploads/2019/02/lambda-docker-basic-command-1-1200x132.png" alt="" class="wp-image-5266" srcset="https://myifew.com/wp-content/uploads/2019/02/lambda-docker-basic-command-1-1200x132.png 1200w, https://myifew.com/wp-content/uploads/2019/02/lambda-docker-basic-command-1-1024x112.png 1024w, https://myifew.com/wp-content/uploads/2019/02/lambda-docker-basic-command-1-768x84.png 768w, https://myifew.com/wp-content/uploads/2019/02/lambda-docker-basic-command-1-700x77.png 700w, https://myifew.com/wp-content/uploads/2019/02/lambda-docker-basic-command-1-600x66.png 600w, https://myifew.com/wp-content/uploads/2019/02/lambda-docker-basic-command-1.png 2104w" sizes="auto, (max-width: 1200px) 100vw, 1200px" /></figure>



<p>ผลลัพธ์ที่ได้</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1200" height="93" src="https://myifew.com/wp-content/uploads/2019/02/lambda-docker-basic-command-result-1-1200x93.png" alt="" class="wp-image-5267" srcset="https://myifew.com/wp-content/uploads/2019/02/lambda-docker-basic-command-result-1-1200x93.png 1200w, https://myifew.com/wp-content/uploads/2019/02/lambda-docker-basic-command-result-1-1024x79.png 1024w, https://myifew.com/wp-content/uploads/2019/02/lambda-docker-basic-command-result-1-768x59.png 768w, https://myifew.com/wp-content/uploads/2019/02/lambda-docker-basic-command-result-1-700x54.png 700w, https://myifew.com/wp-content/uploads/2019/02/lambda-docker-basic-command-result-1-600x46.png 600w, https://myifew.com/wp-content/uploads/2019/02/lambda-docker-basic-command-result-1.png 2402w" sizes="auto, (max-width: 1200px) 100vw, 1200px" /></figure>



<h2 class="wp-block-heading" id="mce_51">ทดสอบรันคำสั่งโดยใส่ Environment Variable</h2>



<p>จากตัวอย่างแรก สังเกตว่ามัน Allocate Memory ให้ถึง 1,536MB เลยทีเดียว (เป็น Default ของมัน) ซึ่งผมอยากกำหนดให้ใช้ Memory แค่ 128MB และ Timeout 30 วินาที เพื่อทดสอบการทำงาน ในสภาพแวดใกล้เคียงของจริง ด้วยคำสั่งนี้</p>



<pre class="wp-block-code"><code>docker run --rm -v "$PWD":/var/task -e AWS_LAMBDA_FUNCTION_MEMORY_SIZE=128 -e AWS_LAMBDA_FUNCTION_TIMEOUT=30 lambci/lambda:dotnetcore2.1  aws-lambda-function::aws_lambda_function.Function::FunctionHandler '"iFew"'</code></pre>



<p>ซึ่ง ใช้ parameter ชื่อ -e ของ Docker นั่นหละเป็นตัวกำหนด โดยลัพลัพธ์ที่ได้ คือ</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1200" height="95" src="https://myifew.com/wp-content/uploads/2019/02/lambda-docker-basic-env-command-result-1200x95.png" alt="" class="wp-image-5269" srcset="https://myifew.com/wp-content/uploads/2019/02/lambda-docker-basic-env-command-result-1200x95.png 1200w, https://myifew.com/wp-content/uploads/2019/02/lambda-docker-basic-env-command-result-1024x81.png 1024w, https://myifew.com/wp-content/uploads/2019/02/lambda-docker-basic-env-command-result-768x61.png 768w, https://myifew.com/wp-content/uploads/2019/02/lambda-docker-basic-env-command-result-700x55.png 700w, https://myifew.com/wp-content/uploads/2019/02/lambda-docker-basic-env-command-result-600x47.png 600w, https://myifew.com/wp-content/uploads/2019/02/lambda-docker-basic-env-command-result.png 2684w" sizes="auto, (max-width: 1200px) 100vw, 1200px" /></figure>



<p>สังเกตว่า Memory ตอนนี้จัดสรรให้แค่ 128MB เท่านั้น ส่วน Timoue ยังไม่เห็นผลในตอนนี้เพราะโค้ดง่ายๆ ทำงานไว</p>



<p>ตัว environment ให้ใช้งานต่างๆ ดูเพิ่มเติมได้ที่ https://github.com/lambci/docker-lambda</p>



<h2 class="wp-block-heading">ทดสอบรันคำสั่งโดยอ้างอิงกับ Database ภายนอก</h2>



<p>การใช้งานจริงๆของเรา มันไม่ง่ายเหมือน Function ด้านบนแน่นอน ซึ่งผมเจอปัญหานี้เช่นกัน และใน Document ไม่ได้บอกไว้ว่าทำอย่างไร งมอยู่สักพัก ถึงเจอวิธี โดยการใช้ทำให้ Lambda Docker ของเราอยู่ในวง Network เดียวกับ Docker Database ที่ผมมีซะ (พอดีผมใช้ MySQL บน Docker, ถ้าใครใช้ MySQL เข้าใจว่าสามารถใช้งานได้เลย เนื่องจาก network เป็นแบบ bridge ที่ทำให้ Container เราต่ออินเทอร์เน็ตหรือวง Network ในเครื่องเราได้อยู่แล้ว)</p>



<p>ลองดูตัวอย่างคำสั่งนี้ครับ</p>



<pre class="wp-block-code"><code>docker run --rm -v "$PWD":/var/task --network mysql -e TEST_LAMBDA_DBCONNECTION="server=mysql;userid=root;password=1234;database=test_lambda;convert zero datetime=True; CharSet=utf8" -e AWS_LAMBDA_FUNCTION_MEMORY_SIZE=128 lambci/lambda:dotnetcore2.1 list_profile::list_profile.Function::Get</code></pre>



<p>จะสังเกตว่ามีตัวแปรแปลกๆมานิดนึง คือ &#8211;network mysql หมายถึง ให้ Container นี้อยู่ในวง Network ชื่อ mysql โดยในวงนั้นผมมี Container MySQL อยู่ในนั้นอยู่แล้ว</p>



<p>จากนั้นผมระบุ Environment Variable ชื่อว่า TEST_LAMBDA_DBCONNECTION เข้าไป เนื่องจากโค้ดของผมเรียกใช้ config ผ่าน Environment Variable (ตาม<a rel="noreferrer noopener" aria-label="หลักการข้อ 3 ใน The Twelve Factor (opens in a new tab)" href="https://12factor.net/config" target="_blank">หลักการข้อ 3. Config ใน The Twelve Factor</a>)</p>



<p>ผลลัพธ์ที่ได้คือ</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1200" height="117" src="https://myifew.com/wp-content/uploads/2019/02/lambda-docker-basic-mysql-command-result-1200x117.png" alt="" class="wp-image-5270" srcset="https://myifew.com/wp-content/uploads/2019/02/lambda-docker-basic-mysql-command-result-1200x117.png 1200w, https://myifew.com/wp-content/uploads/2019/02/lambda-docker-basic-mysql-command-result-1024x100.png 1024w, https://myifew.com/wp-content/uploads/2019/02/lambda-docker-basic-mysql-command-result-768x75.png 768w, https://myifew.com/wp-content/uploads/2019/02/lambda-docker-basic-mysql-command-result-700x68.png 700w, https://myifew.com/wp-content/uploads/2019/02/lambda-docker-basic-mysql-command-result-600x59.png 600w, https://myifew.com/wp-content/uploads/2019/02/lambda-docker-basic-mysql-command-result.png 2680w" sizes="auto, (max-width: 1200px) 100vw, 1200px" /></figure>



<h4 class="wp-block-heading">แถมนิดนึง วิธีการสร้าง Docker Network</h4>



<p>คำสั่งง่ายๆ คือ</p>



<pre class="wp-block-code"><code>docket create network &lt;ชื่อnetworkที่ต้องการ></code></pre>



<p>แค่นี้ก็จะได้วง Network มาแล้วครับ โดยในตัวอย่างผมใช้ชื่อวงว่า mysql</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="976" height="226" src="https://myifew.com/wp-content/uploads/2019/02/lambda-docket-network-2.png" alt="" class="wp-image-5274" srcset="https://myifew.com/wp-content/uploads/2019/02/lambda-docket-network-2.png 976w, https://myifew.com/wp-content/uploads/2019/02/lambda-docket-network-2-768x178.png 768w, https://myifew.com/wp-content/uploads/2019/02/lambda-docket-network-2-700x162.png 700w, https://myifew.com/wp-content/uploads/2019/02/lambda-docket-network-2-600x139.png 600w" sizes="auto, (max-width: 976px) 100vw, 976px" /></figure>



<p>ดังนั้น หากเรามี Container อยู่แล้ว ให้เพิ่มเข้าวง Network ด้วยคำสั่ง</p>



<pre class="wp-block-code"><code>docker network connect &lt;ชื่อnetwork> &lt;ชื่อcontainer></code></pre>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1200" height="98" src="https://myifew.com/wp-content/uploads/2019/02/lambda-docket-create-network-1200x98.png" alt="" class="wp-image-5273" srcset="https://myifew.com/wp-content/uploads/2019/02/lambda-docket-create-network-1200x98.png 1200w, https://myifew.com/wp-content/uploads/2019/02/lambda-docket-create-network-1024x83.png 1024w, https://myifew.com/wp-content/uploads/2019/02/lambda-docket-create-network-768x63.png 768w, https://myifew.com/wp-content/uploads/2019/02/lambda-docket-create-network-700x57.png 700w, https://myifew.com/wp-content/uploads/2019/02/lambda-docket-create-network-600x49.png 600w, https://myifew.com/wp-content/uploads/2019/02/lambda-docket-create-network.png 2134w" sizes="auto, (max-width: 1200px) 100vw, 1200px" /></figure>



<h2 class="wp-block-heading" id="mce_69">ทดสอบรันคำสั่งโดยส่ง Request Input ด้วย APIGatewayProxy (หรือรันคำสั่งแบบมี JSON  Input ยาวๆ)</h2>



<p>โดยปกติผมจะใช้ AWS API Gateway อยู่แล้ว ดังนั้นตัว Request Input ผมจะชอบใช้ในรูปแบบของ API Gateway Proxy Request เพราะมันสามารถส่ง Request Input ได้ทั้งแบบ Path Parameter, Query String, Header, Body ซึ่งลดควมซับซ้อน ไม่ต้องไปทำการ Mapping Input Data อะไรให้ยุ่งยากใน AWS API Gateway ซึ่งถ้าคุณใช้เหมือนกัน ก็สามารถเอาตัวอย่างนี้ไปทดสอบกับ Lambda Docker ได้ครับ</p>



<p>โดยตัว API Gateway Proxy Request เนื้อมันจริงๆคือ JSON ที่ AWS API Gateway รับเข้าไป จากนั้นมันจะกระทำการ Mapping นั่นโน่นนี่ให้เอง และสร้าง Request ตามที่เรากำหนดใน JSON นั้นไปทำงานกับ Function เราอีกที ซึ่งรูปแบบของ JSON มันจะยาวมากๆ ครับ ดังนั้น การส่ง Input ยาวๆ เราสามารถทำได้สามแบบ</p>



<h4 class="wp-block-heading">ในตัวอย่างแรก สามารถใส่ event ต่อท้ายได้ตามรูปแบบมาตรฐานของมันเลยครับ</h4>



<pre class="wp-block-code"><code>docker run --rm -v "$PWD":/var/task --network mysql -e TEST_LAMBDA_DBCONNECTION="server=mysql;userid=root;password=1234;database=test_lambda;convert zero datetime=True; CharSet=utf8" -e AWS_LAMBDA_FUNCTION_MEMORY_SIZE=128 lambci/lambda:dotnetcore2.1 get_profile::get_profile.Function::Get '{"resource":"/{proxy+}","path":"/","httpMethod":"GET","headers":null,"queryStringParameters":null,"pathParameters":{"id":"1"},"stageVariables":null,"requestContext":{"accountId":"AAAAAAAAAAAA","resourceId":"5agfss","stage":"test-invoke-stage","requestId":"test-invoke-request","identity":{"cognitoIdentityPoolId":null,"accountId":"AAAAAAAAAAAA","cognitoIdentityId":null,"caller":"BBBBBBBBBBBB","apiKey":"test-invoke-api-key","sourceIp":"test-invoke-source-ip","cognitoAuthenticationType":null,"cognitoAuthenticationProvider":null,"userArn":"arn:aws:iam::AAAAAAAAAAAA:root","userAgent":"Apache-HttpClient/4.5.x (Java/1.8.0_102)","user":"AAAAAAAAAAAA"},"resourcePath":"/{proxy+}","httpMethod":"GET","apiId":"t2yh6sjnmk"},"body":null}'</code></pre>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1200" height="138" src="https://myifew.com/wp-content/uploads/2019/02/lambda-docker-apiproxy-mysql-command-result1-1200x138.png" alt="" class="wp-image-5275" srcset="https://myifew.com/wp-content/uploads/2019/02/lambda-docker-apiproxy-mysql-command-result1-1200x138.png 1200w, https://myifew.com/wp-content/uploads/2019/02/lambda-docker-apiproxy-mysql-command-result1-1024x118.png 1024w, https://myifew.com/wp-content/uploads/2019/02/lambda-docker-apiproxy-mysql-command-result1-768x89.png 768w, https://myifew.com/wp-content/uploads/2019/02/lambda-docker-apiproxy-mysql-command-result1-700x81.png 700w, https://myifew.com/wp-content/uploads/2019/02/lambda-docker-apiproxy-mysql-command-result1-600x69.png 600w, https://myifew.com/wp-content/uploads/2019/02/lambda-docker-apiproxy-mysql-command-result1.png 2690w" sizes="auto, (max-width: 1200px) 100vw, 1200px" /></figure>



<h4 class="wp-block-heading">ตัวอย่างสอง สามารถใส่ event ด้านหน้า ในรูปแบบ stdin ได้</h4>



<p>โดยใช้การ echo ค่า JSON ที่เราต้องการส่งออกไป จากนั้นให้ ใส่ | คั่นคำสั่งรัน Container เพื่อรับไปใช้ พร้อมกับระบุ parameter เพิ่มเติมอีกหน่อยคือ</p>



<pre class="wp-block-code"><code>-i -e DOCKER_LAMBDA_USE_STDIN=1</code></pre>



<p>ตัวอย่าง</p>



<pre class="wp-block-code"><code>echo '{"resource":"/{proxy+}","path":"/","httpMethod":"GET","headers":null,"queryStringParameters":null,"pathParameters":{"id":"1"},"stageVariables":null,"requestContext":{"accountId":"AAAAAAAAAAAA","resourceId":"5agfss","stage":"test-invoke-stage","requestId":"test-invoke-request","identity":{"cognitoIdentityPoolId":null,"accountId":"AAAAAAAAAAAA","cognitoIdentityId":null,"caller":"BBBBBBBBBBBB","apiKey":"test-invoke-api-key","sourceIp":"test-invoke-source-ip","cognitoAuthenticationType":null,"cognitoAuthenticationProvider":null,"userArn":"arn:aws:iam::AAAAAAAAAAAA:root","userAgent":"Apache-HttpClient/4.5.x (Java/1.8.0_102)","user":"AAAAAAAAAAAA"},"resourcePath":"/{proxy+}","httpMethod":"GET","apiId":"t2yh6sjnmk"},"body":null}' | docker run --rm -v "$PWD":/var/task -i -e DOCKER_LAMBDA_USE_STDIN=1 --network mysql -e TEST_LAMBDA_DBCONNECTION="server=mysql;userid=root;password=1234;database=test_lambda;convert zero datetime=True; CharSet=utf8" -e AWS_LAMBDA_FUNCTION_MEMORY_SIZE=128 lambci/lambda:dotnetcore2.1 get_profile::get_profile.Function::Get</code></pre>



<h4 class="wp-block-heading" id="mce_97">ตัวอย่างสาม สามารถใส่ event ด้านหน้า ในรูปแบบ stdin ได้ โดยเรียกใช้ข้อมูลจากไฟล์</h4>



<p>แบบนี้เหมือนกับด้านบน เพียงแต่ผมแปลงนิดหน่อย ให้พ่นข้อมูลจากไฟล์แทน โดยใช้คำสั่ง cat เช่น</p>



<pre class="wp-block-code"><code>cat /Users/chitpong/Sourcecode/aws-serverless/profile/test/get_profile.Tests/SampleRequests/TestGetMethod.json | docker run --rm -v "$PWD":/var/task -i -e DOCKER_LAMBDA_USE_STDIN=1 --network mysql -e TEST_LAMBDA_DBCONNECTION="server=mysql;userid=root;password=1234;database=test_lambda;convert zero datetime=True; CharSet=utf8" -e AWS_LAMBDA_FUNCTION_MEMORY_SIZE=128 lambci/lambda:dotnetcore2.1 get_profile::get_profile.Function::Get</code></pre>



<p>ในที่นี้ผมใช้ JSON ไฟล์เดียวกันกับที่ผมเขียนทดสอบใน Unit Test</p>



<h2 class="wp-block-heading">สรุป</h2>



<p>เป็นรุปแบบการทดสอบ AWS Lambda แบบที่ไม่เสียเงินอีกรูปแบบหนึ่ง และค่อนข้างทำงานได้เหมือนจริงตามที่เราต้องการ ซึ่งถ้าใช้งานบ่อยๆ การทำคำสั่งแบบนี้อาจจะยาวและเสียเวลาหน่อย ก็ให้ไปเขียนเป็น shell script หรือ makefile เพื่อรันก็จะไวขึ้นอีกนิด</p>



<p>ทั้งนี้ทั้งนั้น ผมก็ยังยืนยันแบบเดิมว่า ทดสอบด้วยการเขียน Unit Test เถอะครับ</p>



<p><strong>Reference</strong></p>



<ul class="wp-block-list"><li>https://github.com/lambci/docker-lambda</li><li>https://github.com/lambci/docker-lambda/issues/23</li></ul>
]]></content:encoded>
					
					<wfw:commentRss>https://myifew.com/5257/how-to-test-aws-lambda-on-local-machine-via-lambda-docker/feed/</wfw:commentRss>
			<slash:comments>0</slash:comments>
		
		
			</item>
		<item>
		<title>4 วิธีจัดการ Versioning และ Environment ของ AWS Lambda Function โดยไม่ต้องแก้ไขโค้ด</title>
		<link>https://myifew.com/5231/4-ways-to-manage-aws-lambda-function-version-and-environment/</link>
					<comments>https://myifew.com/5231/4-ways-to-manage-aws-lambda-function-version-and-environment/#respond</comments>
		
		<dc:creator><![CDATA[iFew]]></dc:creator>
		<pubDate>Wed, 20 Feb 2019 04:23:29 +0000</pubDate>
				<category><![CDATA[Technology]]></category>
		<category><![CDATA[.Net Core]]></category>
		<category><![CDATA[AWS]]></category>
		<category><![CDATA[AWS Lambda]]></category>
		<category><![CDATA[Serverless]]></category>
		<category><![CDATA[Versioning]]></category>
		<guid isPermaLink="false">https://myifew.com/?p=5231</guid>

					<description><![CDATA[เป็นเรื่องสำคัญที่เรามักมองข้ามไป ว่าจะจัดการ Lambda Function อย่างไรให้ใช้ได้กับ Environment ต่างๆ เช่น Development, UAT, Production และต้องไม่กระทบกับการใช้งานของผู้ใช้ด้วย ดังนั้น สิ่งที่ต้องดูต่อ คือ เราต้องกำหนด Lambda Function&#8230;]]></description>
										<content:encoded><![CDATA[
<p>เป็นเรื่องสำคัญที่เรามักมองข้ามไป ว่าจะจัดการ Lambda Function อย่างไรให้ใช้ได้กับ Environment ต่างๆ เช่น Development, UAT, Production และต้องไม่กระทบกับการใช้งานของผู้ใช้ด้วย ดังนั้น สิ่งที่ต้องดูต่อ คือ เราต้องกำหนด Lambda Function Version ให้แต่ละ Environment ด้วยใช่หรือไม่?</p>



<p>จากที่ผมลองเล่นดู พบอยู่ 4 วิธี ที่กำลังจะเขียนแชร์ในบล็อกนี้เอง มันมีทั้งข้อดีและข้อเสีย ซึ่งผู้อ่านคงต้องนำไปพิจารณาและตัดสินใจใช้ตามความเหมาะสมกันเอง</p>



<span id="more-5231"></span>



<p class="has-background has-very-light-gray-background-color"><strong>คำแนะนำ: วิธีการที่ผมเสนอในบล็อกนี้ จะไม่ได้พูดถึงวิธีการที่ต้องมีการแก้โค้ดนะครับ รวมไปถึง ผู้อ่านควรต้องเข้าใจเรื่อง AWS API Gateway และ AWS Lambda อยู่แล้ว</strong></p>



<h2 class="wp-block-heading">1. ปรับ Version ที่ Integration Request ใน API Gateway </h2>



<p><strong>วิธีนี้เหมาะสำหรับ</strong></p>



<ul class="wp-block-list"><li>ผู้ใช้ API Gateway เป็น Event Trigger เรียกใช้งาน Lambda Function</li><li>มี Lambda Function จำนวนน้อย</li><li>ทำไว้ใช้บน Test Environment เท่านั้น</li><li>ไม่ได้ต้องการทำ Version ให้กับ Lambda Function ทุกตัว </li><li>ถ้าใช้ AWS API Gateway วิธีการนี้ จะสามารถใช้งานสิ่งที่เราปรับเปลี่ยนได้ก็ต่อเมื่อมีการกดปุ่ม Deploy API</li></ul>



<p><strong>ปัญหาที่พบในวิธีนี้</strong></p>



<ul class="wp-block-list"><li>ถ้าไม่ใช้ API Gateway จะทำแบบนี้ไม่ได้</li><li>ถ้ามี Lambda Function จำนวนมากๆ ต้องไล่เปลี่ยนทีละตัว ก็จะเป็นงานหนักมาก</li></ul>



<p>วิธีนี้เป็นวิธี basic ที่สุดสำหรับ ผู้ใช้ API Gateway เป็น Event Trigger เรียกใช้งาน Lambda Function ซึ่งตอนผมลองเล่นแรกๆก็ทำแบบนี้แหละ</p>



<p>ในตัวอย่าง บนหน้าของ AWS API Gateway เมื่อเราสร้าง Resource และ Method แล้ว จะมีส่วนของ Integration Request</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1200" height="565" src="https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-gw-pointer-1-1200x565.png" alt="" class="wp-image-5234" srcset="https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-gw-pointer-1-1200x565.png 1200w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-gw-pointer-1-1024x482.png 1024w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-gw-pointer-1-768x362.png 768w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-gw-pointer-1-700x330.png 700w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-gw-pointer-1-600x283.png 600w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-gw-pointer-1.png 2870w" sizes="auto, (max-width: 1200px) 100vw, 1200px" /></figure>



<p>ในหน้าของ Integration Request เราสามารถแก้ไขการเรียกใช้งาน Lambda Function โดยระบุ Version ได้จากที่นี่เลย</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1200" height="797" src="https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-gw-pointer-2-1200x797.png" alt="" class="wp-image-5235" srcset="https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-gw-pointer-2-1200x797.png 1200w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-gw-pointer-2-1024x680.png 1024w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-gw-pointer-2-768x510.png 768w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-gw-pointer-2-700x465.png 700w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-gw-pointer-2-600x398.png 600w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-gw-pointer-2.png 1684w" sizes="auto, (max-width: 1200px) 100vw, 1200px" /></figure>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1200" height="294" src="https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-gw-pointer-3-1200x294.png" alt="" class="wp-image-5236" srcset="https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-gw-pointer-3-1200x294.png 1200w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-gw-pointer-3-1024x251.png 1024w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-gw-pointer-3-768x188.png 768w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-gw-pointer-3-700x171.png 700w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-gw-pointer-3-600x147.png 600w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-gw-pointer-3.png 1796w" sizes="auto, (max-width: 1200px) 100vw, 1200px" /></figure>



<h2 class="wp-block-heading">2. สร้าง Version URL ใน API Gateway</h2>



<p><strong>วิธีนี้เหมาะสำหรับ</strong></p>



<ul class="wp-block-list"><li>ผู้ใช้ API Gateway เป็น Event Trigger เรียกใช้งาน Lambda Function</li><li>ต้องการให้ผู้ใช้ API เดิมยังใช้งานได้ตามปกติ และผู้ใช้ใหม่ก็สามารถใช้ API ใหม่ได้ (หรือจะกำหนด Permission ของกลุ่มผู้ใช้เพิ่มด้วยก็ได้)</li><li>มี Lambda Function จำนวนไม่มากนัก</li><li>สามารถ Custom Lambda ในแต่ละ URL Version ได้ โดยไม่กระทบกับ URL Version อื่น</li><li>ถ้าใช้ AWS API Gateway วิธีการนี้ จะสามารถใช้งานสิ่งที่เราปรับเปลี่ยนได้ก็ต่อเมื่อมีการกดปุ่ม Deploy API</li></ul>



<p><strong>ปัญหาที่พบในวิธีนี้</strong></p>



<ul class="wp-block-list"><li>ถ้าไม่ใช้ API Gateway หรือ Middleware ที่สามารถจัดการ Router ได้ จะทำแบบนี้ไม่ได้</li><li>จะงานหนักทันที ถ้ามี Lambda Function จำนวนมากๆ และต้องสร้าง URL Gateway ใหม่อีกชุดหนึ่ง และกำหนด Version ใหม่ (แต่ใช้ Tools จำพวก <a rel="noreferrer noopener" aria-label="Infrastructure as code (opens in a new tab)" href="https://en.wikipedia.org/wiki/Infrastructure_as_code" target="_blank">Infrastructure as code</a> ช่วยปั๊มได้)</li></ul>



<p>วิธีนี้เพิ่มมิติของการทำ Versioning โดยการกำหนด URL Version ไปด้วย ทำให้แยกกลุ่มของผู้ใช้งานได้ โดยไม่กระทบกัน</p>



<p>มีหลายแบรนด์ดังที่ใช้วิธีการแบบนี้ เช่น</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1200" height="981" src="https://myifew.com/wp-content/uploads/2019/02/Screen-Shot-2562-02-20-at-12.25.05-1200x981.png" alt="" class="wp-image-5237" srcset="https://myifew.com/wp-content/uploads/2019/02/Screen-Shot-2562-02-20-at-12.25.05-1200x981.png 1200w, https://myifew.com/wp-content/uploads/2019/02/Screen-Shot-2562-02-20-at-12.25.05-1024x837.png 1024w, https://myifew.com/wp-content/uploads/2019/02/Screen-Shot-2562-02-20-at-12.25.05-768x628.png 768w, https://myifew.com/wp-content/uploads/2019/02/Screen-Shot-2562-02-20-at-12.25.05-700x572.png 700w, https://myifew.com/wp-content/uploads/2019/02/Screen-Shot-2562-02-20-at-12.25.05-600x491.png 600w, https://myifew.com/wp-content/uploads/2019/02/Screen-Shot-2562-02-20-at-12.25.05.png 1416w" sizes="auto, (max-width: 1200px) 100vw, 1200px" /><figcaption>อ้างอิง: https://mathieu.fenniak.net/aint-nobody-got-time-for-that-api-versioning/</figcaption></figure>



<p>ในตัวอย่าง บนหน้าของ AWS API Gateway คือให้เราสร้าง Resource เป็น Version ไว้เลย</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1200" height="566" src="https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-gw-url-1-1200x566.png" alt="" class="wp-image-5238" srcset="https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-gw-url-1-1200x566.png 1200w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-gw-url-1-1024x483.png 1024w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-gw-url-1-768x362.png 768w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-gw-url-1-700x330.png 700w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-gw-url-1-600x283.png 600w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-gw-url-1.png 2874w" sizes="auto, (max-width: 1200px) 100vw, 1200px" /></figure>



<p>จากนั้นในแต่ละ Method ใน Resource Version ต่างๆ ให้เรากำหนด Version ใน Integration Request เหมือนเดิมแบบวิธีแรก</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1200" height="565" src="https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-gw-url-2-1200x565.png" alt="" class="wp-image-5239" srcset="https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-gw-url-2-1200x565.png 1200w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-gw-url-2-1024x482.png 1024w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-gw-url-2-768x362.png 768w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-gw-url-2-700x330.png 700w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-gw-url-2-600x282.png 600w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-gw-url-2.png 2872w" sizes="auto, (max-width: 1200px) 100vw, 1200px" /></figure>



<h2 class="wp-block-heading">3. สร้าง Version และกำหนด Alias ให้กับ Lambda Function  </h2>



<p><strong>วิธีนี้เหมาะสำหรับ</strong></p>



<ul class="wp-block-list"><li>ทุกคนที่จะนำ Lambda Function ไปใช้งาน</li><li>มี Lambda Function จำนวนน้อยจนถึงมากๆ </li><li>ใช้ได้กับทุก Environment ไม่ว่าจะ Test หรือ Production</li><li>มีการกำหนด Version ให้กับ Lambda Function อยู่เสมอ</li></ul>



<p><strong>ปัญหาที่พบในวิธีนี้</strong></p>



<ul class="wp-block-list"><li>ต้องระบุ Description ใน Version ให้ดี เพื่อที่จะได้รู้ว่า แต่ละตัว มีการปรับเปลี่ยนอะไร</li><li>การเปลี่ยน Verion ใน Alias Name จะทำให้การทำงานไปใช้ Lambda Version นั้นทันที ไม่หมือนกับตอนทำกับ API Gateway ที่ทดสอบก่อนได้ แล้วค่อยกดปุ่ม Deploy API (แต่ทั้งนีก็สามารถ Rollback ได้ทันที)</li></ul>



<p>เป็นวิธีที่ซับซ้อนขึ้นอีกนิดจากข้างต้น แต่มีประสิทธิภาพ และทาง Amazon เองก็แนะนำให้ทำแบบนี้ (<a rel="noreferrer noopener" aria-label="อ้างอิง (opens in a new tab)" href="https://docs.aws.amazon.com/lambda/latest/dg/versioning-aliases.html" target="_blank">AWS Lambda Function Versioning and Aliases</a>)</p>



<p>โปรแกรมเมอร์เองจะต้องมีการกำหนดเวอร์ชั่นภายในทีมให้ชัดเจนว่า เมื่อไรถึงจะต้อง publish version ออกมา และนำไปลิงค์เข้าใน Alias Name เพราะการกำหนดแบบนี้จะทำให้การทำงานของ lambda เปลี่ยนทันที ไม่มีการผ่าน Deploy API</p>



<div class="wp-block-image"><figure class="aligncenter"><img loading="lazy" decoding="async" width="377" height="360" src="https://myifew.com/wp-content/uploads/2019/02/alias_intro_2_10.png" alt="" class="wp-image-5249"/><figcaption>ตัวอย่างมี 2 Environment และเมื่อทำบน Development Env. เสร็จ ต้องการนำไปใช้บน Prodution Env.<br>อ้างอิง https://docs.aws.amazon.com/lambda/latest/dg/versioning-aliases.html</figcaption></figure></div>



<div class="wp-block-image"><figure class="aligncenter"><img loading="lazy" decoding="async" width="564" height="363" src="https://myifew.com/wp-content/uploads/2019/02/alias_scenario_2_30.png" alt="" class="wp-image-5252"/><figcaption>ตัวอย่างมี 3 Environment และปรับให้มีการใช้ Lambda ร่วมกัน<br>อ้างอิง https://docs.aws.amazon.com/lambda/latest/dg/aliases-intro.html</figcaption></figure></div>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="569" height="358" src="https://myifew.com/wp-content/uploads/2019/02/alias_scenario_2_40.png" alt="" class="wp-image-5253"/><figcaption>ตัวอย่างมี 3 Environment และมีการใช้ Lambda Version ของใครของมัน (การทำงานจริง น่าจะเป็นภาพนี้)<br>อ้างอิง https://docs.aws.amazon.com/lambda/latest/dg/aliases-intro.html</figcaption></figure>



<p>วิธีการทำ คือ เมื่อเรามี Version ของ Lambda อยู่แล้ว ให้ทำการสร้าง Alias Name และทำการเชื่อมกันว่า ชื่อ Alias ไหน ชี้ไป Version ไหน</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="908" height="726" src="https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-4.png" alt="" class="wp-image-5242" srcset="https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-4.png 908w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-4-768x614.png 768w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-4-700x560.png 700w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-4-600x480.png 600w" sizes="auto, (max-width: 908px) 100vw, 908px" /></figure>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="892" height="768" src="https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-5.png" alt="" class="wp-image-5243" srcset="https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-5.png 892w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-5-768x661.png 768w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-5-700x603.png 700w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-5-600x517.png 600w" sizes="auto, (max-width: 892px) 100vw, 892px" /></figure>



<p>ถ้าต้องการเปลี่ยน Version ก็สามารถคลิกที่ชื่อ Alias แล้วเลื่อนลงมาแก้ไข Config ได้ภายหลัง, รูปนี้ตัวอย่างของ Alias ชื่อ Prod ชี้ไปหาฟังก์ชั่น simple-hello Version 1</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1200" height="777" src="https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-6-1200x777.png" alt="" class="wp-image-5244" srcset="https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-6-1200x777.png 1200w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-6-1024x663.png 1024w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-6-768x498.png 768w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-6-700x453.png 700w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-6-600x389.png 600w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-6.png 1908w" sizes="auto, (max-width: 1200px) 100vw, 1200px" /></figure>



<p>รูปนี้ตัวอย่างของ Alias ชื่อ Dev ชี้ไปหาฟังก์ชั่น simple-hello Version $LATEST</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1200" height="817" src="https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-7-1200x817.png" alt="" class="wp-image-5245" srcset="https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-7-1200x817.png 1200w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-7-1024x697.png 1024w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-7-768x523.png 768w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-7-700x476.png 700w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-7-600x408.png 600w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-7.png 1860w" sizes="auto, (max-width: 1200px) 100vw, 1200px" /></figure>



<p>ในตัวอย่าง บนหน้าของ AWS API Gateway คือให้เราสร้าง Resource และกำหนดให้ Integration Request ของ Method ชี้ไปหา Alias ชื่อว่า Prod (แต่เดิมเราชี้ไปหาเลข Version, ตอนนี้ชี้ไปหาชื่อ Alias แทน) จากนั้นผม Deploy API ไปหา prod stage รอบหนึ่งก่อน เพื่อให้จำค่านั้น แล้วผมก็กลับมาเปลี่ยน Method ให้ชี้ไปหา Alias ชื่อว่า Dev และทำการ Deploy API ไปหา dev stage  อีกครั้ง</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1200" height="567" src="https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-1-1200x567.png" alt="" class="wp-image-5241" srcset="https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-1-1200x567.png 1200w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-1-1024x484.png 1024w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-1-768x363.png 768w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-1-700x331.png 700w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-1-600x283.png 600w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-1.png 2870w" sizes="auto, (max-width: 1200px) 100vw, 1200px" /></figure>



<p>ผลที่ได้จากการทำวิธีนี้</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1200" height="212" src="https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-2-1200x212.png" alt="" class="wp-image-5247" srcset="https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-2-1200x212.png 1200w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-2-1024x181.png 1024w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-2-768x136.png 768w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-2-700x124.png 700w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-2-600x106.png 600w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-2.png 2180w" sizes="auto, (max-width: 1200px) 100vw, 1200px" /></figure>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="1200" height="194" src="https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-3-1-1200x194.png" alt="" class="wp-image-5248" srcset="https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-3-1-1200x194.png 1200w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-3-1-1024x166.png 1024w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-3-1-768x124.png 768w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-3-1-700x113.png 700w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-3-1-600x97.png 600w, https://myifew.com/wp-content/uploads/2019/02/lambda-version-by-lambda-alias-3-1.png 2176w" sizes="auto, (max-width: 1200px) 100vw, 1200px" /></figure>



<h2 class="wp-block-heading" id="mce_42">4. สร้าง Version และกำหนด Alias ให้กับ Lambda Function พร้อมกับทำ URL Verioning</h2>



<p>วิธีนี้คือการผสมข้อ 3 และ 4 เข้าด้วยกัน กล่าวคือ ใช้ Alias Name ด้วย เพื่อให้แก้ไข Version ที่ตัว Lambda Function ได้เองเลย พร้อมกับทำ URL กำหนด Version ไว้ เพื่อป้องกันผลกระทบ กรณี เปลี่ยน Version ใน Lambda Function ผิด</p>



<h2 class="wp-block-heading">สรุป</h2>



<p>วิธีการระบุ API Version มีหลายวิธีการให้เลือกใช้ พยายามหา Best Practice ก็อาจจะเจอหลายๆแบบ แล้วแต่ว่าคนเขียนเคยทำแบบไหนแล้วเหมาะกับงานของเขา</p>



<h2 class="wp-block-heading">แถม</h2>



<p>อย่างที่ผมกล่าวข้างต้น ในบล็อกนี้มันเป็นวิธีทำที่เราไม่ต้องไปแก้ไขโค้ดใดๆ  ส่วนวิธีการอย่างอื่น ก็มีอีกหลายแบบ ซึ่งลองดูรูปด้านล่างประกอบได้ครับ</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="893" height="305" src="https://myifew.com/wp-content/uploads/2019/02/API-Versioning-Practices.png" alt="" class="wp-image-5251" srcset="https://myifew.com/wp-content/uploads/2019/02/API-Versioning-Practices.png 893w, https://myifew.com/wp-content/uploads/2019/02/API-Versioning-Practices-768x262.png 768w, https://myifew.com/wp-content/uploads/2019/02/API-Versioning-Practices-700x239.png 700w, https://myifew.com/wp-content/uploads/2019/02/API-Versioning-Practices-600x205.png 600w" sizes="auto, (max-width: 893px) 100vw, 893px" /></figure>



<p>ส่วนข้อดีข้อเสีย มีเทียบไว้ให้เข้าใจง่ายๆ</p>



<figure class="wp-block-image"><img loading="lazy" decoding="async" width="712" height="217" src="https://myifew.com/wp-content/uploads/2019/02/API_1.png" alt="" class="wp-image-5255" srcset="https://myifew.com/wp-content/uploads/2019/02/API_1.png 712w, https://myifew.com/wp-content/uploads/2019/02/API_1-700x213.png 700w, https://myifew.com/wp-content/uploads/2019/02/API_1-600x183.png 600w" sizes="auto, (max-width: 712px) 100vw, 712px" /><figcaption>อ้างอิง https://www.icapps.com/blog/api-versioning-overview</figcaption></figure>



<p>อ้างอิง</p>



<ul class="wp-block-list"><li>https://docs.aws.amazon.com/lambda/latest/dg/versioning-aliases.html</li><li>https://docs.aws.amazon.com/lambda/latest/dg/aliases-intro.html</li><li>https://mathieu.fenniak.net/aint-nobody-got-time-for-that-api-versioning/</li><li>https://www.icapps.com/blog/api-versioning-overview</li></ul>
]]></content:encoded>
					
					<wfw:commentRss>https://myifew.com/5231/4-ways-to-manage-aws-lambda-function-version-and-environment/feed/</wfw:commentRss>
			<slash:comments>0</slash:comments>
		
		
			</item>
	</channel>
</rss>
