عرض المحتوى الديناميكي واستضافة الخدمات المصغَّرة باستخدام Cloud Run

يمكنك دمج Cloud Run مع Firebase Hosting لإنشاء المحتوى الديناميكي وعرضه أو إنشاء واجهات برمجة تطبيقات REST كخدمات مصغّرة.

باستخدام Cloud Run، يمكنك نشر تطبيق مجمَّع في صورة حاوية. بعد ذلك، يمكنك استخدام Firebase Hosting لتوجيه طلبات HTTPS لتشغيل تطبيقك المضمّن في حاوية.

  • تتوافق Cloud Run مع عدة لغات (بما في ذلك Go وNode.js وPython وJava)، ما يمنحك المرونة في استخدام لغة البرمجة وإطار العمل الذي تختارهما.
  • Cloud Run توسيع نطاق صورة الحاوية وتصغيره تلقائيًا وبشكل أفقي للتعامل مع الطلبات المستلَمة، ثم تصغير نطاقها عند انخفاض الطلب
  • تدفع فقط مقابل وحدة المعالجة المركزية والذاكرة والشبكة المستخدَمة أثناء معالجة الطلب.

للاطّلاع على أمثلة على حالات الاستخدام ونماذج لـ Cloud Run المدمجة مع Firebase Hosting، انتقِل إلى نظرتنا العامة حول الخدمات غير الخادمية.


يوضّح لك هذا الدليل كيفية تنفيذ ما يلي:

  1. كتابة تطبيق بسيط "Hello World"
  2. تضمين تطبيق في حاوية وتحميله إلى Artifact Registry
  3. نشر صورة الحاوية إلى Cloud Run
  4. توجيه الطلبات Hosting إلى تطبيقك الذي يعمل في حاوية

يُرجى العِلم أنّه لتحسين أداء عرض المحتوى الديناميكي، يمكنك ضبط إعدادات ذاكرة التخزين المؤقت بشكل اختياري.

قبل البدء

قبل استخدام Cloud Run، عليك إكمال بعض المهام الأولية، بما في ذلك إعداد حساب Cloud Billing وتفعيل واجهة برمجة التطبيقات Cloud Run وتثبيت أداة سطر الأوامر gcloud.

إعداد الفوترة لمشروعك

توفّر Cloud Run حصة استخدام مجانية، ولكن يجب أن يكون لديك حساب Cloud Billing مرتبط بمشروعك على Firebase لاستخدام Cloud Run أو تجربته.

تفعيل واجهة برمجة التطبيقات وتثبيت حزمة تطوير البرامج (SDK)

  1. فعِّل واجهة برمجة التطبيقات Cloud Run في وحدة تحكّم Google APIs باتّباع الخطوات التالية:

    1. افتح صفحة واجهة برمجة التطبيقات Cloud Run في وحدة تحكّم Google APIs.

    2. عندما يُطلب منك ذلك، اختَر مشروعك على Firebase.

    3. انقر على تفعيل في صفحة واجهة برمجة التطبيقات Cloud Run.

  2. ثبِّت حزمة Cloud SDK وابدأ إعدادها.

  3. تأكَّد من ضبط أداة gcloud على المشروع الصحيح:

    gcloud config list

الخطوة 1: كتابة نموذج التطبيق

يُرجى العِلم أنّ Cloud Run تتيح العديد من اللغات الأخرى بالإضافة إلى اللغات المعروضة في المثال التالي.

Go

  1. أنشئ دليلاً جديدًا باسم helloworld-go، ثم انتقِل إلى هذا الدليل:

    mkdir helloworld-go
    cd helloworld-go
  2. أنشئ ملفًا جديدًا باسم helloworld.go، ثم أضِف الرمز التالي:

    package main
    
    import (
    	"fmt"
    	"log"
    	"net/http"
    	"os"
    )
    
    func handler(w http.ResponseWriter, r *http.Request) {
    	log.Print("helloworld: received a request")
    	target := os.Getenv("TARGET")
    	if target == "" {
    		target = "World"
    	}
    	fmt.Fprintf(w, "Hello %s!\n", target)
    }
    
    func main() {
    	log.Print("helloworld: starting server...")
    
    	http.HandleFunc("/", handler)
    
    	port := os.Getenv("PORT")
    	if port == "" {
    		port = "8080"
    	}
    
    	log.Printf("helloworld: listening on port %s", port)
    	log.Fatal(http.ListenAndServe(fmt.Sprintf(":%s", port), nil))
    }
    

    ينشئ هذا الرمز خادم ويب أساسيًا يستمع إلى المنفذ المحدّد بواسطة متغيّر البيئة PORT.

