diff --git a/frontend/src/layout/navigation/ProjectNotice.tsx b/frontend/src/layout/navigation/ProjectNotice.tsx
index 5a0a9c4dd28ce..8d4df0246a3a0 100644
--- a/frontend/src/layout/navigation/ProjectNotice.tsx
+++ b/frontend/src/layout/navigation/ProjectNotice.tsx
@@ -1,4 +1,5 @@
import { IconGear, IconPlus } from '@posthog/icons'
+import { Spinner } from '@posthog/lemon-ui'
import { useActions, useValues } from 'kea'
import { dayjs } from 'lib/dayjs'
import { LemonBanner } from 'lib/lemon-ui/LemonBanner'
@@ -22,17 +23,29 @@ interface ProjectNoticeBlueprint {
closeable?: boolean
}
-function CountDown({ datetime }: { datetime: dayjs.Dayjs }): JSX.Element {
+function CountDown({ datetime, callback }: { datetime: dayjs.Dayjs; callback?: () => void }): JSX.Element {
const [now, setNow] = useState(dayjs())
+ // Format the time difference as 00:00:00
+ const duration = dayjs.duration(datetime.diff(now))
+ const pastCountdown = duration.seconds() < 0
+
+ const countdown = pastCountdown
+ ? 'Expired'
+ : duration.hours() > 0
+ ? duration.format('HH:mm:ss')
+ : duration.format('mm:ss')
+
useEffect(() => {
const interval = setInterval(() => setNow(dayjs()), 1000)
return () => clearInterval(interval)
}, [])
- // Format the time difference as 00:00:00
- const duration = dayjs.duration(datetime.diff(now))
- const countdown = duration.hours() > 0 ? duration.format('HH:mm:ss') : duration.format('mm:ss')
+ useEffect(() => {
+ if (pastCountdown) {
+ callback?.()
+ }
+ }, [pastCountdown])
return <>{countdown}>
}
@@ -40,8 +53,8 @@ function CountDown({ datetime }: { datetime: dayjs.Dayjs }): JSX.Element {
export function ProjectNotice(): JSX.Element | null {
const { projectNoticeVariant } = useValues(navigationLogic)
const { currentOrganization } = useValues(organizationLogic)
- const { logout } = useActions(userLogic)
- const { user } = useValues(userLogic)
+ const { logout, loadUser } = useActions(userLogic)
+ const { user, userLoading } = useValues(userLogic)
const { closeProjectNotice } = useActions(navigationLogic)
const { showInviteModal } = useActions(inviteLogic)
const { requestVerificationLink } = useActions(verifyEmailLogic)
@@ -124,7 +137,14 @@ export function ProjectNotice(): JSX.Element | null {
You are currently logged in as a customer.{' '}
{user?.is_impersonated_until && (
<>
- Expires in
+ Expires in
+ {userLoading ? (
+
+ ) : (
+ loadUser()}>
+ Refresh
+
+ )}
>
)}
>
diff --git a/posthog/middleware.py b/posthog/middleware.py
index ee132dc78d0af..57b561f27600d 100644
--- a/posthog/middleware.py
+++ b/posthog/middleware.py
@@ -677,7 +677,17 @@ def get_impersonated_session_expires_at(request: HttpRequest) -> Optional[dateti
init_time = get_or_set_session_cookie_created_at(request=request)
- return datetime.fromtimestamp(init_time) + timedelta(seconds=settings.IMPERSONATION_TIMEOUT_SECONDS)
+ last_activity_time = request.session.get(settings.IMPERSONATION_COOKIE_LAST_ACTIVITY_KEY, init_time)
+
+ # If the last activity time is less than the idle timeout, we extend the session
+ if time.time() - last_activity_time < settings.IMPERSONATION_IDLE_TIMEOUT_SECONDS:
+ last_activity_time = request.session[settings.IMPERSONATION_COOKIE_LAST_ACTIVITY_KEY] = time.time()
+ request.session.modified = True
+ else:
+ # If the idle timeout has passed then we return it instead of the total timeout
+ return datetime.fromtimestamp(init_time)
+
+ return datetime.fromtimestamp(last_activity_time) + timedelta(seconds=settings.IMPERSONATION_IDLE_TIMEOUT_SECONDS)
class AutoLogoutImpersonateMiddleware:
diff --git a/posthog/settings/web.py b/posthog/settings/web.py
index 98b01dac20b2c..a2158084d52ad 100644
--- a/posthog/settings/web.py
+++ b/posthog/settings/web.py
@@ -376,7 +376,15 @@
# Used only to display in the UI to inform users of allowlist options
PUBLIC_EGRESS_IP_ADDRESSES = get_list(os.getenv("PUBLIC_EGRESS_IP_ADDRESSES", ""))
+# The total time allowed for an impersonated session
IMPERSONATION_TIMEOUT_SECONDS = get_from_env("IMPERSONATION_TIMEOUT_SECONDS", 15 * 60, type_cast=int)
+# The time allowed for an impersonated session to be idle before it expires
+IMPERSONATION_IDLE_TIMEOUT_SECONDS = get_from_env("IMPERSONATION_IDLE_TIMEOUT_SECONDS", 15 * 60, type_cast=int)
+# Impersonation cookie last activity key
+IMPERSONATION_COOKIE_LAST_ACTIVITY_KEY = get_from_env(
+ "IMPERSONATION_COOKIE_LAST_ACTIVITY_KEY", "impersonation_last_activity"
+)
+
SESSION_COOKIE_CREATED_AT_KEY = get_from_env("SESSION_COOKIE_CREATED_AT_KEY", "session_created_at")
PROJECT_SWITCHING_TOKEN_ALLOWLIST = get_list(os.getenv("PROJECT_SWITCHING_TOKEN_ALLOWLIST", "sTMFPsFhdP1Ssg"))
diff --git a/posthog/test/test_middleware.py b/posthog/test/test_middleware.py
index e6a9e95ac9ba0..d952676eb2514 100644
--- a/posthog/test/test_middleware.py
+++ b/posthog/test/test_middleware.py
@@ -501,7 +501,8 @@ def test_logout(self):
self.assertNotIn("ph_current_project_name", response.cookies)
-@override_settings(IMPERSONATION_TIMEOUT_SECONDS=30)
+@override_settings(IMPERSONATION_TIMEOUT_SECONDS=120)
+@override_settings(IMPERSONATION_IDLE_TIMEOUT_SECONDS=30)
class TestAutoLogoutImpersonateMiddleware(APIBaseTest):
other_user: User
@@ -538,21 +539,58 @@ def test_not_staff_user_cannot_login(self):
assert response.status_code == 200
assert self.client.get("/api/users/@me").json()["email"] == self.user.email
- def test_after_timeout_api_requests_401(self):
- now = datetime.now()
+ def test_after_idle_timeout_api_requests_401(self):
+ now = datetime(2024, 1, 1, 12, 0, 0)
with freeze_time(now):
self.login_as_other_user()
res = self.client.get("/api/users/@me")
assert res.status_code == 200
assert res.json()["email"] == "other-user@posthog.com"
+ assert res.json()["is_impersonated_until"] == "2024-01-01T12:00:30+00:00"
assert self.client.session.get("session_created_at") == now.timestamp()
- with freeze_time(now + timedelta(seconds=10)):
+ # Move forward by 20
+ now = now + timedelta(seconds=20)
+ with freeze_time(now):
res = self.client.get("/api/users/@me")
assert res.status_code == 200
assert res.json()["email"] == "other-user@posthog.com"
+ assert res.json()["is_impersonated_until"] == "2024-01-01T12:00:50+00:00"
- with freeze_time(now + timedelta(seconds=35)):
+ # Past idle timeout
+ now = now + timedelta(seconds=35)
+
+ with freeze_time(now):
+ res = self.client.get("/api/users/@me")
+ assert res.status_code == 401
+
+ def test_after_total_timeout_api_requests_401(self):
+ now = datetime(2024, 1, 1, 12, 0, 0)
+ with freeze_time(now):
+ self.login_as_other_user()
+ res = self.client.get("/api/users/@me")
+ assert res.status_code == 200
+ assert res.json()["email"] == "other-user@posthog.com"
+ assert res.json()["is_impersonated_until"] == "2024-01-01T12:00:30+00:00"
+ assert self.client.session.get("session_created_at") == now.timestamp()
+
+ for _ in range(5):
+ # Move forward by 20 seconds 5 times
+ now = now + timedelta(seconds=20)
+ with freeze_time(now):
+ res = self.client.get("/api/users/@me")
+ assert res.status_code == 200
+ assert res.json()["email"] == "other-user@posthog.com"
+ # Format exactly like the date above
+ assert res.json()["is_impersonated_until"] == (now + timedelta(seconds=30)).strftime(
+ "%Y-%m-%dT%H:%M:%S+00:00"
+ )
+
+ now = now + timedelta(
+ seconds=25
+ ) # Final time takes us past the total timeout, despite being under idle timeout
+
+ with freeze_time(now):
res = self.client.get("/api/users/@me")
assert res.status_code == 401