เคล็ดลับและคำแนะนำ

เอกสารนี้อธิบายแนวทางปฏิบัติแนะนำสำหรับการออกแบบ การติดตั้งใช้งาน การทดสอบ และการติดตั้งใช้งาน Cloud Functions

ความถูกต้อง

ส่วนนี้อธิบายแนวทางปฏิบัติแนะนำทั่วไปสำหรับการออกแบบและการติดตั้งใช้งาน Cloud Functions

เขียนฟังก์ชันไอดีมโพเทนต์

ฟังก์ชันควรให้ผลลัพธ์เดียวกันแม้ว่าจะมีการเรียกใช้หลายครั้งก็ตาม ซึ่งจะช่วยให้คุณลองเรียกใช้ฟังก์ชันอีกครั้งได้หากการเรียกใช้ก่อนหน้าล้มเหลว ระหว่างการดำเนินการในโค้ด ดูข้อมูลเพิ่มเติมได้ที่ ลองฟังก์ชันที่ขับเคลื่อนด้วยเหตุการณ์อีกครั้ง

อย่าเริ่มกิจกรรมในเบื้องหลัง

กิจกรรมในเบื้องหลังคือสิ่งที่เกิดขึ้นหลังจากฟังก์ชันสิ้นสุด การเรียกใช้ฟังก์ชันจะสิ้นสุดเมื่อฟังก์ชันแสดงผลหรือส่งสัญญาณ การเสร็จสมบูรณ์ เช่น โดยการเรียกอาร์กิวเมนต์ callback ในฟังก์ชันที่ขับเคลื่อนด้วยเหตุการณ์ของ Node.js โค้ดที่เรียกใช้หลังการสิ้นสุดอย่างราบรื่นจะเข้าถึง CPU ไม่ได้และ จะไม่มีความคืบหน้าใดๆ

นอกจากนี้ เมื่อมีการเรียกใช้ครั้งต่อๆ ไปในสภาพแวดล้อมเดียวกัน กิจกรรมในเบื้องหลังจะกลับมาทำงานต่อ ซึ่งจะรบกวนการเรียกใช้ครั้งใหม่ ซึ่งอาจ ทําให้เกิดลักษณะการทํางานและข้อผิดพลาดที่ไม่คาดคิดซึ่งวินิจฉัยได้ยาก การเข้าถึง เครือข่ายหลังจากฟังก์ชันสิ้นสุดลงมักจะทำให้การเชื่อมต่อถูกรีเซ็ต (รหัสข้อผิดพลาด ECONNRESET)

โดยทั่วไปแล้ว ระบบจะตรวจหากิจกรรมในเบื้องหลังได้ในบันทึกจากการเรียกใช้แต่ละครั้ง โดยค้นหาสิ่งที่บันทึกไว้หลังจากบรรทัดที่ระบุว่าการเรียกใช้ เสร็จสิ้นแล้ว บางครั้งกิจกรรมในเบื้องหลังอาจซ่อนอยู่ลึกลงไปในโค้ด โดยเฉพาะเมื่อมีการดำเนินการแบบไม่พร้อมกัน เช่น การเรียกกลับหรือตัวจับเวลา ตรวจสอบโค้ดเพื่อให้แน่ใจว่าการดำเนินการแบบไม่พร้อมกันทั้งหมดเสร็จสมบูรณ์ก่อนที่จะ สิ้นสุดฟังก์ชัน

ลบไฟล์ชั่วคราวเสมอ

พื้นที่เก็บข้อมูลในดิสก์ในเครื่องในไดเรกทอรีชั่วคราวคือระบบไฟล์ในหน่วยความจำ ไฟล์ ที่คุณเขียนจะใช้หน่วยความจำที่ฟังก์ชันของคุณใช้ได้ และบางครั้งจะยังคงอยู่ ระหว่างการเรียกใช้ การไม่ลบไฟล์เหล่านี้อย่างชัดเจนอาจส่งผลให้เกิดข้อผิดพลาดเกี่ยวกับหน่วยความจำไม่เพียงพอและเริ่มระบบใหม่ในภายหลัง