تم الانتهاء من تطبيقك وأصبح جاهزًا ليتم وضعه في حاوية وتحميله إلى Artifact Registry.

Node.js

  1. أنشئ دليلاً جديدًا باسم helloworld-nodejs، ثم غيِّر الدليل إلى ما يلي:

    mkdir helloworld-nodejs
    cd helloworld-nodejs
  2. أنشِئ ملف package.json يتضمّن المحتوى التالي:

    {
      "name": "knative-serving-helloworld",
      "version": "1.0.0",
      "description": "Simple hello world sample in Node",
      "main": "index.js",
      "scripts": {
        "start": "node index.js"
      },
      "author": "",
      "license": "Apache-2.0",
      "dependencies": {
        "express": "^4.21.2"
      }
    }
    
  3. أنشئ ملفًا جديدًا باسم index.js، ثم أضِف الرمز التالي:

    const express = require('express');
    const app = express();
    
    app.get('/', (req, res) => {
      console.log('Hello world received a request.');
    
      const target = process.env.TARGET || 'World';
      res.send(`Hello ${target}!\n`);
    });
    
    const port = process.env.PORT || 8080;
    app.listen(port, () => {
      console.log('Hello world listening on port', port);
    });
    

    ينشئ هذا الرمز خادم ويب أساسيًا يستمع إلى المنفذ المحدّد بواسطة متغيّر البيئة PORT.

تم الانتهاء من تطبيقك وأصبح جاهزًا ليتم وضعه في حاوية وتحميله إلى Artifact Registry.

Python

  1. أنشئ دليلاً جديدًا باسم helloworld-python، ثم غيِّر الدليل إلى ما يلي:

    mkdir helloworld-python
    cd helloworld-python
  2. أنشئ ملفًا جديدًا باسم app.py، ثم أضِف الرمز التالي:

    import os
    
    from flask import Flask
    
    app = Flask(__name__)
    
    @app.route('/')
    def hello_world():
        target = os.environ.get('TARGET', 'World')
        return 'Hello {}!\n'.format(target)
    
    if __name__ == "__main__":
        app.run(debug=True,host='0.0.0.0',port=int(os.environ.get('PORT', 8080)))
    

    ينشئ هذا الرمز خادم ويب أساسيًا يستمع إلى المنفذ المحدّد بواسطة متغيّر البيئة PORT.

تم الانتهاء من تطبيقك وأصبح جاهزًا ليتم وضعه في حاوية وتحميله إلى Artifact Registry.

Java

  1. ثبِّت Java SE 8 أو إصدارًا أحدث من JDK وCURL.

    يُرجى العِلم أنّه علينا تنفيذ هذه الخطوة فقط لإنشاء مشروع الويب الجديد في الخطوة التالية. سيتم تحميل جميع التبعيات إلى الحاوية باستخدام ملف Dockerfile الذي سيتم توضيح طريقة استخدامه لاحقًا.

  2. من وحدة التحكّم، أنشئ مشروع ويب فارغًا جديدًا باستخدام cURL ثمّ أوامر فك الضغط:

    curl https://start.spring.io/starter.zip \
        -d dependencies=web \
        -d name=helloworld \
        -d artifactId=helloworld \
        -o helloworld.zip
    unzip helloworld.zip

    سيؤدي ذلك إلى إنشاء مشروع SpringBoot.

  3. عدِّل الفئة SpringBootApplication في src/main/java/com/example/helloworld/HelloworldApplication.java من خلال إضافة @RestController للتعامل مع عملية الربط /، وأضِف أيضًا الحقل @Value لتوفير متغيّر البيئة TARGET:

    package com.example.helloworld;
    
    import org.springframework.beans.factory.annotation.Value;
    import org.springframework.boot.SpringApplication;
    import org.springframework.boot.autoconfigure.SpringBootApplication;
    import org.springframework.web.bind.annotation.GetMapping;
    import org.springframework.web.bind.annotation.RestController;
    
    @SpringBootApplication
    public class HelloworldApplication {
    
      @Value("${TARGET:World}")
      String target;
    
      @RestController
      class HelloworldController {
        @GetMapping("/")
        String hello() {
          return "Hello " + target + "!";
        }
      }
    
      public static void main(String[] args) {
        SpringApplication.run(HelloworldApplication.class, args);
      }
    }
    

    ينشئ هذا الرمز خادم ويب أساسيًا يستمع إلى المنفذ المحدّد بواسطة متغيّر البيئة PORT.

