惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

T
Tailwind CSS Blog
H
Heimdal Security Blog
The Register - Security
The Register - Security
奇客Solidot–传递最新科技情报
奇客Solidot–传递最新科技情报
博客园 - 聂微东
Apple Machine Learning Research
Apple Machine Learning Research
Engineering at Meta
Engineering at Meta
Hugging Face - Blog
Hugging Face - Blog
大猫的无限游戏
大猫的无限游戏
Recent Announcements
Recent Announcements
博客园 - Franky
G
Google Developers Blog
OSCHINA 社区最新新闻
OSCHINA 社区最新新闻
Google DeepMind News
Google DeepMind News
B
Blog
让小产品的独立变现更简单 - ezindie.com
让小产品的独立变现更简单 - ezindie.com
美团技术团队
酷 壳 – CoolShell
酷 壳 – CoolShell
博客园 - 司徒正美
IT之家
IT之家
博客园 - 【当耐特】
Hacker News: Ask HN
Hacker News: Ask HN
SecWiki News
SecWiki News
有赞技术团队
有赞技术团队
H
Hacker News: Front Page
Application and Cybersecurity Blog
Application and Cybersecurity Blog
S
Security Affairs
S
Securelist
Last Week in AI
Last Week in AI
L
Lohrmann on Cybersecurity
博客园_首页
T
Troy Hunt's Blog
N
News and Events Feed by Topic
www.infosecurity-magazine.com
www.infosecurity-magazine.com
C
Cyber Attacks, Cyber Crime and Cyber Security
J
Java Code Geeks
V
Visual Studio Blog
量子位
AWS News Blog
AWS News Blog
罗磊的独立博客
S
SegmentFault 最新的问题
V
V2EX
腾讯CDC
Spread Privacy
Spread Privacy
Help Net Security
Help Net Security
S
Security Archives - TechRepublic
Attack and Defense Labs
Attack and Defense Labs
Cyberwarzone
Cyberwarzone
Project Zero
Project Zero
P
Privacy & Cybersecurity Law Blog

Recent Commits to openclaw:main