คุณดูหน่วยความจำที่ฟังก์ชันแต่ละรายการใช้ได้โดยเลือกฟังก์ชันในรายการฟังก์ชันในคอนโซล Google Cloud แล้วเลือกพล็อตการใช้หน่วยความจำ

หากต้องการเข้าถึงพื้นที่เก็บข้อมูลระยะยาว ให้ลองใช้Cloud Run การติดตั้งโวลุ่มด้วย Cloud Storage หรือโวลุ่ม NFS

คุณลดข้อกำหนดด้านหน่วยความจำเมื่อประมวลผลไฟล์ขนาดใหญ่ได้โดยใช้การไปป์ไลน์ เช่น คุณสามารถประมวลผลไฟล์ใน Cloud Storage ได้โดยการสร้างสตรีมการอ่าน ส่งผ่านกระบวนการที่อิงตามสตรีม และเขียนสตรีมเอาต์พุต ไปยัง Cloud Storage โดยตรง

Functions Framework

เราขอแนะนำให้คุณรวมไลบรารี Functions Framework ไว้ในเครื่องมือจัดการแพ็กเกจและตรึงการอ้างอิงไว้กับ Functions Framework เวอร์ชันใดเวอร์ชันหนึ่ง เพื่อให้มั่นใจว่ามีการติดตั้งการอ้างอิงเดียวกันอย่างสม่ำเสมอในสภาพแวดล้อมต่างๆ

โดยให้ระบุเวอร์ชันที่ต้องการในไฟล์ล็อกที่เกี่ยวข้อง (เช่น package-lock.json สำหรับ Node.js หรือ requirements.txt สำหรับ Python)

หากไม่ได้ระบุ Functions Framework เป็นการพึ่งพาอย่างชัดเจน ระบบจะเพิ่ม Framework โดยอัตโนมัติในระหว่างกระบวนการสร้างโดยใช้เวอร์ชันล่าสุดที่มี

เครื่องมือ

ส่วนนี้จะให้หลักเกณฑ์เกี่ยวกับวิธีใช้เครื่องมือเพื่อติดตั้งใช้งาน ทดสอบ และ โต้ตอบกับ Cloud Functions

การพัฒนาในพื้นที่

การติดตั้งใช้งานฟังก์ชันจะใช้เวลาสักครู่ ดังนั้นการทดสอบโค้ด ของฟังก์ชันในเครื่องจึงมักจะเร็วกว่า

นักพัฒนาแอป Firebase สามารถใช้โปรแกรมจำลอง Firebase CLI Cloud Functions ได้

หลีกเลี่ยงการหมดเวลาในการติดตั้งใช้งานระหว่างการเริ่มต้น

หากการติดตั้งใช้งานฟังก์ชันล้มเหลวเนื่องจากข้อผิดพลาดการหมดเวลา แสดงว่าโค้ดขอบเขตส่วนกลางของฟังก์ชันใช้เวลานานเกินไปในการดำเนินการระหว่างกระบวนการติดตั้งใช้งาน

Firebase CLI มีการหมดเวลาเริ่มต้นสำหรับการค้นหาฟังก์ชันของคุณในระหว่างการ ติดตั้งใช้งาน หากตรรกะการเริ่มต้นในซอร์สโค้ดของฟังก์ชัน (การโหลดโมดูล การเรียกเครือข่าย และอื่นๆ) เกินระยะหมดเวลานี้ การทําให้ใช้งานได้อาจล้มเหลว

หากต้องการหลีกเลี่ยงการหมดเวลา ให้ใช้กลยุทธ์ใดกลยุทธ์หนึ่งต่อไปนี้

