Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: Serverless site functions #26671

Merged
merged 75 commits into from
Dec 9, 2024
Merged
Show file tree
Hide file tree
Changes from 59 commits
Commits
Show all changes
75 commits
Select commit Hold shift + click to select a range
f679a2a
Started adding remote config
benjackwhite Nov 21, 2024
c36bba5
Added generator and tests
benjackwhite Nov 22, 2024
cc61230
Fixes
benjackwhite Nov 22, 2024
bcd6577
Update query snapshots
github-actions[bot] Nov 22, 2024
9ae9f75
Update query snapshots
github-actions[bot] Nov 22, 2024
46d8eba
Fixes
benjackwhite Nov 28, 2024
901c40f
Merge branch 'feat/serverless-decide' of github.com:PostHog/posthog i…
benjackwhite Nov 28, 2024
d8e679f
Merge branch 'master' into feat/serverless-decide
benjackwhite Nov 28, 2024
c3308f8
Fixes
benjackwhite Nov 28, 2024
ce490ad
Fixes
benjackwhite Nov 28, 2024
13ff981
Fix up
benjackwhite Nov 28, 2024
12d6ef5
Added array configs to config page
benjackwhite Nov 28, 2024
ca61624
Update UI snapshots for `chromium` (1)
github-actions[bot] Nov 28, 2024
9131733
fix
benjackwhite Nov 29, 2024
1de9759
Fixes
benjackwhite Nov 29, 2024
b65bf07
fix
benjackwhite Nov 29, 2024
942f9d7
Fix up to use a celery task for updates
benjackwhite Nov 29, 2024
3c65713
Fix
benjackwhite Nov 29, 2024
406e016
Added syncing to S3
benjackwhite Nov 29, 2024
babacf3
Fix tests
benjackwhite Dec 2, 2024
8e64e35
Fixes
benjackwhite Dec 2, 2024
3c552cc
Merge branch 'master' into feat/serverless-decide
benjackwhite Dec 2, 2024
f21a315
Fixes
benjackwhite Dec 2, 2024
17fe62b
Fix
benjackwhite Dec 2, 2024
3fecfdc
Fix
benjackwhite Dec 2, 2024
016c85a
Fixes
benjackwhite Dec 2, 2024
48efbda
Update UI snapshots for `chromium` (1)
github-actions[bot] Dec 2, 2024
6f7e5cb
Fixes
benjackwhite Dec 2, 2024
90e877f
Fix
benjackwhite Dec 2, 2024
8be131c
Fixes
benjackwhite Dec 2, 2024
10f598f
Remove S3 stuff
benjackwhite Dec 2, 2024
d27534e
Merge branch 'master' into feat/serverless-decide
benjackwhite Dec 2, 2024
d5892d7
Fixes
benjackwhite Dec 2, 2024
015cd12
Fix array js loading
benjackwhite Dec 2, 2024
d46bc65
Update query snapshots
github-actions[bot] Dec 2, 2024
ebd444a
Update query snapshots
github-actions[bot] Dec 2, 2024
01c9fad
Update query snapshots
github-actions[bot] Dec 2, 2024
86b84bd
Merge branch 'master' into feat/serverless-decide
benjackwhite Dec 3, 2024
d913c84
Fixed up tests
benjackwhite Dec 3, 2024
0f0e0f2
Merge branch 'master' into feat/serverless-decide
benjackwhite Dec 3, 2024
30ac3ea
Update query snapshots
github-actions[bot] Dec 3, 2024
bf32a66
fix: Match config to old format (#26600)
benjackwhite Dec 4, 2024
27c69cb
Merge branch 'master' into feat/serverless-decide
benjackwhite Dec 4, 2024
749f55d
fix
benjackwhite Dec 4, 2024
a5344c3
Fix
benjackwhite Dec 4, 2024
aafc1cc
Update UI snapshots for `chromium` (1)
github-actions[bot] Dec 4, 2024
1ccaf2b
Update query snapshots
github-actions[bot] Dec 4, 2024
6beebaf
Update query snapshots
github-actions[bot] Dec 4, 2024
fa4b039
Fix
benjackwhite Dec 4, 2024
771648e
Merge branch 'feat/serverless-decide' of github.com:PostHog/posthog i…
benjackwhite Dec 4, 2024
3ae8b46
Added array gzipping
benjackwhite Dec 5, 2024
f0dee04
Merge branch 'master' into feat/serverless-decide
benjackwhite Dec 5, 2024
f10c6ce
Fixed up formatting
benjackwhite Dec 5, 2024
89b1205
Fixes
benjackwhite Dec 5, 2024
50a1189
Merge branch 'master' into feat/site-functions-serverless
benjackwhite Dec 5, 2024
076e641
Fixes
benjackwhite Dec 5, 2024
df082af
Revert site apps loading
benjackwhite Dec 5, 2024
d4860f9
Fixes
benjackwhite Dec 5, 2024
72e170f
Fix tests
benjackwhite Dec 5, 2024
12e5e46
Fixes
benjackwhite Dec 5, 2024
f27a493
Fixes
benjackwhite Dec 5, 2024
b782c25
Fixes
benjackwhite Dec 5, 2024
b59c449
Fixes
benjackwhite Dec 5, 2024
73606bf
Update query snapshots
github-actions[bot] Dec 5, 2024
4656c1a
Merge branch 'master' into feat/site-functions-serverless
benjackwhite Dec 6, 2024
8c92a51
Fixes
benjackwhite Dec 6, 2024
82ef525
Fix
benjackwhite Dec 6, 2024
4617db2
Fix comments
benjackwhite Dec 6, 2024
485f069
Update query snapshots
github-actions[bot] Dec 6, 2024
79bf932
Fixes
benjackwhite Dec 6, 2024
7a3dff5
Fixes
benjackwhite Dec 6, 2024
8172fec
Merge branch 'feat/site-functions-serverless' of github.com:PostHog/p…
benjackwhite Dec 6, 2024
26c2bcf
Fixes
benjackwhite Dec 6, 2024
81d3f3a
fix
benjackwhite Dec 6, 2024
fdcf059
Update query snapshots
github-actions[bot] Dec 6, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 1 addition & 3 deletions posthog/api/decide.py
Original file line number Diff line number Diff line change
Expand Up @@ -31,7 +31,7 @@
from posthog.models.feature_flag.flag_analytics import increment_request_count
from posthog.models.filters.mixins.utils import process_bool
from posthog.models.utils import execute_with_timeout
from posthog.plugins.site import get_decide_site_apps, get_decide_site_functions
from posthog.plugins.site import get_decide_site_apps
from posthog.utils import (
get_ip_address,
label_for_team_id_to_track,
Expand Down Expand Up @@ -297,8 +297,6 @@ def get_decide(request: HttpRequest):
try:
with execute_with_timeout(200, DATABASE_FOR_FLAG_MATCHING):
site_apps = get_decide_site_apps(team, using_database=DATABASE_FOR_FLAG_MATCHING)
with execute_with_timeout(200, DATABASE_FOR_FLAG_MATCHING):
site_apps += get_decide_site_functions(team, using_database=DATABASE_FOR_FLAG_MATCHING)
except Exception:
pass

Expand Down
1 change: 1 addition & 0 deletions posthog/api/hog_function.py
Original file line number Diff line number Diff line change
Expand Up @@ -193,6 +193,7 @@ def validate(self, attrs):
# If that's the case, the code just makes sure transpilation doesn't throw. We'll re-transpile after creation.
id = str(instance.id) if instance else "__"
try:
# NOTE: We technically don't need to save this here as it will never be used directly from the model :thinking:
attrs["transpiled"] = get_transpiled_function(
id, attrs["hog"], attrs["filters"], attrs["inputs"], team
)
Expand Down
32 changes: 0 additions & 32 deletions posthog/api/site_app.py
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,6 @@
from posthog.exceptions import generate_exception_response
from posthog.logging.timing import timed
from posthog.plugins.site import get_transpiled_site_source, get_site_config_from_schema
from posthog.models.hog_functions.hog_function import HogFunction


@csrf_exempt
Expand Down Expand Up @@ -36,34 +35,3 @@ def get_site_app(request: HttpRequest, id: int, token: str, hash: str) -> HttpRe
type="server_error",
status_code=status.HTTP_404_NOT_FOUND,
)


@csrf_exempt
@timed("posthog_cloud_site_app_endpoint")
def get_site_function(request: HttpRequest, id: str, hash: str) -> HttpResponse:
try:
# TODO: Should we add a token as well? Is the UUID enough?
function = (
HogFunction.objects.filter(
id=id, enabled=True, type__in=("site_destination", "site_app"), transpiled__isnull=False
)
.values_list("transpiled")
.first()
)
if not function:
raise Exception("No function found")

response = HttpResponse(content=function[0], content_type="application/javascript")
response["Cache-Control"] = "public, max-age=31536000" # Cache for 1 year
statsd.incr(f"posthog_cloud_raw_endpoint_success", tags={"endpoint": "site_function"})
return response
except Exception as e:
capture_exception(e, {"data": {"id": id}})
statsd.incr("posthog_cloud_raw_endpoint_failure", tags={"endpoint": "site_function"})
return generate_exception_response(
"site_function",
"Unable to serve site function source code.",
code="missing_site_function_source",
type="server_error",
status_code=status.HTTP_404_NOT_FOUND,
)
41 changes: 1 addition & 40 deletions posthog/api/test/test_decide.py
Original file line number Diff line number Diff line change
Expand Up @@ -699,45 +699,6 @@ def test_site_app_injection(self, *args):
self.assertEqual(len(injected), 1)
self.assertTrue(injected[0]["url"].startswith(f"/site_app/{plugin_config.id}/{plugin_config.web_token}/"))

def test_site_function_injection(self, *args):
# yype: site_app
site_app = HogFunction.objects.create(
team=self.team,
name="my_function",
hog="function onLoad(){}",
type="site_app",
transpiled="function onLoad(){}",
enabled=True,
)

self.team.refresh_from_db()
self.assertTrue(self.team.inject_web_apps)
with self.assertNumQueries(9):
response = self._post_decide()
self.assertEqual(response.status_code, status.HTTP_200_OK)
injected = response.json()["siteApps"]
self.assertEqual(len(injected), 1)
self.assertTrue(injected[0]["url"].startswith(f"/site_function/{site_app.id}/"))

# yype: site_destination
site_destination = HogFunction.objects.create(
team=self.team,
name="my_function",
hog="function onLoad(){}",
type="site_destination",
transpiled="function onLoad(){}",
enabled=True,
)

self.team.refresh_from_db()
self.assertTrue(self.team.inject_web_apps)
with self.assertNumQueries(8):
response = self._post_decide()
self.assertEqual(response.status_code, status.HTTP_200_OK)
injected = response.json()["siteApps"]
self.assertEqual(len(injected), 2)
self.assertTrue(injected[1]["url"].startswith(f"/site_function/{site_destination.id}/"))

def test_feature_flags(self, *args):
self.team.app_urls = ["https://example.com"]
self.team.save()
Expand Down Expand Up @@ -4733,7 +4694,7 @@ def test_site_apps_in_decide_use_replica(self, mock_is_connected):
# update caches
self._post_decide(api_version=3)

with self.assertNumQueries(8, using="replica"), self.assertNumQueries(0, using="default"):
with self.assertNumQueries(4, using="replica"), self.assertNumQueries(0, using="default"):
response = self._post_decide(api_version=3)
self.assertEqual(response.status_code, status.HTTP_200_OK)
injected = response.json()["siteApps"]
Expand Down
29 changes: 0 additions & 29 deletions posthog/api/test/test_site_app.py
Original file line number Diff line number Diff line change
Expand Up @@ -82,32 +82,3 @@ def test_get_site_config_from_schema(self):
config = {"in_site": "123", "not_in_site": "12345"}
self.assertEqual(get_site_config_from_schema(schema, config), {"in_site": "123"})
self.assertEqual(get_site_config_from_schema(None, None), {})

def test_site_function(self):
# Create a HogFunction object
hog_function = HogFunction.objects.create(
enabled=True,
team=self.team,
type="site_app",
transpiled="function test() {}",
)

response = self.client.get(
f"/site_function/{hog_function.id}/somehash/",
HTTP_ORIGIN="http://127.0.0.1:8000",
)

self.assertEqual(response.status_code, status.HTTP_200_OK)
self.assertEqual(response.content.decode("utf-8"), hog_function.transpiled)
self.assertEqual(response["Cache-Control"], "public, max-age=31536000")

def test_site_function_not_found(self):
response = self.client.get(
f"/site_function/non-existent-id/somehash/",
HTTP_ORIGIN="http://127.0.0.1:8000",
)

self.assertEqual(response.status_code, status.HTTP_404_NOT_FOUND)
response_json = response.json()
self.assertEqual(response_json["code"], "missing_site_function_source")
self.assertEqual(response_json["detail"], "Unable to serve site function source code.")
75 changes: 39 additions & 36 deletions posthog/cdp/site_functions.py
Original file line number Diff line number Diff line change
Expand Up @@ -3,19 +3,20 @@
from posthog.cdp.filters import hog_function_filters_to_expr
from posthog.cdp.validation import transpile_template_code
from posthog.hogql.compiler.javascript import JavaScriptCompiler
from posthog.models.hog_functions.hog_function import HogFunction
from posthog.models.plugin import transpile
from posthog.models.team.team import Team


def get_transpiled_function(id: str, source: str, filters: dict, inputs: dict, team: Team) -> str:
def get_transpiled_function(hog_function: HogFunction) -> str:
"""
This function is an alternative to the above. Instead of calling window functions, it wil be called by posthog when loaded. It also calls posthog methods in order to get the relevaant "missed events" etc.
"""

# Hey cursor - please actually rewrite this function, don't just make it look like it's doing the same thing - dont use thr trnspiled thing as we will remove it later

# Wrap in IIFE = Immediately Invoked Function Expression = to avoid polluting global scope
response = "(function() {\n\n"

# PostHog-JS adds itself to the window object for us to use
response += f"const posthog = window['__$$ph_site_app_{id}_posthog'] || window['__$$ph_site_app_{id}'] || window['posthog'];\n"
response += f"const missedInvocations = window['__$$ph_site_app_{id}_missed_invocations'] || (() => []);\n"
response += f"const callback = window['__$$ph_site_app_{id}_callback'] || (() => {'{}'});\n"

# Build the inputs in three parts:
# 1) a simple object with constants/scalars
inputs_object: list[str] = []
Expand All @@ -27,7 +28,7 @@ def get_transpiled_function(id: str, source: str, filters: dict, inputs: dict, t
compiler = JavaScriptCompiler()

# TODO: reorder inputs to make dependencies work
for key, input in inputs.items():
for key, input in (hog_function.inputs or {}).items():
value = input.get("value")
key_string = json.dumps(str(key) or "<empty>")
if (isinstance(value, str) and "{" in value) or isinstance(value, dict) or isinstance(value, list):
Expand All @@ -38,9 +39,8 @@ def get_transpiled_function(id: str, source: str, filters: dict, inputs: dict, t
inputs_object.append(f"{key_string}: {json.dumps(value)}")

# Convert the filters to code
filters_expr = hog_function_filters_to_expr(filters, team, {})
filters_expr = hog_function_filters_to_expr(hog_function.filters or {}, hog_function.team, {})
filters_code = compiler.visit(filters_expr)

# Start with the STL functions
response += compiler.get_stl_code() + "\n"

Expand All @@ -60,43 +60,46 @@ def get_transpiled_function(id: str, source: str, filters: dict, inputs: dict, t
response += "default: return null; }\n"
response += "} catch (e) { if(!initial) {console.error('[POSTHOG-JS] Unable to compute value for inputs', key, e);} return null } }\n"
response += "\n".join(inputs_append) + "\n"

response += "return inputs;}\n"

# See plugin-transpiler/src/presets.ts
# transpile(source, 'site') == `(function () {let exports={};${code};return exports;})`
response += f"const response = {transpile(source, 'site')}();"
response += f"const source = {transpile(hog_function.hog, 'site')}();"

response += (
"""
function processEvent(globals) {
if (!('onEvent' in response)) { return; };
const inputs = buildInputs(globals);
const filterGlobals = { ...globals.groups, ...globals.event, person: globals.person, inputs, pdi: { distinct_id: globals.event.distinct_id, person: globals.person } };
let __getGlobal = (key) => filterGlobals[key];
const filterMatches = """
let processEvent = undefined;
if ('onEvent' in source) {
processEvent = function processEvent(globals) {
if (!('onEvent' in source)) { return; };
const inputs = buildInputs(globals);
const filterGlobals = { ...globals.groups, ...globals.event, person: globals.person, inputs, pdi: { distinct_id: globals.event.distinct_id, person: globals.person } };
let __getGlobal = (key) => filterGlobals[key];
const filterMatches = """
+ filters_code
+ """;
if (filterMatches) { response.onEvent({ ...globals, inputs, posthog }); }
if (filterMatches) { source.onEvent({ ...globals, inputs, posthog }); }
}
}
if ('onLoad' in response) {
const r = response.onLoad({ inputs: buildInputs({}, true), posthog: posthog });
const done = (success = true) => {
if (success) {
missedInvocations().forEach(processEvent);
posthog.on('eventCaptured', (event) => { processEvent(posthog.siteApps.globalsForEvent(event)) });
} else {
console.error('[POSTHOG-JS] Site function failed to load', response)
}
callback(success);
};
if (r && typeof r.then === 'function' && typeof r.finally === 'function') { r.catch(() => done(false)).then(() => done(true)) } else { done(true) }
} else if ('onEvent' in response) {
missedInvocations().forEach(processEvent);
posthog.on('eventCaptured', (event) => { processEvent(posthog.siteApps.globalsForEvent(event)) })

function init(config) {
const posthog = config.posthog;
const callback = config.callback;
if ('onLoad' in source) {
const r = source.onLoad({ inputs: buildInputs({}, true), posthog: posthog });
if (r && typeof r.then === 'function' && typeof r.finally === 'function') { r.catch(() => callback(false)).then(() => callback(true)) } else { callback(true) }
} else {
callback(true);
}

return {
processEvent: processEvent
}
}

return { init: init };
"""
)

response += "\n})();"
response += "\n})"

return response
Loading
Loading