test: merge chat side-result checks · openclaw/openclaw@ddd2c2a test: merge cron history checks · openclaw/openclaw@f7eb746 test: merge responsive navigation shell checks · openclaw/openclaw@c2e4b47 docs(changelog): add codex oauth fixes · openclaw/openclaw@628e6cd test: merge navigation routing cases · openclaw/openclaw@5d8cecb Tests: mock channel registry bundled fallback · openclaw/openclaw@2b08233 Secrets: avoid broad web search discovery for single plugin config · openclaw/openclaw@a464f59 test: merge config view browser checks · openclaw/openclaw@20cf511 fix(status): align oauth health with runtime · openclaw/openclaw@eed7116 feat: add macOS screen snapshots for monitor preview (#67954) thanks … · openclaw/openclaw@f377db1 fix: report shared auth scopes in hello-ok (#67810) thanks @BunsDev · openclaw/openclaw@0b6c39b Auto-reply: avoid eager bundled route fallback · openclaw/openclaw@3ea1bf4 Tests: narrow session binding contract setup · openclaw/openclaw@54e4e16 fix(macOS): enable undo/redo in webchat composer text input (#34962) · openclaw/openclaw@00951dc Tests: speed up channel setup promotion · openclaw/openclaw@82b529a Docs: refresh agent instructions · openclaw/openclaw@5775fe2 fix(auth): serialize OAuth refresh across agents to fix #26322 (#67876) · openclaw/openclaw@8e79080 test: allow ollama public surface boundary test · openclaw/openclaw@7d4f1a6 Docs: add test performance guardrails · openclaw/openclaw@89706d3 Tests: restore context-engine usage proof · openclaw/openclaw@e4c4f95 Tests: slim context engine runtime coverage · openclaw/openclaw@74c198f ci: retry failed custom checkouts · openclaw/openclaw@0ee5baf test: trim duplicate provider auth onboarding cases · openclaw/openclaw@1ffc02e matrix: fix sessions_spawn --thread subagent session spawning (#67643) · openclaw/openclaw@1ce2596 test: reduce auth choice fixture churn · openclaw/openclaw@857b9cd test: mock health status config boundaries · openclaw/openclaw@9d5ab4a test: mock onboard config io boundary · openclaw/openclaw@299694d test: mock legacy state plugin boundaries · openclaw/openclaw@2713089 test: mock channel install boundaries · openclaw/openclaw@b945248 test: mock doctor preview channel boundaries · openclaw/openclaw@b1a3ad4 test: trim doctor command hotspots · openclaw/openclaw@c66f16a test: isolate agent auth and spawn hotspots · openclaw/openclaw@9285935 test: stabilize MCP startup disposal race · openclaw/openclaw@dd9d2eb test: merge browser contract server suites · openclaw/openclaw@5817a76 test: narrow ollama provider discovery setup · openclaw/openclaw@a0d9598 build: declare qa-lab aimock runtime dependency · openclaw/openclaw@24431e5 test: speed up safe-bins exec harness · openclaw/openclaw@ee856ab test: preserve tool helpers in embedded runner mocks · openclaw/openclaw@acd86a0 refactor: move memory embeddings into provider plugins · openclaw/openclaw@77e6e4c test: reuse system-run temp fixtures · openclaw/openclaw@7e9ff0f test: trim hotspot wait overhead · openclaw/openclaw@12a59b0 Check: avoid duplicate boundary prep · openclaw/openclaw@baf11b8 test: reduce hotspot fixture overhead · openclaw/openclaw@3a59edd feat(ui): overhaul settings and slash command UX (#67819) thanks @Bun… · openclaw/openclaw@2cfb660 QA Matrix: exit cleanly on failure · openclaw/openclaw@42805d2 QA Matrix: isolate scenario coverage · openclaw/openclaw@7e659e1 Matrix: refresh crypto bootstrap state · openclaw/openclaw@94081d8 QA Lab: add provider registry · openclaw/openclaw@bb7e982 Matrix: add plugin changelog · openclaw/openclaw@4acab55 test: trim more hotspot overhead · openclaw/openclaw@f485311 test: trim remaining hotspot tests · openclaw/openclaw@6ba8626 test: narrow hotspot mocks · openclaw/openclaw@dbc8179 test: isolate gemini embedding request helpers · openclaw/openclaw@cd330f5 test: trim memory and mcp hotspots · openclaw/openclaw@fd48dfa test: slim provider registry mocks · openclaw/openclaw@2e08c77 test: harden Parallels update smoke · openclaw/openclaw@1a98090 feat: default Anthropic to Opus 4.7 · openclaw/openclaw@628b454 fix: harden node-host shell payload mutability checks · openclaw/openclaw@75c551e fix: land node-host approval binding for native binaries (#66731) (th… · openclaw/openclaw@29919bb CI: add daily schedule to CodeQL workflow (#67645) · openclaw/openclaw@69d25f5 fix(gateway): capture config hash after plugin auto-enable to prevent… · openclaw/openclaw@8c11210 fix: repair sanitized replay tool results before send (#67620) (thank… · openclaw/openclaw@c3c7a99 fix: restrict HTML timeout short-circuit to transient statuses · openclaw/openclaw@de129a6 fix: keep TUI watchdog bound to active run (#67401) (thanks @xantorres) · openclaw/openclaw@3525273 Gateway/skills: dedupe skills prefix-match + drop dead fallback on log · openclaw/openclaw@d7f489f Extensions/lmstudio: back off inference preload after consecutive fai… · openclaw/openclaw@b555214 TUI/streaming: add watchdog that resets the activity indicator after … · openclaw/openclaw@f44ab20 Agents/tool-loop: enable unknown-tool stream guard by default · openclaw/openclaw@36ed367 Gateway/skills: invalidate session skills snapshot on config write · openclaw/openclaw@b23d59a fix: classify HTML provider error pages correctly (#67642) (thanks @s… · openclaw/openclaw@e588e90 fix(skills): remove unused model-usage import (#67641) · openclaw/openclaw@55f05df docs(changelog): credit codex fix superseded PRs · openclaw/openclaw@e485f24 fix(openai-codex): normalize stale transport metadata in resolution a… · openclaw/openclaw@90801ba CI: pin Docker-related GitHub Actions (#67632) · openclaw/openclaw@f697b01 Android: modernize WebView and discovery API usage (#67627) · openclaw/openclaw@44a6e50 fix(deps): bump hono to 4.12.14 and @hono/node-server to 1.19.14 (GHS… · openclaw/openclaw@fbccc18 fix(deps): bump dompurify to 3.4.0 (#67614) · openclaw/openclaw@2c2dc00 CI: add explicit permissions to all workflow jobs (fixes code-scannin… · openclaw/openclaw@01b7516 fix: register bundled TTS providers and route overrides correctly (#6… · openclaw/openclaw@6ea3cdd fix: align host tilde paths with OS home (#62804) (thanks @stainlu) · openclaw/openclaw@ecfaf64 fix: flush creds queue before reconnect socket open (#67464) (thanks … · openclaw/openclaw@405c63f fix: strip standalone <function> tool call tags from visible text (#6… · openclaw/openclaw@78df859 fix(agents): preserve cli session metadata before transcript persist … · openclaw/openclaw@898fd04 docs(changelog): move cli transcript entry · openclaw/openclaw@c1817c6 fix(agents): normalize cli transcript api field · openclaw/openclaw@3a3fae0 docs(changelog): note cli transcript persistence · openclaw/openclaw@6c343f1 fix(agents): persist cli transcript turns · openclaw/openclaw@b8ef507 fix(msteams): harden security-sensitive flows (#65841) · openclaw/openclaw@c56b56e [Dashboard] Fix exec approval modal overflow for long command content… · openclaw/openclaw@053c5b0 Docs: remove QA changelog entry · openclaw/openclaw@7fd5771 QA: fix private runtime source loading (#67428) · openclaw/openclaw@d5933af docs(gateway): correct protocol.md schema path, hello-ok example, aut… · openclaw/openclaw@489404d CI: pin Node 22 runners to 22.18.0 · openclaw/openclaw@4ffa621 models.authStatus: normalize provider ids + tighten env-backed escape… · openclaw/openclaw@f2fdb9d Update CHANGELOG.md · openclaw/openclaw@7694a92 test(parallels): clean up npm update guard jobs · openclaw/openclaw@045ea7b Plugins: prefer scanDir override paths · openclaw/openclaw@b2974da fix(dreaming): default storage.mode to "separate" so phase blocks sto… · openclaw/openclaw@8c392f0 fix(memory-core): skip dreaming transcript ingestion via session stor… · openclaw/openclaw@a1b01f0 fix: dedupe replayed exec.finished node events (#67281) · openclaw/openclaw@5dcf526
fix(openshell): pin mirror remote mutations (#93361) · openclaw/openclaw@ee81082
eleqtrizit · 2026-06-16 · via Recent Commits to openclaw:main

@@ -45,6 +45,107 @@ type PendingExec = {

4545

};

46464747

const MATERIALIZED_SKILLS_REMOTE_PARTS = [".openclaw", "sandbox-skills"] as const;

48+

export const PINNED_REMOTE_PATH_MUTATION_SCRIPT = [

49+

"set -eu",

50+

'die() { echo "$1" >&2; exit 1; }',

51+

"validate_basename() {",

52+

' case "$1" in ""|"."|".."|*/*) die "unsafe remote basename: $1" ;; esac',

53+

"}",

54+

"pin_dir() {",

55+

' root="$1"',

56+

' relative="$2"',

57+

' create="$3"',

58+

' case "$root" in /*) ;; *) die "remote root must be absolute: $root" ;; esac',

59+

' root="${root%/}"',

60+

' [ -n "$root" ] || root="/"',

61+

' if [ -L "$root" ]; then die "unsafe remote root symlink: $root"; fi',

62+

' mkdir -p -- "$root"',

63+

' canonical_root="$(cd "$root" && pwd -P)"',

64+

' current="$canonical_root"',

65+

' relative="${relative#/}"',

66+

' while [ -n "$relative" ]; do',

67+

' part="${relative%%/*}"',

68+

' if [ "$part" = "$relative" ]; then relative=""; else relative="${relative#*/}"; fi',

69+

' [ -n "$part" ] || continue',

70+

' case "$part" in "."|"..") die "unsafe remote directory component: $part" ;; esac',

71+

' if [ "$current" = "/" ]; then next="/$part"; else next="$current/$part"; fi',

72+

' if [ -L "$next" ]; then die "unsafe remote directory symlink: $next"; fi',

73+

' if [ -e "$next" ]; then',

74+

' if [ ! -d "$next" ]; then die "unsafe remote directory component: $next"; fi',

75+

" else",

76+

' if [ "$create" != "1" ]; then die "remote directory not found: $next"; fi',

77+

' mkdir -- "$next"',

78+

" fi",

79+

' current="$next"',

80+

" done",

81+

' printf "%s\\n" "$current"',

82+

"}",

83+

"pin_dir_or_missing() {",

84+

' root="$1"',

85+

' relative="$2"',

86+

' missing_ok="$3"',

87+

' case "$root" in /*) ;; *) die "remote root must be absolute: $root" ;; esac',

88+

' root="${root%/}"',

89+

' [ -n "$root" ] || root="/"',

90+

' if [ -L "$root" ]; then die "unsafe remote root symlink: $root"; fi',

91+

' if [ ! -d "$root" ]; then',

92+

' if [ -e "$root" ]; then die "unsafe remote root component: $root"; fi',

93+

' if [ "$missing_ok" = "1" ]; then printf "\\n"; return 0; fi',

94+

' die "remote directory not found: $root"',

95+

" fi",

96+

' canonical_root="$(cd "$root" && pwd -P)"',

97+

' current="$canonical_root"',

98+

' relative="${relative#/}"',

99+

' while [ -n "$relative" ]; do',

100+

' part="${relative%%/*}"',

101+

' if [ "$part" = "$relative" ]; then relative=""; else relative="${relative#*/}"; fi',

102+

' [ -n "$part" ] || continue',

103+

' case "$part" in "."|"..") die "unsafe remote directory component: $part" ;; esac',

104+

' if [ "$current" = "/" ]; then next="/$part"; else next="$current/$part"; fi',

105+

' if [ -L "$next" ]; then die "unsafe remote directory symlink: $next"; fi',

106+

' if [ -e "$next" ]; then',

107+

' if [ ! -d "$next" ]; then die "unsafe remote directory component: $next"; fi',

108+

" else",

109+

' if [ "$missing_ok" = "1" ]; then printf "\\n"; return 0; fi',

110+

' die "remote directory not found: $next"',

111+

" fi",

112+

' current="$next"',

113+

" done",

114+

' printf "%s\\n" "$current"',

115+

"}",

116+

'operation="$1"',

117+

'case "$operation" in',

118+

" mkdirp)",

119+

' pin_dir "$2" "$3" 1 >/dev/null',

120+

" ;;",

121+

" remove)",

122+

' validate_basename "$4"',

123+

' parent="$(pin_dir_or_missing "$2" "$3" "${5:-0}")"',

124+

' [ -n "$parent" ] || exit 0',

125+

' target="$parent/$4"',

126+

' if [ -d "$target" ] && [ ! -L "$target" ]; then rm -rf -- "$target"; elif [ -e "$target" ] || [ -L "$target" ]; then rm -f -- "$target"; fi',

127+

" ;;",

128+

" removefile)",

129+

' validate_basename "$4"',

130+

' parent="$(pin_dir_or_missing "$2" "$3" "${5:-0}")"',

131+

' [ -n "$parent" ] || exit 0',

132+

' target="$parent/$4"',

133+

' if [ -d "$target" ] && [ ! -L "$target" ]; then rmdir -- "$target"; elif [ -e "$target" ] || [ -L "$target" ]; then rm -f -- "$target"; fi',

134+

" ;;",

135+

" rename)",

136+

' src_parent="$(pin_dir "$2" "$3" 0)"',

137+

' validate_basename "$4"',

138+

' dst_parent="$(pin_dir "$5" "$6" 1)"',

139+

' validate_basename "$7"',

140+

' if [ -L "$dst_parent/$7" ]; then die "unsafe remote rename target symlink: $dst_parent/$7"; fi',

141+

' if [ -d "$dst_parent/$7" ]; then die "unsafe remote rename target directory: $dst_parent/$7"; fi',

142+

' mv -- "$src_parent/$4" "$dst_parent/$7"',

143+

" ;;",

144+

" *)",

145+

' die "unknown remote path mutation: $operation"',

146+

" ;;",

147+

"esac",

148+

].join("\n");

48149

const ENSURE_REMOTE_REAL_DIRECTORY_SCRIPT = [

49150

"set -e",

50151

'target="$1"',

@@ -190,6 +291,11 @@ async function createOpenShellSandboxBackend(params: {

190291

remoteWorkspaceDir: params.pluginConfig.remoteWorkspaceDir,

191292

remoteAgentWorkspaceDir: params.pluginConfig.remoteAgentWorkspaceDir,

192293

runRemoteShellScript: async (command) => await impl.runRemoteShellScript(command),

294+

mkdirpRemotePath: async (remotePath, signal) => await impl.mkdirpRemotePath(remotePath, signal),

295+

removeRemotePath: async (remotePath, removeParams) =>

296+

await impl.removeRemotePath(remotePath, removeParams),

297+

renameRemotePath: async (fromRemotePath, toRemotePath, signal) =>

298+

await impl.renameRemotePath(fromRemotePath, toRemotePath, signal),

193299

syncLocalPathToRemote: async (localPath, remotePath) =>

194300

await impl.syncLocalPathToRemote(localPath, remotePath),

195301

};

@@ -244,6 +350,12 @@ class OpenShellSandboxBackendImpl {

244350

backend: this.asHandle(),

245351

}),

246352

runRemoteShellScript: async (command) => await this.runRemoteShellScript(command),

353+

mkdirpRemotePath: async (remotePath, signal) =>

354+

await this.mkdirpRemotePath(remotePath, signal),

355+

removeRemotePath: async (remotePath, removeParams) =>

356+

await this.removeRemotePath(remotePath, removeParams),

357+

renameRemotePath: async (fromRemotePath, toRemotePath, signal) =>

358+

await this.renameRemotePath(fromRemotePath, toRemotePath, signal),

247359

syncLocalPathToRemote: async (localPath, remotePath) =>

248360

await this.syncLocalPathToRemote(localPath, remotePath),

249361

};

@@ -310,6 +422,58 @@ class OpenShellSandboxBackendImpl {

310422

return await this.runRemoteShellScriptInternal(params);

311423

}

312424425+

async mkdirpRemotePath(remotePath: string, signal?: AbortSignal): Promise<void> {

426+

const target = this.resolveRemoteTarget(remotePath);

427+

await this.runPinnedRemotePathMutation({

428+

args: ["mkdirp", target.root, target.relativePath],

429+

signal,

430+

});

431+

}

432+433+

async removeRemotePath(

434+

remotePath: string,

435+

params?: {

436+

recursive?: boolean;

437+

signal?: AbortSignal;

438+

ignoreMissing?: boolean;

439+

},

440+

): Promise<void> {

441+

const target = this.resolveRemoteTarget(remotePath);

442+

await this.runPinnedRemotePathMutation({

443+

args: [

444+

params?.recursive ? "remove" : "removefile",

445+

target.root,

446+

path.posix.dirname(target.relativePath) === "."

447+

? ""

448+

: path.posix.dirname(target.relativePath),

449+

path.posix.basename(target.relativePath),

450+

params?.ignoreMissing ? "1" : "0",

451+

],

452+

signal: params?.signal,

453+

});

454+

}

455+456+

async renameRemotePath(

457+

fromRemotePath: string,

458+

toRemotePath: string,

459+

signal?: AbortSignal,

460+

): Promise<void> {

461+

const from = this.resolveRemoteTarget(fromRemotePath);

462+

const to = this.resolveRemoteTarget(toRemotePath);

463+

await this.runPinnedRemotePathMutation({

464+

args: [

465+

"rename",

466+

from.root,

467+

path.posix.dirname(from.relativePath) === "." ? "" : path.posix.dirname(from.relativePath),

468+

path.posix.basename(from.relativePath),

469+

to.root,

470+

path.posix.dirname(to.relativePath) === "." ? "" : path.posix.dirname(to.relativePath),

471+

path.posix.basename(to.relativePath),

472+

],

473+

signal,

474+

});

475+

}

476+313477

private async runRemoteShellScriptInternal(

314478

params: SandboxBackendCommandParams,

315479

): Promise<SandboxBackendCommandResult> {

@@ -338,33 +502,48 @@ class OpenShellSandboxBackendImpl {

338502

async syncLocalPathToRemote(localPath: string, remotePath: string): Promise<void> {

339503

await this.ensureSandboxExists();

340504

await this.maybeSeedRemoteWorkspace();

505+

const target = this.resolveRemoteTarget(remotePath);

341506

const stats = await fs.lstat(localPath).catch(() => null);

342507

if (!stats) {

343-

await this.runRemoteShellScript({

344-

script: 'rm -rf -- "$1"',

345-

args: [remotePath],

346-

allowFailure: true,

508+

await this.runPinnedRemotePathMutation({

509+

args: [

510+

"remove",

511+

target.root,

512+

path.posix.dirname(target.relativePath) === "."

513+

? ""

514+

: path.posix.dirname(target.relativePath),

515+

path.posix.basename(target.relativePath),

516+

"1",

517+

],

347518

});

348519

return;

349520

}

350521

if (stats.isSymbolicLink()) {

351-

await this.runRemoteShellScript({

352-

script: 'rm -rf -- "$1"',

353-

args: [remotePath],

354-

allowFailure: true,

522+

await this.runPinnedRemotePathMutation({

523+

args: [

524+

"remove",

525+

target.root,

526+

path.posix.dirname(target.relativePath) === "."

527+

? ""

528+

: path.posix.dirname(target.relativePath),

529+

path.posix.basename(target.relativePath),

530+

"1",

531+

],

355532

});

356533

return;

357534

}

358535

if (stats.isDirectory()) {

359-

await this.runRemoteShellScript({

360-

script: 'mkdir -p -- "$1"',

361-

args: [remotePath],

362-

});

536+

await this.mkdirpRemotePath(remotePath);

363537

return;

364538

}

365-

await this.runRemoteShellScript({

366-

script: 'mkdir -p -- "$(dirname -- "$1")"',

367-

args: [remotePath],

539+

await this.runPinnedRemotePathMutation({

540+

args: [

541+

"mkdirp",

542+

target.root,

543+

path.posix.dirname(target.relativePath) === "."

544+

? ""

545+

: path.posix.dirname(target.relativePath),

546+

],

368547

});

369548

const result = await runOpenShellCli({

370549

context: this.params.execContext,

@@ -383,6 +562,32 @@ class OpenShellSandboxBackendImpl {

383562

}

384563

}

385564565+

private async runPinnedRemotePathMutation(params: {

566+

args: string[];

567+

signal?: AbortSignal;

568+

}): Promise<SandboxBackendCommandResult> {

569+

return await this.runRemoteShellScript({

570+

script: PINNED_REMOTE_PATH_MUTATION_SCRIPT,

571+

args: params.args,

572+

signal: params.signal,

573+

});

574+

}

575+576+

private resolveRemoteTarget(remotePath: string): { root: string; relativePath: string } {

577+

const normalized = normalizeRemotePath(remotePath);

578+

const roots = [

579+

normalizeRemotePath(this.params.remoteWorkspaceDir),

580+

normalizeRemotePath(this.params.remoteAgentWorkspaceDir),

581+

].toSorted((a, b) => b.length - a.length);

582+

for (const root of roots) {

583+

if (isRemotePathInside(root, normalized)) {

584+

const relativePath = path.posix.relative(root, normalized);

585+

return { root, relativePath: relativePath === "." ? "" : relativePath };

586+

}

587+

}

588+

throw new Error(`Remote path escapes OpenShell managed roots: ${remotePath}`);

589+

}

590+386591

private async ensureSandboxExists(): Promise<void> {

387592

if (this.ensurePromise) {

388593

return await this.ensurePromise;

@@ -676,3 +881,19 @@ async function restoreMaterializedSkillsShadow(params: {

676881

function resolveOpenShellTmpRoot(): string {

677882

return path.resolve(resolvePreferredOpenClawTmpDir());

678883

}

884+885+

function normalizeRemotePath(remotePath: string): string {

886+

const normalized = path.posix.normalize(remotePath.replace(/\\/g, "/"));

887+

if (!path.posix.isAbsolute(normalized)) {

888+

throw new Error(`OpenShell remote path must be absolute: ${remotePath}`);

889+

}

890+

return normalized;

891+

}

892+893+

function isRemotePathInside(root: string, candidate: string): boolean {

894+

const relative = path.posix.relative(root, candidate);

895+

return (

896+

relative === "" ||

897+

(relative !== ".." && !relative.startsWith("../") && !path.posix.isAbsolute(relative))

898+

);

899+

}