تم الانتهاء من تطبيقك وأصبح جاهزًا ليتم وضعه في حاوية وتحميله إلى Artifact Registry.

الخطوة 2: إنشاء حاوية لتطبيق وتحميلها إلى Artifact Registry

  1. يمكنك إنشاء حاوية للتطبيق النموذجي من خلال إنشاء ملف جديد باسم Dockerfile في الدليل نفسه الذي توجد فيه ملفات المصدر. انسخ المحتوى التالي في ملفك.

    Go

    # Use the official Golang image to create a build artifact.
    # This is based on Debian and sets the GOPATH to /go.
    FROM golang:latest AS builder
    
    ARG TARGETOS
    ARG TARGETARCH
    
    # Create and change to the app directory.
    WORKDIR /app
    
    # Copy local code to the container image.
    COPY . ./
    
    # Install dependencies and tidy up the go.mod and go.sum files.
    RUN go mod tidy
    
    # Build the binary.
    # -mod=readonly ensures immutable go.mod and go.sum in container builds.
    RUN CGO_ENABLED=0 GOOS=${TARGETOS} GOARCH=${TARGETARCH} go build -mod=readonly -v -o server
    
    # Use the official Alpine image for a lean production container.
    # https://hub.docker.com/_/alpine
    # https://docs.docker.com/develop/develop-images/multistage-build/#use-multi-stage-builds
    FROM alpine:3
    RUN apk add --no-cache ca-certificates
    
    # Copy the binary to the production image from the builder stage.
    COPY --from=builder /app/server /server
    
    # Run the web service on container startup.
    CMD ["/server"]
    

    Node.js

    # Use the official lightweight Node.js 12 image.
    # https://hub.docker.com/_/node
    FROM node:12-slim
    
    # Create and change to the app directory.
    WORKDIR /usr/src/app
    
    # Copy application dependency manifests to the container image.
    # A wildcard is used to ensure both package.json AND package-lock.json are copied.
    # Copying this separately prevents re-running npm install on every code change.
    COPY package*.json ./
    
    # Install production dependencies.
    RUN npm install --only=production
    
    # Copy local code to the container image.
    COPY . ./
    
    # Run the web service on container startup.
    CMD [ "npm", "start" ]
    

    Python

    # Use the official lightweight Python image.
    # https://hub.docker.com/_/python
    FROM python:3.7-slim
    
    # Allow statements and log messages to immediately appear in the Knative logs
    ENV PYTHONUNBUFFERED True
    
    # Copy local code to the container image.
    ENV APP_HOME /app
    WORKDIR $APP_HOME
    COPY . ./
    
    # Install production dependencies.
    RUN pip install Flask gunicorn
    
    # Run the web service on container startup. Here we use the gunicorn
    # webserver, with one worker process and 8 threads.
    # For environments with multiple CPU cores, increase the number of workers
    # to be equal to the cores available.
    CMD exec gunicorn --bind :$PORT --workers 1 --threads 8 --timeout 0 app:app
    

    Java

    # Use the official maven/Java 8 image to create a build artifact: https://hub.docker.com/_/maven
    FROM maven:3.5-jdk-8-alpine AS builder
    
    # Copy local code to the container image.
    WORKDIR /app
    COPY pom.xml .
    COPY src ./src
    
    # Build a release artifact.
    RUN mvn package -DskipTests
    
    # Use the Official OpenJDK image for a lean production stage of our multi-stage build.
    # https://hub.docker.com/_/openjdk
    # https://docs.docker.com/develop/develop-images/multistage-build/#use-multi-stage-builds
    FROM openjdk:8-jre-alpine
    
    # Copy the jar to the production image from the builder stage.
    COPY --from=builder /app/target/helloworld-*.jar /helloworld.jar
    
    # Run the web service on container startup.
    CMD ["java", "-Djava.security.egd=file:/dev/./urandom", "-jar", "/helloworld.jar"]
    

  2. أنشئ صورة الحاوية باستخدام Cloud Build من خلال تنفيذ الأمر التالي من الدليل الذي يحتوي على ملف Dockerfile:

    gcloud builds submit --tag gcr.io/PROJECT_ID/helloworld

    عند نجاح العملية، ستظهر لك الرسالة SUCCESS (تم بنجاح) التي تحتوي على اسم الصورة
    (gcr.io/PROJECT_ID/helloworld).