ใช้ฮุก onInit() เพื่อหลีกเลี่ยงการเรียกใช้โค้ดการเริ่มต้นระหว่างการ ติดตั้งใช้งาน โค้ดภายในฮุก onInit() จะทำงานเมื่อมีการติดตั้งใช้งานฟังก์ชันใน Cloud Run เท่านั้น ไม่ใช่ในระหว่างกระบวนการติดตั้งใช้งาน

Node.js

const { onInit } = require('firebase-functions/v2/core');
const { onRequest } = require('firebase-functions/v2/https');

// Example of a slow initialization task
function slowInitialization() {
  // Simulate a long-running operation (e.g., loading a large model, network request).
  return new Promise(resolve => {
      setTimeout(() => {
          console.log("Slow initialization complete");
          resolve("Initialized Value");
      }, 20000); // Simulate a 20-second delay
  });
}
let initializedValue;

onInit(async () => {
  initializedValue = await slowInitialization();
});

exports.myFunction = onRequest((req, res) => {
  // Access the initialized value. It will be ready after the first invocation.
  res.send(`Value: ${initializedValue}`);
});

Python

from firebase_functions.core import init
from firebase_functions import https_fn
import time

# Example of a slow initialization task
def _slow_initialization():
  time.sleep(20)  # Simulate a 20-second delay
  print("Slow initialization complete")
  return "Initialized Value"

_initialized_value = None

@init
def initialize():
  global _initialized_value
  _initialized_value = _slow_initialization()

@https_fn.on_request()
def my_function(req: https_fn.Request) -> https_fn.Response:
  # Access the initialized value. It will be ready after the first invocation.
  return https_fn.Response(f"Value: {_initialized_value}")

(ทางเลือก) เพิ่มระยะหมดเวลาการค้นหา

หากปรับโครงสร้างโค้ดเพื่อใช้ onInit() ไม่ได้ คุณสามารถเพิ่มการหมดเวลาในการติดตั้งใช้งานของ CLI ได้โดยใช้ตัวแปรสภาพแวดล้อม FUNCTIONS_DISCOVERY_TIMEOUT ดังนี้

$ export FUNCTIONS_DISCOVERY_TIMEOUT=30
$ firebase deploy --only functions

ใช้ Sendgrid เพื่อส่งอีเมล

Cloud Functions ไม่อนุญาตการเชื่อมต่อขาออกในพอร์ต 25 คุณจึงไม่สามารถ ทำการเชื่อมต่อที่ไม่ปลอดภัยกับเซิร์ฟเวอร์ SMTP วิธีที่แนะนำในการส่งอีเมลคือการใช้บริการของบุคคลที่สาม เช่น SendGrid คุณดูตัวเลือกอื่นๆ สำหรับการส่งอีเมลได้ในบทแนะนำ การส่งอีเมลจากอินสแตนซ์ สำหรับ Google Compute Engine

ประสิทธิภาพ

ส่วนนี้จะอธิบายแนวทางปฏิบัติแนะนำในการเพิ่มประสิทธิภาพ

หลีกเลี่ยงการทำงานพร้อมกันต่ำ

เนื่องจาก Cold Start มีค่าใช้จ่ายสูง การนำอินสแตนซ์ที่เพิ่งเริ่มต้นมาใช้ซ้ำในช่วงที่มีการใช้งานสูงสุดจึงเป็นการเพิ่มประสิทธิภาพที่ยอดเยี่ยมในการจัดการภาระงาน การจำกัด การทำงานพร้อมกันจะจำกัดวิธีใช้ประโยชน์จากอินสแตนซ์ที่มีอยู่ จึงทำให้เกิด Cold Start มากขึ้น

การเพิ่มการทำงานพร้อมกัน ช่วยเลื่อนคำขอหลายรายการต่ออินสแตนซ์ ทำให้จัดการการเพิ่มขึ้นของภาระงานได้ง่ายขึ้น

ใช้การขึ้นต่อกันอย่างชาญฉลาด

