Pubsub2Inbox is a generic tool to handle input from Pub/Sub messages and turn them into email, webhooks or GCS objects. It's based on an extendable framework consisting of input and output processors. Input processors can enrich the incoming messages with details (for example, fetching the budget from Cloud Billing Budgets API). Multiple output processors can be chained together.
Pubsub2Inbox is written in Python 3 and can be deployed as a Cloud Function easily. To guard credentials and other sensitive information, the tool can fetch its YAML configuration from Google Cloud Secret Manager.
The tool also supports templating of emails, messages and other parameters through Jinja2 templating.
Please note: You cannot connect to SMTP port 25 from GCP. Use alternative ports 465 or 587, or connect via Serverless VPC Connector to your own mailservers.
Out of the box, you'll have the following functionality:
- Budget alert notifications
- Cloud Security Command Center notifications
- Cloud Storage notifications
- How to set up Cloud Storage notifications
- For example, you can automatically send reports via email that are generated in a Cloud Storage bucket
- BigQuery queries
- For example, you can turn any BigQuery query results into CSV files or email messages.
- Any JSON
Available input processors are:
- budget.py: retrieves details from Cloud Billing Budgets API and presents.
- scc.py: enriches Cloud Security Command Center findings notifications.
- bigquery.py: queries from BigQuery datasets.
- genericjson.py: Parses message data as JSON and presents it to output processors.
Please note that the input processors have some IAM requirements to be able to pull information from GCP:
- Resend mechanism (see below)
- Storage Object Admin (
roles/storage.objectAdmin
)
- Storage Object Admin (
- Signed URL generation (see
filters/strings.py:generate_signed_url
)- Storage Admin on the bucket (
roles/storage.admin
)
- Storage Admin on the bucket (
budget.py
- Billing Account Viewer (
roles/billing.viewer
) to retrieve budget details. - Browser (
roles/browser
) to fetch project details.
- Billing Account Viewer (
scc.py
- Browser (
roles/browser
) to fetch project details.
- Browser (
Available output processors are:
- mail.py: can send HTML and/or text emails via SMTP gateways or SendGrid.
- gcs.py: can create objects on GCS from any inputs.
- webhook.py: can send arbitrary HTTP requests, optionally with added OAuth2 bearer token from GCP.
Please note that the output processors have some IAM requirements to be able to pull information from GCP:
mail.py
- Group membership expansion requires following the instructions at Groups API: Authenticating as a service account without domain-wide delegation to grant permissions to the service account the function is running under. You can also use the helper script.
- In addition, the service account that the script runs under will need to have
roles/iam.serviceAccountTokenCreator
on itself when running in Cloud Function/Cloud Run (for Directory API scoped tokens).
Pubsub2Inbox is configured through a YAML file (for examples, see the examples/
directory). Input processors are configured under processors
key and outputs under outputs
.
Features of the specific processors are explain in the corresponding examples.
Pubsub2Inbox has two mechanisms to prevent excessive retries and resend of messages.
The retry mechanism acknowledges and discards any messages that are older than a
configured period (retryPeriod
in configuration, default 2 days).
The resend mechanism is to prevent recurring notifications from being send. It relies
on a Cloud Storage bucket where is stores zero-length files, that are named by
hashing the resendKey
(if it is omitted, all template parameters are used). The
resend period is configurable through resendPeriod
. To prevent the resend bucket
from accumulating unlimited files, set an Object Lifecycle Management policy
on the bucket.
First, we have the configuration in config.yaml
and we're going to store the configuration for
the function as a Cloud Secret Manager secret.
Let's define some variables first:
export PROJECT_ID=your-project # Project ID where function will be deployed
export REGION=europe-west1 # Where to deploy the functions
export SECRET_ID=pubsub2inbox # Secret Manager secret name
export SA_NAME=pubsub2inbox # Service account name
export SECRET_URL="projects/$PROJECT_ID/secrets/$SECRET_ID/versions/latest"
export FUNCTION_NAME="pubsub2inbox"
export PUBSUB_TOPIC="billing-alerts" # projects/$PROJECT_ID/topics/billing-alerts
Then we'll create the secrets in Secret Manager:
gcloud secrets create $SECRET_ID \
--replication-policy="automatic" \
--project $PROJECT_ID
gcloud secrets versions add $SECRET_ID \
--data-file=config.yaml \
--project $PROJECT_ID
We will also create a service account for the Cloud Function:
gcloud iam service-accounts create $SA_NAME \
--project $PROJECT_ID
gcloud secrets add-iam-policy-binding $SECRET_ID \
--member "serviceAccount:$SA_NAME@$PROJECT_ID.iam.gserviceaccount.com" \
--role "roles/secretmanager.secretAccessor" \
--project $PROJECT_ID
gcloud iam service-accounts add-iam-policy-binding $SA_NAME@$PROJECT_ID.iam.gserviceaccount.com \
--member "serviceAccount:$SA_NAME@$PROJECT_ID.iam.gserviceaccount.com" \
--role "roles/iam.serviceAccountTokenCreator" \
--project $PROJECT_ID
Now we can deploy the Cloud Function:
gcloud functions deploy $FUNCTION_NAME \
--entry-point process_pubsub \
--runtime python38 \
--trigger-topic $PUBSUB_TOPIC \
--service-account "$SA_NAME@$PROJECT_ID.iam.gserviceaccount.com" \
--set-env-vars "CONFIG=$SECRET_URL" \
--region $REGION \
--project $PROJECT_ID
If you change the configuration, you can update it via:
Sample Terraform scripts are provided in main.tf
, variables.tf
and outputs.tf
.
Run the command:
# python3 -m unittest discover