تم الآن تخزين صورة الحاوية في Artifact Registry ويمكن إعادة استخدامها إذا أردت ذلك.

يُرجى العِلم أنّه بدلاً من Cloud Build، يمكنك استخدام إصدار مثبَّت محليًا من Docker من أجل إنشاء الحاوية محليًا.

الخطوة 3: نشر صورة الحاوية إلى Cloud Run

  1. يمكنك نشر التطبيق باستخدام الأمر التالي:

    gcloud run deploy --image gcr.io/PROJECT_ID/helloworld

  2. عندما يُطلب منك ذلك:

  3. انتظِر بضع لحظات حتى يكتمل النشر. عند النجاح، يعرض سطر الأوامر عنوان URL الخاص بالخدمة. على سبيل المثال: https://helloworld-RANDOM_HASH-us-central1.a.run.app

  4. انتقِل إلى الحاوية التي تم نشرها من خلال فتح عنوان URL للخدمة في متصفّح ويب.

توضّح لك الخطوة التالية كيفية الوصول إلى هذا التطبيق الذي تم إنشاؤه في حاوية من خلال Firebase Hosting عنوان URL كي يتمكّن من إنشاء محتوى ديناميكي لموقعك الإلكتروني المستضاف على Firebase.

الخطوة 4: توجيه طلبات الاستضافة إلى تطبيقك المستند إلى حاوية

باستخدام قواعد إعادة الكتابة، يمكنك توجيه الطلبات التي تتطابق مع أنماط معيّنة إلى وجهة واحدة.

يوضّح المثال التالي كيفية توجيه جميع الطلبات من الصفحة /helloworld على موقعك الإلكتروني Hosting لتفعيل بدء تشغيل مثيل الحاوية helloworld.

  1. تأكَّد مما يلي:

    للحصول على تعليمات مفصّلة حول تثبيت واجهة سطر الأوامر وتهيئة Hosting، يُرجى الاطّلاع على دليل البدء في استخدام Hosting.

  2. افتح ملف firebase.json.

  3. أضِف إعدادات rewrite التالية ضمن القسم hosting:

    "hosting": {
      // ...
    
      // Add the "rewrites" attribute within "hosting"
      "rewrites": [ {
        "source": "/helloworld",
        "run": {
          "serviceId": "helloworld",  // "service name" (from when you deployed the container image)
          "region": "us-central1",    // optional (if omitted, default is us-central1)
          "pinTag": true              // optional (see note below)
        }
      } ]
    }
  4. انشر إعدادات الاستضافة على موقعك الإلكتروني من خلال تنفيذ الأمر التالي من جذر دليل مشروعك:

    firebase deploy --only hosting

يمكن الآن الوصول إلى الحاوية من خلال عناوين URL التالية:

  • النطاقات الفرعية في Firebase:
    PROJECT_ID.web.app/ و PROJECT_ID.firebaseapp.com/

  • أي نطاقات مخصّصة مرتبطة:
    CUSTOM_DOMAIN/

انتقِل إلى صفحة إعدادات Hosting للحصول على مزيد من التفاصيل حول قواعد إعادة الكتابة. يمكنك أيضًا التعرّف على ترتيب الأولوية للردود لإعدادات Hosting المختلفة.

الاختبار محليًا

أثناء عملية التطوير، يمكنك تشغيل صورة الحاوية واختبارها محليًا. للحصول على تعليمات مفصّلة، يُرجى الانتقال إلى Cloud Run المستندات.

الخطوات التالية