เนื่องจากฟังก์ชันไม่มีสถานะ ระบบจึงมักเริ่มต้นสภาพแวดล้อมการดำเนินการ ตั้งแต่ต้น (ในระหว่างที่เรียกว่าการเริ่มแบบเย็น) เมื่อเกิดการเริ่มระบบใหม่ ระบบจะประเมินบริบทส่วนกลางของฟังก์ชัน

หากฟังก์ชันนำเข้าโมดูล เวลาในการโหลดโมดูลเหล่านั้นอาจเพิ่มเวลาในการตอบสนองของการเรียกใช้ระหว่างการเริ่มแบบเย็น คุณสามารถลดเวลาในการตอบสนองนี้ รวมถึง เวลาที่ต้องใช้ในการทําให้ฟังก์ชันใช้งานได้โดยการโหลดการอ้างอิงอย่างถูกต้องและ ไม่โหลดการอ้างอิงที่ฟังก์ชันไม่ได้ใช้

ใช้ตัวแปรส่วนกลางเพื่อนำออบเจ็กต์กลับมาใช้ใหม่ในการเรียกใช้ในอนาคต

เราไม่รับประกันว่าสถานะของฟังก์ชันจะ ยังคงอยู่สำหรับการเรียกใช้ในอนาคต อย่างไรก็ตาม Cloud Functions มักจะนำ สภาพแวดล้อมการดำเนินการของการเรียกใช้ก่อนหน้ากลับมาใช้ใหม่ หากประกาศตัวแปรในขอบเขตส่วนกลาง คุณจะนำค่าของตัวแปรนั้นมาใช้ซ้ำในการเรียกใช้ครั้งต่อๆ ไปได้โดยไม่ต้องคำนวณใหม่

วิธีนี้ช่วยให้คุณแคชออบเจ็กต์ที่อาจมีค่าใช้จ่ายสูงในการสร้างใหม่ทุกครั้งที่เรียกใช้ฟังก์ชันได้ การย้ายออบเจ็กต์ดังกล่าวจากเนื้อหาของฟังก์ชันไปยังขอบเขตส่วนกลาง อาจส่งผลให้ประสิทธิภาพดีขึ้นอย่างมาก ตัวอย่างต่อไปนี้ สร้างออบเจ็กต์ที่มีขนาดใหญ่เพียงครั้งเดียวต่ออินสแตนซ์ของฟังก์ชัน และแชร์ออบเจ็กต์ดังกล่าวในการเรียกใช้ฟังก์ชันทั้งหมด ที่เข้าถึงอินสแตนซ์ที่ระบุ

Node.js

console.log('Global scope');
const perInstance = heavyComputation();
const functions = require('firebase-functions');

exports.function = functions.https.onRequest((req, res) => {
  console.log('Function invocation');
  const perFunction = lightweightComputation();

  res.send(`Per instance: ${perInstance}, per function: ${perFunction}`);
});

Python

import time

from firebase_functions import https_fn

# Placeholder
def heavy_computation():
  return time.time()

# Placeholder
def light_computation():
  return time.time()

# Global (instance-wide) scope
# This computation runs at instance cold-start
instance_var = heavy_computation()

@https_fn.on_request()
def scope_demo(request):

  # Per-function scope
  # This computation runs every time this function is called
  function_var = light_computation()
  return https_fn.Response(f"Instance: {instance_var}; function: {function_var}")
  

ฟังก์ชัน HTTP นี้ใช้ออบเจ็กต์คำขอ (flask.Request) และแสดงผลข้อความตอบกลับ หรือชุดค่าใดก็ได้ที่สามารถเปลี่ยนเป็นออบเจ็กต์ Response ได้โดยใช้ make_response

การแคชการเชื่อมต่อเครือข่าย การอ้างอิงไลบรารี และออบเจ็กต์ไคลเอ็นต์ API ในขอบเขตส่วนกลางมีความสำคัญเป็นอย่างยิ่ง ดูตัวอย่างได้ที่การเพิ่มประสิทธิภาพเครือข่าย

