This article introduces how to run Python Lambda functions locally for debugging purposes, using the python–lambda–local module.
This is a continuation of the following article, so I recommend first trying to run your function on AWS Lambda as described there:
https://gonkunblog.com/lambda-selenium-setup/1761/
Prerequisites
My local environment is as follows:
- pyenv: stable 2.3.31
- python: 3.7.3
- pip 23.2.1
- selenium 4.1.0
If you have trouble setting up Python 3.7.3, please refer to this article:
https://gonkunblog.com/python-3-7-3-install-error/1755/
Final Directory Structure
The final directory structure will look like this:
~/D/g/selenium_test ❯❯❯ tree . main ✭ ✖ ✱
.
├── app
│ └── lambda_function.py
├── chromedriver
├── headless-chromium
└── test
└── event.json
Here’s the source code we’ll use (lambda_function.py):
#coding: UTF-8
import os
import time
import pytz
from selenium import webdriver
from selenium.webdriver.chrome.options import Options # Required to use options
from selenium.webdriver.support.ui import WebDriverWait
from selenium.webdriver.common.by import By
from datetime import datetime, timedelta
import warnings
warnings.filterwarnings('ignore')
CHROMEDRIVER_PATH = os.environ["CHROMEDRIVER_PATH"]
HEADLESS_CHROMIUM_PATH = os.environ["HEADLESS_CHROMIUM_PATH"]
def lambda_handler(event, context):
options = webdriver.ChromeOptions()
options.add_argument("--disable-gpu")
options.add_argument("--hide-scrollbars")
options.add_argument("--ignore-certificate-errors")
options.add_argument("--window-size=880x996")
options.add_argument("--no-sandbox")
options.add_argument("--homedir=/tmp")
if "env" in event and event["env"] == "local":
print("This is local env.")
else:
options.add_experimental_option("w3c", True)
options.add_argument("--headless")
options.add_argument("--single-process") # if enable this option, it doesn't work in local
options.binary_location = HEADLESS_CHROMIUM_PATH
driver = webdriver.Chrome(
executable_path=CHROMEDRIVER_PATH,
options=options
)
driver.get("https://www.google.com/")
driver.quit()
The event.json file is as follows, indicating that the execution environment is local:
{
"env": "local"
}
When you run the following command, the Lambda function will start in the local environment:
python-lambda-local -t 100 -f lambda_handler app/lambda_function.py test/event.json
Steps to Run Lambda Locally
Download the chromedriver for your local environment.
(Skip this if you’ve already done it)
We won’t use headless-chromium this time, but let’s download it just in case.
curl -SL https://chromedriver.storage.googleapis.com/2.37/chromedriver_linux64.zip > chromedriver.zip
curl -SL https://github.com/adieuadieu/serverless-chrome/releases/download/v1.0.0-37/stable-headless-chromium-amazonlinux-2017-03.zip > headless-chromium.zip
unzip -o chromedriver.zip -d .unzip -o headless-chromium.zip -d .
Install python-lambda-local:
~/D/g/selenium_test ❯❯❯ pip install python-lambda-local
Create the ./test/event.json file. This will be passed as the event argument to the lambda_handler function:
{
"env": "local"
}
Next, prepare lambda_handler.py:
#coding: UTF-8
import os
import time
import pytz
from selenium import webdriver
from selenium.webdriver.chrome.options import Options # Required to use options
from selenium.webdriver.support.ui import WebDriverWait
from selenium.webdriver.common.by import By
from datetime import datetime, timedelta
import warnings
warnings.filterwarnings('ignore')
CHROMEDRIVER_PATH = os.environ["CHROMEDRIVER_PATH"]
HEADLESS_CHROMIUM_PATH = os.environ["HEADLESS_CHROMIUM_PATH"]
def lambda_handler(event, context):
options = webdriver.ChromeOptions()
options.add_argument("--disable-gpu")
options.add_argument("--hide-scrollbars")
options.add_argument("--ignore-certificate-errors")
options.add_argument("--window-size=880x996")
options.add_argument("--no-sandbox")
options.add_argument("--homedir=/tmp")
if "env" in event and event["env"] == "local":
print("This is local env.")
else:
options.add_experimental_option("w3c", True)
options.add_argument("--headless")
options.add_argument("--single-process") # if enable this option, it doesn't work in local
options.binary_location = HEADLESS_CHROMIUM_PATH
driver = webdriver.Chrome(
executable_path=CHROMEDRIVER_PATH,
options=options
)
driver.get("https://www.google.com/")
driver.quit()
In the if statement, we’re skipping some options that aren’t needed for local environments.
(Including these options would prevent Selenium from starting locally)
Now let’s set the environment variables used in the program.
Specify the two files we downloaded earlier:
export HEADLESS_CHROMIUM_PATH=[path to file]/headless-chromium
export CHROMEDRIVER_PATH=[path to file]/chromedriver
Run the Lambda function from the local environment.
We’re passing the timeout duration with the -t option:
~/D/g/selenium_test ❯❯❯ python-lambda-local -t 100 -f lambda_handler app/lambda_function.py test/event.json main ✭ ✖ ✱
[root - INFO - 2023-10-25 00:09:35,835] Event: {'env': 'local'}
[root - INFO - 2023-10-25 00:09:35,836] START RequestId: 9e7527bd-09e9-4c19-af65-21611df6b9b9 Version:
This is local env.
[root - INFO - 2023-10-25 00:09:41,684] END RequestId: 9e7527bd-09e9-4c19-af65-21611df6b9b9
[root - INFO - 2023-10-25 00:09:41,687] REPORT RequestId: 9e7527bd-09e9-4c19-af65-21611df6b9b9 Duration: 5077.01 ms
[root - INFO - 2023-10-25 00:09:41,687] RESULT:
None
We’ve confirmed that Lambda works locally.
This allows you to test your function without having to run it on AWS Lambda every time.
The next step to improve usability would be to manage this source code in a GitHub repository and set up automatic deployment to Lambda when changes are made to the master branch.
(I’ll cover that in a separate article)
Thank you for reading this far.