-
Notifications
You must be signed in to change notification settings - Fork 961
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add python support for Functions Emulator. (#5423)
Add support for loading and serving functions written using the Firebase Functions Python SDK (WIP: https://github.com/firebase/firebase-functions-python) This PR is a fork of #4653 extended with support for emulating Python functions. Python Runtime Delegate implementation is unsurprising but does include some additional wrapper code to make sure all commands (e.g. spinning up the admin server) runs within the virtualenv environment. For now, we hardcode virtual environment `venv` directory exists on the developer's laptop, but we'll later add support for specifying arbitrary directory for specifying virtualenv directory via firebase.json configuration. Another note is that each emulated Python function will bind to a Port instead of Unix Domain Socket (UDS) as done when emulating Node.js function. This is because there is no straightfoward, platform-neutral way to bind python webserver to UDS. Finding large number of open port might have a bit more performance penalty and cause bugs due to race condition (similar to #5418) but it seems that we have no other choice atm.
- Loading branch information
Showing
10 changed files
with
354 additions
and
49 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,152 @@ | ||
import * as fs from "fs"; | ||
import * as path from "path"; | ||
import fetch from "node-fetch"; | ||
import { promisify } from "util"; | ||
|
||
import * as portfinder from "portfinder"; | ||
|
||
import * as runtimes from ".."; | ||
import * as backend from "../../backend"; | ||
import * as discovery from "../discovery"; | ||
import { logger } from "../../../../logger"; | ||
import { runWithVirtualEnv } from "../../../../functions/python"; | ||
import { FirebaseError } from "../../../../error"; | ||
import { Build } from "../../build"; | ||
|
||
const LATEST_VERSION: runtimes.Runtime = "python310"; | ||
|
||
/** | ||
* Create a runtime delegate for the Python runtime, if applicable. | ||
* | ||
* @param context runtimes.DelegateContext | ||
* @return Delegate Python runtime delegate | ||
*/ | ||
export async function tryCreateDelegate( | ||
context: runtimes.DelegateContext | ||
): Promise<Delegate | undefined> { | ||
const requirementsTextPath = path.join(context.sourceDir, "requirements.txt"); | ||
|
||
if (!(await promisify(fs.exists)(requirementsTextPath))) { | ||
logger.debug("Customer code is not Python code."); | ||
return; | ||
} | ||
const runtime = context.runtime ? context.runtime : LATEST_VERSION; | ||
if (!runtimes.isValidRuntime(runtime)) { | ||
throw new FirebaseError(`Runtime ${runtime} is not a valid Python runtime`); | ||
} | ||
return Promise.resolve(new Delegate(context.projectId, context.sourceDir, runtime)); | ||
} | ||
|
||
export class Delegate implements runtimes.RuntimeDelegate { | ||
public readonly name = "python"; | ||
constructor( | ||
private readonly projectId: string, | ||
private readonly sourceDir: string, | ||
public readonly runtime: runtimes.Runtime | ||
) {} | ||
|
||
private _bin = ""; | ||
private _modulesDir = ""; | ||
|
||
get bin(): string { | ||
if (this._bin === "") { | ||
this._bin = this.getPythonBinary(); | ||
} | ||
return this._bin; | ||
} | ||
|
||
async modulesDir(): Promise<string> { | ||
if (!this._modulesDir) { | ||
const child = runWithVirtualEnv( | ||
[ | ||
this.bin, | ||
"-c", | ||
'"import firebase_functions; import os; print(os.path.dirname(firebase_functions.__file__))"', | ||
], | ||
this.sourceDir, | ||
{} | ||
); | ||
let out = ""; | ||
child.stdout?.on("data", (chunk: Buffer) => { | ||
const chunkString = chunk.toString(); | ||
out = out + chunkString; | ||
logger.debug(`stdout: ${chunkString}`); | ||
}); | ||
await new Promise((resolve, reject) => { | ||
child.on("exit", resolve); | ||
child.on("error", reject); | ||
}); | ||
this._modulesDir = out.trim(); | ||
} | ||
return this._modulesDir; | ||
} | ||
|
||
getPythonBinary(): string { | ||
if (process.platform === "win32") { | ||
// There is no easy way to get specific version of python executable in Windows. | ||
return "python.exe"; | ||
} | ||
if (this.runtime === "python310") { | ||
return "python3.10"; | ||
} else if (this.runtime === "python311") { | ||
return "python3.11"; | ||
} | ||
return "python"; | ||
} | ||
|
||
validate(): Promise<void> { | ||
// TODO: make sure firebase-functions is included as a dep | ||
return Promise.resolve(); | ||
} | ||
|
||
watch(): Promise<() => Promise<void>> { | ||
return Promise.resolve(() => Promise.resolve()); | ||
} | ||
|
||
async build(): Promise<void> { | ||
return Promise.resolve(); | ||
} | ||
|
||
async serveAdmin(port: number, envs: backend.EnvironmentVariables): Promise<() => Promise<void>> { | ||
const modulesDir = await this.modulesDir(); | ||
const envWithAdminPort = { | ||
...envs, | ||
ADMIN_PORT: port.toString(), | ||
}; | ||
const args = [this.bin, path.join(modulesDir, "private", "serving.py")]; | ||
logger.debug( | ||
`Running admin server with args: ${JSON.stringify(args)} and env: ${JSON.stringify( | ||
envWithAdminPort | ||
)} in ${this.sourceDir}` | ||
); | ||
const childProcess = runWithVirtualEnv(args, this.sourceDir, envWithAdminPort); | ||
return Promise.resolve(async () => { | ||
await fetch(`http://127.0.0.1:${port}/__/quitquitquit`); | ||
const quitTimeout = setTimeout(() => { | ||
if (!childProcess.killed) { | ||
childProcess.kill("SIGKILL"); | ||
} | ||
}, 10_000); | ||
clearTimeout(quitTimeout); | ||
}); | ||
} | ||
|
||
async discoverBuild( | ||
_configValues: backend.RuntimeConfigValues, | ||
envs: backend.EnvironmentVariables | ||
): Promise<Build> { | ||
let discovered = await discovery.detectFromYaml(this.sourceDir, this.projectId, this.runtime); | ||
if (!discovered) { | ||
const adminPort = await portfinder.getPortPromise({ | ||
port: 8081, | ||
}); | ||
const killProcess = await this.serveAdmin(adminPort, envs); | ||
try { | ||
discovered = await discovery.detectFromPort(adminPort, this.projectId, this.runtime); | ||
} finally { | ||
await killProcess(); | ||
} | ||
} | ||
return discovered; | ||
} | ||
} |
Oops, something went wrong.