ลด Cold Start โดยการกำหนดจำนวนอินสแตนซ์ขั้นต่ำ

โดยค่าเริ่มต้น Cloud Functions จะปรับขนาดจำนวนอินสแตนซ์ตาม จำนวนคำขอขาเข้า คุณเปลี่ยนลักษณะการทำงานเริ่มต้นนี้ได้โดยการตั้งค่า จำนวนอินสแตนซ์ขั้นต่ำที่ Cloud Functions ต้องพร้อม ให้บริการคำขอ การตั้งค่าจำนวนอินสแตนซ์ขั้นต่ำจะช่วยลด Cold Start ของ แอปพลิเคชัน เราขอแนะนำให้ตั้งค่าจำนวนอินสแตนซ์ขั้นต่ำและ ทำการเริ่มต้นให้เสร็จสมบูรณ์ในเวลาโหลด หากแอปพลิเคชันของคุณมีความไวต่อเวลาในการตอบสนอง

ดูข้อมูลเพิ่มเติมเกี่ยวกับตัวเลือกขณะรันไทม์เหล่านี้ได้ที่ ควบคุมลักษณะการทำงานของการปรับขนาด

หมายเหตุเกี่ยวกับการเริ่มต้นแบบ Cold Start และการเริ่มต้น

การเริ่มต้นส่วนกลางจะเกิดขึ้นในเวลาโหลด หากไม่มีการอุ่นเครื่อง คำขอแรกจะต้องเริ่มต้นและโหลดโมดูลให้เสร็จสมบูรณ์ ซึ่งจะทำให้เกิดเวลาในการตอบสนองที่สูงขึ้น

อย่างไรก็ตาม การเริ่มต้นใช้งานทั่วโลกก็ส่งผลต่อ Cold Start ด้วย หากต้องการลดผลกระทบนี้ ให้เริ่มต้นเฉพาะสิ่งที่จำเป็นสำหรับคำขอแรก เพื่อให้เวลาในการตอบสนองของคำขอแรกต่ำที่สุด

ซึ่งมีความสำคัญอย่างยิ่งหากคุณกำหนดค่าอินสแตนซ์ขั้นต่ำตามที่อธิบายไว้ข้างต้นสำหรับฟังก์ชันที่คำนึงถึงเวลาในการตอบสนอง ใน สถานการณ์ดังกล่าว การเริ่มต้นให้เสร็จสมบูรณ์ในเวลาโหลดและการแคชข้อมูลที่เป็นประโยชน์ จะช่วยให้มั่นใจได้ว่าคำขอแรกไม่จำเป็นต้องดำเนินการดังกล่าวและจะแสดงโดยมีเวลาในการตอบสนองต่ำ

หากคุณเริ่มต้นตัวแปรในขอบเขตส่วนกลาง เวลาเริ่มต้นที่นานอาจส่งผลให้เกิดลักษณะการทำงาน 2 อย่างต่อไปนี้ ทั้งนี้ขึ้นอยู่กับภาษา - สำหรับชุดค่าผสมของภาษาและไลบรารีแบบไม่พร้อมกันบางชุด เฟรมเวิร์กฟังก์ชัน จะทำงานแบบไม่พร้อมกันและกลับมาทันที ซึ่งทำให้โค้ดทำงานต่อไปใน เบื้องหลัง และอาจทำให้เกิดปัญหาต่างๆ เช่น เข้าถึง CPU ไม่ได้ คุณควรบล็อกการเริ่มต้นโมดูลตามที่อธิบายไว้ด้านล่างเพื่อหลีกเลี่ยงปัญหานี้ นอกจากนี้ยังช่วยให้มั่นใจได้ว่าระบบจะไม่แสดงคำขอจนกว่าการเริ่มต้นจะเสร็จสมบูรณ์ - ในทางกลับกัน หากการเริ่มต้นเป็นแบบซิงโครนัส เวลาเริ่มต้นที่นานจะทำให้ Cold Start นานขึ้น ซึ่งอาจเป็นปัญหาโดยเฉพาะอย่างยิ่งกับฟังก์ชันที่มีการทำงานพร้อมกันต่ำในช่วงที่มีการโหลดสูง

