Skip to content

Commit 77c9a52

Browse files
authored
Add docs about persistence (e2b-dev#507)
This PR adds docs about Sandbox's persistence
2 parents 27d2afd + ff7cf4e commit 77c9a52

File tree

6 files changed

+172
-78
lines changed

6 files changed

+172
-78
lines changed

apps/web/src/app/(docs)/docs/page.mdx

-2
Original file line numberDiff line numberDiff line change
@@ -31,14 +31,12 @@ Some of the typical use cases for E2B are AI data analysis or visualization, run
3131
The E2B Sandbox is a small isolated VM the can be started very quickly (~150ms). You can think of it as a small computer for the AI model. You can run many sandboxes at once. Typically, you run separate sandbox for each LLM, user, or AI agent session in your app.
3232
For example, if you were building an AI data analysis chatbot, you would start the sandbox for every user session.
3333

34-
3534
## Quickstart
3635
<Quickstart/>
3736

3837
## Code interpreting with AI
3938
<CodeInterpreting/>
4039

41-
4240
## Learn the core concepts
4341
<Concepts/>
4442

Original file line numberDiff line numberDiff line change
@@ -1,16 +1,148 @@
11
# Sandbox persistence
22

3-
We're working on a feature that will allow you to persist sandboxes between runs.
3+
<Note>
4+
Sandbox persistence is currently in beta:
5+
1. [Reach out to us](/docs/support) to get access to the beta.
6+
1. You need to install the [beta version of the SDKs](#installing-the-beta-version-of-the-sdks).
7+
1. Consider [some limitations](#limitations-while-in-beta).
8+
1. The persistence is free for all users during the beta.
9+
</Note>
410

5-
In the meantime, you can mount cloud storage like Amazon's S3, Google Cloud Storage, or Cloudflare's R2 to the sandbox's filesystem.
11+
The sandbox persistence allows you to pause your sandbox and resume it later from the same state it was in when you paused it.
612

7-
**Prerequisites**
8-
- Famil
13+
This includes not only state of the sandbox's filesystem but also the sandbox's memory. This means all running processes, loaded variables, data, etc.
914

15+
## 1. Installing the beta version of the SDKs
16+
To use the sandbox persistence, you need to install the beta version of the SDKs.
1017

18+
<CodeGroup>
19+
```bash {{ language: 'js' }}
20+
npm i e2b@beta
21+
```
1122

12-
## Amazon S3
23+
```bash {{ language: 'python' }}
24+
# Install the latest beta version of the SDK on PyPi
25+
# https://pypi.org/project/e2b/#history
26+
pip install e2b-code-interpreter==1.1.0.b17
27+
```
28+
</CodeGroup>
1329

14-
## Google Cloud Storage
1530

16-
## Cloudflare R2
31+
## 2. Pausing sandbox
32+
When you pause a sandbox, both the sandbox's filesystem and memory state will be saved. This includes all the files in the sandbox's filesystem and all the running processes, loaded variables, data, etc.
33+
34+
<CodeGroup>
35+
```js
36+
import { Sandbox } from 'e2b'
37+
// or use Code Interpreter: https://github.com/e2b-dev/code-interpreter
38+
// import { Sandbox } from '@e2b/code-interpreter'
39+
//
40+
// or use Desktop: https://github.com/e2b-dev/desktop
41+
// import { Sandbox } from '@e2b/desktop'
42+
43+
const sbx = await Sandbox.create()
44+
console.log('Sandbox created', sbx.sandboxId)
45+
46+
// Pause the sandbox
47+
// You can save the sandbox ID in your database
48+
// to resume the sandbox later
49+
const sandboxId = await sbx.pause() // $HighlightLine
50+
console.log('Sandbox paused', sandboxId) // $HighlightLine
51+
```
52+
```python
53+
from e2b import Sandbox
54+
# or use Code Interpreter: https://github.com/e2b-dev/code-interpreter
55+
# from e2b_code_interpreter import Sandbox
56+
#
57+
# or use Desktop: https://github.com/e2b-dev/desktop
58+
# from e2b_desktop import Sandbox
59+
60+
sbx = Sandbox()
61+
print('Sandbox created', sbx.sandbox_id)
62+
63+
# Pause the sandbox
64+
# You can save the sandbox ID in your database
65+
# to resume the sandbox later
66+
sandbox_id = sbx.pause() # $HighlightLine
67+
print('Sandbox paused', sandbox_id) # $HighlightLine
68+
```
69+
</CodeGroup>
70+
71+
72+
## 3. Resuming sandbox
73+
When you resume a sandbox, it will be in the same state it was in when you paused it.
74+
This means that all the files in the sandbox's filesystem will be restored and all the running processes, loaded variables, data, etc. will be restored.
75+
76+
<CodeGroup>
77+
```js
78+
import { Sandbox } from 'e2b'
79+
// or use Code Interpreter: https://github.com/e2b-dev/code-interpreter
80+
// import { Sandbox } from '@e2b/code-interpreter'
81+
//
82+
// or use Desktop: https://github.com/e2b-dev/desktop
83+
// import { Sandbox } from '@e2b/desktop'
84+
85+
const sbx = await Sandbox.create()
86+
console.log('Sandbox created', sbx.sandboxId)
87+
88+
// Pause the sandbox
89+
// You can save the sandbox ID in your database
90+
// to resume the sandbox later
91+
const sandboxId = await sbx.pause()
92+
console.log('Sandbox paused', sandboxId)
93+
94+
// Resume the sandbox from the same state
95+
const sameSbx = await Sandbox.resume(sandboxId) // $HighlightLine
96+
console.log('Sandbox resumed', sameSbx.sandboxId) // $HighlightLine
97+
```
98+
```python
99+
from e2b import Sandbox
100+
# or use Code Interpreter: https://github.com/e2b-dev/code-interpreter
101+
# from e2b_code_interpreter import Sandbox
102+
#
103+
# or use Desktop: https://github.com/e2b-dev/desktop
104+
# from e2b_desktop import Sandbox
105+
106+
sbx = Sandbox()
107+
print('Sandbox created', sbx.sandbox_id)
108+
109+
# Pause the sandbox
110+
# You can save the sandbox ID in your database
111+
# to resume the sandbox later
112+
sandbox_id = sbx.pause()
113+
print('Sandbox paused', sandbox_id)
114+
115+
# Resume the sandbox from the same state
116+
same_sbx = Sandbox.resume(sandbox_id) # $HighlightLine
117+
print('Sandbox resumed', same_sbx.sandbox_id) # $HighlightLine
118+
```
119+
</CodeGroup>
120+
121+
## Sandbox's timeout
122+
When you resume a sandbox, the sandbox's timeout is reset to the default timeout of an E2B sandbox - 5 minutes.
123+
124+
125+
You can pass a custom timeout to the `Sandbox.resume()` method like this:
126+
127+
<CodeGroup>
128+
```js
129+
import { Sandbox } from 'e2b'
130+
131+
const sbx = await Sandbox.resume(sandboxId, { timeoutMs: 60 * 1000 }) // 60 seconds
132+
```
133+
```python
134+
from e2b import Sandbox
135+
136+
sbx = Sandbox.resume(sandbox_id, timeout=60) # 60 seconds
137+
```
138+
</CodeGroup>
139+
140+
## Network
141+
If you have a service (for example a server) running inside your sandbox and you pause the sandbox, the service won't be accessible from the outside and all the clients will be disconnected.
142+
If you resume the sandbox, the service will be accessible again but you need to connect clients again.
143+
144+
145+
## Limitations while in beta
146+
- It takes about 4 seconds per 1 GB RAM to pause the sandbox
147+
- It takes about 1 second to resume the sandbox
148+
- Sandbox can be paused up to 30 days

apps/web/src/components/Concepts.tsx

+11-4
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,8 @@
11
import {
22
FolderTree,
33
Terminal,
4-
Hourglass
4+
Hourglass,
5+
RefreshCcw,
56
} from 'lucide-react'
67

78
import {
@@ -16,6 +17,12 @@ const concepts: BoxItem[] = [
1617
description: 'Learn about how to start the sandbox, manage its lifecycle, and interact with it.',
1718
icon: <Hourglass strokeWidth={1.5} className="h-6 w-6 transition-colors duration-300 fill-white/10 stroke-zinc-400 group-hover:fill-brand-300/10 group-hover:stroke-brand-400" />,
1819
},
20+
{
21+
href: '/docs/sandbox/persistence',
22+
title: 'Sandbox persistence',
23+
description: 'Learn how to achieve data persistence by pausing and resuming sandboxes.',
24+
icon: <RefreshCcw strokeWidth={1.5} className="h-6 w-6 transition-colors duration-300 fill-white/10 stroke-zinc-400 group-hover:fill-brand-300/10 group-hover:stroke-brand-400" />,
25+
},
1926
// {
2027
// href: '/docs/code-execution',
2128
// title: 'AI code execution',
@@ -25,17 +32,17 @@ const concepts: BoxItem[] = [
2532
{
2633
href: '/docs/filesystem',
2734
title: 'Filesystem',
28-
description: 'Each sandbox has its own isolated filesystem that you can use to create, read, write, and delete files.',
35+
description: 'Sandbox has an isolated filesystem that you can use to create, read, write, and delete files.',
2936
icon: <FolderTree strokeWidth={1.5} className="h-6 w-6 transition-colors duration-300 fill-white/10 stroke-zinc-400 group-hover:fill-brand-300/10 group-hover:stroke-brand-400" />,
3037
},
3138
{
3239
href: '/docs/commands',
3340
title: 'Commands',
34-
description: 'You can run terminal commands inside the Sandbox. This allows you to start any process inside the Sandbox.',
41+
description: 'Run terminal commands inside the Sandbox and start any process inside the Sandbox.',
3542
icon: <Terminal strokeWidth={1.5} className="h-6 w-6 transition-colors duration-300 fill-white/10 stroke-zinc-400 group-hover:fill-brand-300/10 group-hover:stroke-brand-400" />,
3643
}
3744
]
3845

3946
export function Concepts() {
4047
return <BoxGrid items={concepts} noBackground />
41-
}
48+
}

apps/web/src/components/Navigation/NavigationLink.tsx

+4-9
Original file line numberDiff line numberDiff line change
@@ -8,12 +8,10 @@ import { NavLink } from './routes'
88
export function NavigationLink({
99
className,
1010
link,
11-
tag,
1211
}: {
1312

1413
className?: string
1514
link: NavLink
16-
tag?: string
1715
}) {
1816
const pathname = usePathname()
1917
// Add this to get the hash
@@ -41,17 +39,14 @@ export function NavigationLink({
4139
>
4240
<div className="flex items-center justify-start gap-1">
4341
{link.icon}
44-
{tag ? (
42+
{link.tag ? (
4543
<div className="flex items-center gap-2">
46-
<Tag
47-
variant="small"
48-
color="emerald"
49-
>
50-
{tag}
51-
</Tag>
5244
<span className={clsx('truncate', isActive ? 'text-white' : '')}>
5345
{link.title}
5446
</span>
47+
<Tag>
48+
{link.tag}
49+
</Tag>
5550
</div>
5651
) : (
5752
<span className={clsx('truncate', isActive ? 'text-white' : '')}>

apps/web/src/components/Navigation/routes.tsx

+11-4
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,14 @@
11
import { Braces, CheckCircle, Home, MessagesSquare } from 'lucide-react'
22
import sdkRefRoutesJson from './sdkRefRoutes.json'
33

4+
enum Tag {
5+
New = 'New',
6+
}
7+
48
export interface NavLink {
59
title: string
610
href: string
11+
tag?: Tag
712
icon?: React.ReactNode
813
}
914

@@ -270,6 +275,11 @@ export const docRoutes: NavGroup[] = [
270275
title: 'Lifecycle',
271276
href: '/docs/sandbox',
272277
},
278+
{
279+
title: 'Persistence',
280+
tag: Tag.New,
281+
href: '/docs/sandbox/persistence',
282+
},
273283
{
274284
title: 'Metadata',
275285
href: '/docs/sandbox/metadata',
@@ -295,9 +305,6 @@ export const docRoutes: NavGroup[] = [
295305
// href: '/docs/sandbox/request-timeouts',
296306
// },
297307
// {
298-
// title: '* Persistence',
299-
// href: '/docs/sandbox/persistence',
300-
// },
301308
],
302309
},
303310
{
@@ -479,4 +486,4 @@ export const sdkRefRoutes: VersionedNavGroup[] = (
479486
group?.title && sdkRefNameMap[group.title]
480487
? sdkRefNameMap[group.title]
481488
: group.title,
482-
}))
489+
}))

apps/web/src/components/Tag.tsx

+7-52
Original file line numberDiff line numberDiff line change
@@ -1,64 +1,19 @@
11
import clsx from 'clsx'
2-
3-
const variantStyles = {
4-
small: '',
5-
medium: 'rounded-lg px-1.5 ring-1 ring-inset',
6-
}
7-
8-
const colorStyles = {
9-
emerald: {
10-
small: 'text-brand-500 dark:text-brand-400',
11-
medium:
12-
'ring-brand-300 dark:ring-brand-400/30 bg-brand-400/10 text-brand-500 dark:text-brand-400',
13-
},
14-
sky: {
15-
small: 'text-sky-500',
16-
medium:
17-
'ring-sky-300 bg-sky-400/10 text-sky-500 dark:ring-sky-400/30 dark:bg-sky-400/10 dark:text-sky-400',
18-
},
19-
amber: {
20-
small: 'text-amber-500',
21-
medium:
22-
'ring-amber-300 bg-amber-400/10 text-amber-500 dark:ring-amber-400/30 dark:bg-amber-400/10 dark:text-amber-400',
23-
},
24-
rose: {
25-
small: 'text-red-500 dark:text-rose-500',
26-
medium:
27-
'ring-rose-200 bg-rose-50 text-red-500 dark:ring-rose-500/20 dark:bg-rose-400/10 dark:text-rose-400',
28-
},
29-
zinc: {
30-
small: 'text-zinc-400 dark:text-zinc-500',
31-
medium:
32-
'ring-zinc-200 bg-zinc-50 text-zinc-500 dark:ring-zinc-500/20 dark:bg-zinc-400/10 dark:text-zinc-400',
33-
},
34-
}
35-
36-
const valueColorMap = {
37-
GET: 'emerald',
38-
POST: 'sky',
39-
PUT: 'amber',
40-
DELETE: 'rose',
41-
} as Record<string, keyof typeof colorStyles>
2+
import React from 'react'
423

434
export function Tag({
44-
children,
45-
variant = 'medium',
46-
color = valueColorMap[children] ?? 'emerald',
47-
}: {
48-
// eslint-disable-next-line @typescript-eslint/ban-types
49-
children: keyof typeof valueColorMap & (string | {})
50-
variant?: keyof typeof variantStyles
51-
color?: keyof typeof colorStyles
5+
children,
6+
}: {
7+
children: React.ReactNode
528
}) {
539
return (
5410
<span
5511
className={clsx(
56-
'relative top-[1px] font-mono text-[0.625rem] font-semibold',
57-
variantStyles[variant],
58-
colorStyles[color][variant],
12+
'relative top-[1.5px] font-mono text-[0.625rem] font-semibold',
13+
'text-brand-400 bg-brand-1000 ring-1 ring-inset rounded-lg px-1.5 ring-brand-400/20',
5914
)}
6015
>
6116
{children}
6217
</span>
6318
)
64-
}
19+
}

0 commit comments

Comments
 (0)