ตัวอย่างการอุ่นไลบรารี Node.js แบบไม่พร้อมกัน

Node.js ที่มี Firestore เป็นตัวอย่างของไลบรารี Node.js แบบอะซิงโครนัส หากต้องการใช้ประโยชน์จาก min_instances โค้ดต่อไปนี้จะโหลดให้เสร็จสมบูรณ์ และเริ่มต้นที่เวลาโหลด โดยจะบล็อกการโหลดโมดูล

มีการใช้ TLA ซึ่งหมายความว่าต้องใช้ ES6 โดยใช้ส่วนขยาย .mjs สำหรับโค้ด node.js หรือเพิ่ม type: module ลงในไฟล์ package.json

{
  "main": "main.js",
  "type": "module",
  "dependencies": {
    "@google-cloud/firestore": "^7.10.0",
    "@google-cloud/functions-framework": "^3.4.5"
  }
}

Node.js

import Firestore from '@google-cloud/firestore';
import * as functions from '@google-cloud/functions-framework';

const firestore = new Firestore({preferRest: true});

// Pre-warm firestore connection pool, and preload our global config
// document in cache. In order to ensure no other request comes in,
// block the module loading with a synchronous global request:
const config = await firestore.collection('collection').doc('config').get();

functions.http('fetch', (req, res) => {

// Do something with config and firestore client, which are now preloaded
// and will execute at lower latency.
});

ตัวอย่างการเริ่มต้นส่วนกลาง

Node.js

const functions = require('firebase-functions');
let myCostlyVariable;

exports.function = functions.https.onRequest((req, res) => {
  doUsualWork();
  if(unlikelyCondition()){
      myCostlyVariable = myCostlyVariable || buildCostlyVariable();
  }
  res.status(200).send('OK');
});

Python

from firebase_functions import https_fn

# Always initialized (at cold-start)
non_lazy_global = file_wide_computation()

# Declared at cold-start, but only initialized if/when the function executes
lazy_global = None

@https_fn.on_request()
def lazy_globals(request):

  global lazy_global, non_lazy_global

  # This value is initialized only if (and when) the function is called
  if not lazy_global:
      lazy_global = function_specific_computation()

  return https_fn.Response(f"Lazy: {lazy_global}, non-lazy: {non_lazy_global}.")
  

ฟังก์ชัน HTTP นี้ใช้ตัวแปรส่วนกลางที่เริ่มต้นอย่างเฉื่อยชา โดยจะรับออบเจ็กต์คำขอ (flask.Request) และแสดงผลข้อความตอบกลับ หรือชุดค่าใดก็ได้ที่ สามารถเปลี่ยนเป็นออบเจ็กต์ Response ได้โดยใช้ make_response

ซึ่งมีความสำคัญอย่างยิ่งหากคุณกำหนดฟังก์ชันหลายรายการในไฟล์เดียว และฟังก์ชันต่างๆ ใช้ตัวแปรที่แตกต่างกัน หากไม่ได้ใช้การเริ่มต้นแบบเลื่อนเวลา คุณอาจสิ้นเปลืองทรัพยากรกับตัวแปรที่เริ่มต้นแล้วแต่ไม่เคยใช้

แหล่งข้อมูลเพิ่มเติม

ดูข้อมูลเพิ่มเติมเกี่ยวกับการเพิ่มประสิทธิภาพในวิดีโอ "Google Cloud Performance Atlas" Cloud Functions เวลา Cold Boot