@prb/effect-evm 1.0.0-beta.4 → 1.0.0-beta.6
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/DOCS.md +13 -0
- package/README.md +1 -0
- package/dist/constants/index.d.ts +0 -4
- package/dist/constants/index.d.ts.map +1 -1
- package/dist/constants/index.js +0 -4
- package/dist/constants/index.js.map +1 -1
- package/dist/contract/pipeline/internal/core.d.ts +2 -2
- package/dist/contract/pipeline/internal/core.d.ts.map +1 -1
- package/dist/contract/pipeline/internal/core.js.map +1 -1
- package/dist/contract/pipeline/internal/receipt.d.ts +2 -2
- package/dist/contract/pipeline/internal/receipt.d.ts.map +1 -1
- package/dist/contract/pipeline/internal/receipt.js +1 -1
- package/dist/contract/pipeline/internal/receipt.js.map +1 -1
- package/dist/contract/pipeline/service.d.ts +3 -3
- package/dist/contract/pipeline/service.d.ts.map +1 -1
- package/dist/contract/pipeline/service.js.map +1 -1
- package/dist/contract/pipeline/types.d.ts +2 -2
- package/dist/contract/pipeline/types.d.ts.map +1 -1
- package/dist/contract/pipeline/types.js.map +1 -1
- package/dist/contract/pipeline/write-and-track.js +1 -1
- package/dist/contract/pipeline/write-and-track.js.map +1 -1
- package/dist/contract/pipeline/write-and-wait.d.ts +1 -1
- package/dist/contract/pipeline/write-and-wait.d.ts.map +1 -1
- package/dist/core/errors/transaction.d.ts +7 -7
- package/dist/core/errors/transaction.d.ts.map +1 -1
- package/dist/core/errors/transaction.js +2 -2
- package/dist/core/errors/transaction.js.map +1 -1
- package/dist/core/errors/viem-mapper.d.ts +3 -3
- package/dist/core/errors/viem-mapper.d.ts.map +1 -1
- package/dist/core/errors/viem-mapper.js +3 -3
- package/dist/core/errors/viem-mapper.js.map +1 -1
- package/dist/deploy/service.d.ts +3 -3
- package/dist/deploy/service.d.ts.map +1 -1
- package/dist/deploy/service.js.map +1 -1
- package/dist/eip7702/errors.d.ts +3 -3
- package/dist/eip7702/errors.d.ts.map +1 -1
- package/dist/eip7702/errors.js +1 -1
- package/dist/eip7702/errors.js.map +1 -1
- package/dist/eip7702/service.d.ts +8 -8
- package/dist/eip7702/service.d.ts.map +1 -1
- package/dist/eip7702/service.js +2 -2
- package/dist/eip7702/service.js.map +1 -1
- package/dist/react-hooks/index.d.ts +0 -4
- package/dist/react-hooks/index.d.ts.map +1 -1
- package/dist/react-hooks/index.js +0 -4
- package/dist/react-hooks/index.js.map +1 -1
- package/dist/subscriptions/pending-tx.d.ts +1 -1
- package/dist/subscriptions/pending-tx.d.ts.map +1 -1
- package/dist/subscriptions/pending-tx.js +1 -1
- package/dist/subscriptions/pending-tx.js.map +1 -1
- package/dist/subscriptions/service.d.ts +2 -2
- package/dist/subscriptions/service.d.ts.map +1 -1
- package/dist/subscriptions/service.js +4 -4
- package/dist/subscriptions/service.js.map +1 -1
- package/dist/telemetry/tracer.d.ts +2 -2
- package/dist/telemetry/tracer.js +2 -2
- package/dist/telemetry/tracer.js.map +1 -1
- package/dist/testing-kit/mock-subscription-service.d.ts +2 -2
- package/dist/testing-kit/mock-subscription-service.d.ts.map +1 -1
- package/dist/testing-kit/mock-subscription-service.js +5 -5
- package/dist/testing-kit/mock-subscription-service.js.map +1 -1
- package/dist/testing-kit/mock-transfer-service.d.ts +3 -3
- package/dist/testing-kit/mock-transfer-service.d.ts.map +1 -1
- package/dist/testing-kit/mock-transfer-service.js.map +1 -1
- package/dist/transfer/service.d.ts +3 -3
- package/dist/transfer/service.d.ts.map +1 -1
- package/dist/transfer/service.js +3 -3
- package/dist/transfer/service.js.map +1 -1
- package/dist/tx/manager.d.ts +2 -2
- package/dist/tx/manager.d.ts.map +1 -1
- package/dist/tx/manager.js +5 -5
- package/dist/tx/manager.js.map +1 -1
- package/dist/tx/manager.test.integration.js +4 -4
- package/dist/tx/manager.test.integration.js.map +1 -1
- package/dist/tx/replacement.d.ts +3 -3
- package/dist/tx/replacement.d.ts.map +1 -1
- package/dist/tx/replacement.js +3 -3
- package/dist/tx/replacement.js.map +1 -1
- package/dist/tx/tracker.d.ts +6 -6
- package/dist/tx/tracker.d.ts.map +1 -1
- package/dist/tx/tracker.js +9 -9
- package/dist/tx/tracker.js.map +1 -1
- package/dist/types/params.d.ts +1 -1
- package/dist/types/params.d.ts.map +1 -1
- package/dist/types/params.js.map +1 -1
- package/dist/wallet/errors.d.ts +3 -3
- package/dist/wallet/errors.d.ts.map +1 -1
- package/dist/wallet/errors.js +1 -1
- package/dist/wallet/errors.js.map +1 -1
- package/dist/wallet/operations.d.ts +2 -2
- package/dist/wallet/operations.d.ts.map +1 -1
- package/dist/wallet/operations.js +2 -2
- package/dist/wallet/operations.js.map +1 -1
- package/dist/wallet/service.d.ts +2 -2
- package/dist/wallet/service.d.ts.map +1 -1
- package/dist/wallet/service.js.map +1 -1
- package/package.json +1 -6
- package/dist/react-hooks/safe-origins.d.ts +0 -8
- package/dist/react-hooks/safe-origins.d.ts.map +0 -1
- package/dist/react-hooks/safe-origins.js +0 -124
- package/dist/react-hooks/safe-origins.js.map +0 -1
- package/dist/react-hooks/use-is-host-safe-multisig.d.ts +0 -2
- package/dist/react-hooks/use-is-host-safe-multisig.d.ts.map +0 -1
- package/dist/react-hooks/use-is-host-safe-multisig.js +0 -16
- package/dist/react-hooks/use-is-host-safe-multisig.js.map +0 -1
- package/dist/react-hooks/use-is-wallet-safe-multisig.d.ts +0 -2
- package/dist/react-hooks/use-is-wallet-safe-multisig.d.ts.map +0 -1
- package/dist/react-hooks/use-is-wallet-safe-multisig.js +0 -24
- package/dist/react-hooks/use-is-wallet-safe-multisig.js.map +0 -1
- package/dist/react-hooks/use-safe-context.d.ts +0 -2
- package/dist/react-hooks/use-safe-context.d.ts.map +0 -1
- package/dist/react-hooks/use-safe-context.js +0 -41
- package/dist/react-hooks/use-safe-context.js.map +0 -1
- package/dist/safe/adapter.d.ts +0 -9
- package/dist/safe/adapter.d.ts.map +0 -1
- package/dist/safe/adapter.js +0 -13
- package/dist/safe/adapter.js.map +0 -1
- package/dist/safe/detection.d.ts +0 -26
- package/dist/safe/detection.d.ts.map +0 -1
- package/dist/safe/detection.js +0 -102
- package/dist/safe/detection.js.map +0 -1
- package/dist/safe/detection.test.integration.d.ts +0 -2
- package/dist/safe/detection.test.integration.d.ts.map +0 -1
- package/dist/safe/detection.test.integration.js +0 -92
- package/dist/safe/detection.test.integration.js.map +0 -1
- package/dist/safe/errors.d.ts +0 -79
- package/dist/safe/errors.d.ts.map +0 -1
- package/dist/safe/errors.js +0 -34
- package/dist/safe/errors.js.map +0 -1
- package/dist/safe/index.d.ts +0 -8
- package/dist/safe/index.d.ts.map +0 -1
- package/dist/safe/index.js +0 -6
- package/dist/safe/index.js.map +0 -1
- package/dist/safe/live.d.ts +0 -8
- package/dist/safe/live.d.ts.map +0 -1
- package/dist/safe/live.js +0 -250
- package/dist/safe/live.js.map +0 -1
- package/dist/safe/service.d.ts +0 -26
- package/dist/safe/service.d.ts.map +0 -1
- package/dist/safe/service.js +0 -4
- package/dist/safe/service.js.map +0 -1
- package/dist/safe/service.test.integration.d.ts +0 -2
- package/dist/safe/service.test.integration.d.ts.map +0 -1
- package/dist/safe/service.test.integration.js +0 -173
- package/dist/safe/service.test.integration.js.map +0 -1
- package/dist/safe/simulation/abis.d.ts +0 -73
- package/dist/safe/simulation/abis.d.ts.map +0 -1
- package/dist/safe/simulation/abis.js +0 -61
- package/dist/safe/simulation/abis.js.map +0 -1
- package/dist/safe/simulation/addresses.d.ts +0 -4
- package/dist/safe/simulation/addresses.d.ts.map +0 -1
- package/dist/safe/simulation/addresses.js +0 -54
- package/dist/safe/simulation/addresses.js.map +0 -1
- package/dist/safe/simulation/encoding.d.ts +0 -16
- package/dist/safe/simulation/encoding.d.ts.map +0 -1
- package/dist/safe/simulation/encoding.js +0 -36
- package/dist/safe/simulation/encoding.js.map +0 -1
- package/dist/safe/simulation/errors.d.ts +0 -56
- package/dist/safe/simulation/errors.d.ts.map +0 -1
- package/dist/safe/simulation/errors.js +0 -37
- package/dist/safe/simulation/errors.js.map +0 -1
- package/dist/safe/simulation/index.d.ts +0 -7
- package/dist/safe/simulation/index.d.ts.map +0 -1
- package/dist/safe/simulation/index.js +0 -6
- package/dist/safe/simulation/index.js.map +0 -1
- package/dist/safe/simulation/internal/calldata/calldata.d.ts +0 -5
- package/dist/safe/simulation/internal/calldata/calldata.d.ts.map +0 -1
- package/dist/safe/simulation/internal/calldata/calldata.js +0 -17
- package/dist/safe/simulation/internal/calldata/calldata.js.map +0 -1
- package/dist/safe/simulation/internal/calldata/index.d.ts +0 -2
- package/dist/safe/simulation/internal/calldata/index.d.ts.map +0 -1
- package/dist/safe/simulation/internal/calldata/index.js +0 -2
- package/dist/safe/simulation/internal/calldata/index.js.map +0 -1
- package/dist/safe/simulation/internal/contracts/contracts.d.ts +0 -5
- package/dist/safe/simulation/internal/contracts/contracts.d.ts.map +0 -1
- package/dist/safe/simulation/internal/contracts/contracts.js +0 -25
- package/dist/safe/simulation/internal/contracts/contracts.js.map +0 -1
- package/dist/safe/simulation/internal/contracts/index.d.ts +0 -2
- package/dist/safe/simulation/internal/contracts/index.d.ts.map +0 -1
- package/dist/safe/simulation/internal/contracts/index.js +0 -2
- package/dist/safe/simulation/internal/contracts/index.js.map +0 -1
- package/dist/safe/simulation/internal/evaluation/evaluation.d.ts +0 -6
- package/dist/safe/simulation/internal/evaluation/evaluation.d.ts.map +0 -1
- package/dist/safe/simulation/internal/evaluation/evaluation.js +0 -20
- package/dist/safe/simulation/internal/evaluation/evaluation.js.map +0 -1
- package/dist/safe/simulation/internal/evaluation/index.d.ts +0 -2
- package/dist/safe/simulation/internal/evaluation/index.d.ts.map +0 -1
- package/dist/safe/simulation/internal/evaluation/index.js +0 -2
- package/dist/safe/simulation/internal/evaluation/index.js.map +0 -1
- package/dist/safe/simulation/internal/execution/execution.d.ts +0 -9
- package/dist/safe/simulation/internal/execution/execution.d.ts.map +0 -1
- package/dist/safe/simulation/internal/execution/execution.js +0 -65
- package/dist/safe/simulation/internal/execution/execution.js.map +0 -1
- package/dist/safe/simulation/internal/execution/index.d.ts +0 -2
- package/dist/safe/simulation/internal/execution/index.d.ts.map +0 -1
- package/dist/safe/simulation/internal/execution/index.js +0 -2
- package/dist/safe/simulation/internal/execution/index.js.map +0 -1
- package/dist/safe/simulation/internal/limits/index.d.ts +0 -2
- package/dist/safe/simulation/internal/limits/index.d.ts.map +0 -1
- package/dist/safe/simulation/internal/limits/index.js +0 -2
- package/dist/safe/simulation/internal/limits/index.js.map +0 -1
- package/dist/safe/simulation/internal/limits/limits.d.ts +0 -5
- package/dist/safe/simulation/internal/limits/limits.d.ts.map +0 -1
- package/dist/safe/simulation/internal/limits/limits.js +0 -18
- package/dist/safe/simulation/internal/limits/limits.js.map +0 -1
- package/dist/safe/simulation/internal/types/index.d.ts +0 -2
- package/dist/safe/simulation/internal/types/index.d.ts.map +0 -1
- package/dist/safe/simulation/internal/types/index.js +0 -2
- package/dist/safe/simulation/internal/types/index.js.map +0 -1
- package/dist/safe/simulation/internal/types/types.d.ts +0 -11
- package/dist/safe/simulation/internal/types/types.d.ts.map +0 -1
- package/dist/safe/simulation/internal/types/types.js +0 -2
- package/dist/safe/simulation/internal/types/types.js.map +0 -1
- package/dist/safe/simulation/internal/validation/index.d.ts +0 -2
- package/dist/safe/simulation/internal/validation/index.d.ts.map +0 -1
- package/dist/safe/simulation/internal/validation/index.js +0 -2
- package/dist/safe/simulation/internal/validation/index.js.map +0 -1
- package/dist/safe/simulation/internal/validation/validation.d.ts +0 -5
- package/dist/safe/simulation/internal/validation/validation.d.ts.map +0 -1
- package/dist/safe/simulation/internal/validation/validation.js +0 -27
- package/dist/safe/simulation/internal/validation/validation.js.map +0 -1
- package/dist/safe/simulation/service.d.ts +0 -14
- package/dist/safe/simulation/service.d.ts.map +0 -1
- package/dist/safe/simulation/service.js +0 -25
- package/dist/safe/simulation/service.js.map +0 -1
- package/dist/safe/simulation/types.d.ts +0 -20
- package/dist/safe/simulation/types.d.ts.map +0 -1
- package/dist/safe/simulation/types.js +0 -2
- package/dist/safe/simulation/types.js.map +0 -1
- package/dist/safe/types.d.ts +0 -61
- package/dist/safe/types.d.ts.map +0 -1
- package/dist/safe/types.js +0 -2
- package/dist/safe/types.js.map +0 -1
|
@@ -1,124 +0,0 @@
|
|
|
1
|
-
"use client";
|
|
2
|
-
export const DEFAULT_SAFE_ORIGINS = [
|
|
3
|
-
"https://app.safe.global",
|
|
4
|
-
"https://gnosis-safe.io",
|
|
5
|
-
"https://safe.global",
|
|
6
|
-
"https://safe.berachain.com",
|
|
7
|
-
"https://safe.chiliz.com",
|
|
8
|
-
"https://safe.lightlink.io",
|
|
9
|
-
"https://safe.optimism.io",
|
|
10
|
-
];
|
|
11
|
-
const ORIGIN_PROTOCOL_PATTERN = /^[a-z]+:\/\//i;
|
|
12
|
-
let safeOrigins = normalizeOrigins(DEFAULT_SAFE_ORIGINS);
|
|
13
|
-
let safeOriginSet = new Set(safeOrigins);
|
|
14
|
-
const safeOriginListeners = new Set();
|
|
15
|
-
let safeOriginsConfigured = false;
|
|
16
|
-
export function setSafeOrigins(origins) {
|
|
17
|
-
configureSafeOrigins(normalizeOrigins(origins));
|
|
18
|
-
}
|
|
19
|
-
export function extendSafeOrigins(origins) {
|
|
20
|
-
configureSafeOrigins(normalizeOrigins([...safeOrigins, ...origins]));
|
|
21
|
-
}
|
|
22
|
-
export function getSafeOrigins() {
|
|
23
|
-
return [...safeOrigins];
|
|
24
|
-
}
|
|
25
|
-
export function subscribeSafeOrigins(listener) {
|
|
26
|
-
safeOriginListeners.add(listener);
|
|
27
|
-
return () => {
|
|
28
|
-
safeOriginListeners.delete(listener);
|
|
29
|
-
};
|
|
30
|
-
}
|
|
31
|
-
export function isValidSafeOrigin() {
|
|
32
|
-
const origin = getAncestorOrigin();
|
|
33
|
-
if (!origin) {
|
|
34
|
-
return false;
|
|
35
|
-
}
|
|
36
|
-
return safeOriginSet.has(origin);
|
|
37
|
-
}
|
|
38
|
-
export function isHostEmbedded() {
|
|
39
|
-
return typeof window !== "undefined" && window.parent !== window;
|
|
40
|
-
}
|
|
41
|
-
function getAncestorOrigin() {
|
|
42
|
-
if (typeof window === "undefined") {
|
|
43
|
-
return null;
|
|
44
|
-
}
|
|
45
|
-
try {
|
|
46
|
-
return window.parent.location.origin;
|
|
47
|
-
}
|
|
48
|
-
catch {
|
|
49
|
-
if (window.location.ancestorOrigins?.length) {
|
|
50
|
-
return window.location.ancestorOrigins[0];
|
|
51
|
-
}
|
|
52
|
-
if (document.referrer) {
|
|
53
|
-
try {
|
|
54
|
-
return new URL(document.referrer).origin;
|
|
55
|
-
}
|
|
56
|
-
catch {
|
|
57
|
-
return null;
|
|
58
|
-
}
|
|
59
|
-
}
|
|
60
|
-
return null;
|
|
61
|
-
}
|
|
62
|
-
}
|
|
63
|
-
function updateSafeOrigins(nextOrigins) {
|
|
64
|
-
if (areSameOrigins(safeOrigins, nextOrigins)) {
|
|
65
|
-
return;
|
|
66
|
-
}
|
|
67
|
-
safeOrigins = nextOrigins;
|
|
68
|
-
safeOriginSet = new Set(safeOrigins);
|
|
69
|
-
notifySafeOriginListeners();
|
|
70
|
-
}
|
|
71
|
-
function configureSafeOrigins(nextOrigins) {
|
|
72
|
-
if (safeOriginsConfigured) {
|
|
73
|
-
throw new Error("Safe origins already configured.");
|
|
74
|
-
}
|
|
75
|
-
safeOriginsConfigured = true;
|
|
76
|
-
updateSafeOrigins(nextOrigins);
|
|
77
|
-
}
|
|
78
|
-
function notifySafeOriginListeners() {
|
|
79
|
-
for (const listener of safeOriginListeners) {
|
|
80
|
-
listener();
|
|
81
|
-
}
|
|
82
|
-
}
|
|
83
|
-
function normalizeOrigins(origins) {
|
|
84
|
-
const normalized = [];
|
|
85
|
-
const seen = new Set();
|
|
86
|
-
for (const origin of origins) {
|
|
87
|
-
const normalizedOrigin = normalizeOrigin(origin);
|
|
88
|
-
if (!normalizedOrigin || seen.has(normalizedOrigin)) {
|
|
89
|
-
continue;
|
|
90
|
-
}
|
|
91
|
-
seen.add(normalizedOrigin);
|
|
92
|
-
normalized.push(normalizedOrigin);
|
|
93
|
-
}
|
|
94
|
-
return normalized;
|
|
95
|
-
}
|
|
96
|
-
function normalizeOrigin(origin) {
|
|
97
|
-
const trimmed = origin.trim();
|
|
98
|
-
if (!trimmed) {
|
|
99
|
-
return null;
|
|
100
|
-
}
|
|
101
|
-
const candidate = ORIGIN_PROTOCOL_PATTERN.test(trimmed) ? trimmed : `https://${trimmed}`;
|
|
102
|
-
try {
|
|
103
|
-
const url = new URL(candidate);
|
|
104
|
-
if (url.protocol !== "https:" && url.protocol !== "http:") {
|
|
105
|
-
return null;
|
|
106
|
-
}
|
|
107
|
-
return url.origin;
|
|
108
|
-
}
|
|
109
|
-
catch {
|
|
110
|
-
return null;
|
|
111
|
-
}
|
|
112
|
-
}
|
|
113
|
-
function areSameOrigins(left, right) {
|
|
114
|
-
if (left.length !== right.length) {
|
|
115
|
-
return false;
|
|
116
|
-
}
|
|
117
|
-
for (let index = 0; index < left.length; index += 1) {
|
|
118
|
-
if (left[index] !== right[index]) {
|
|
119
|
-
return false;
|
|
120
|
-
}
|
|
121
|
-
}
|
|
122
|
-
return true;
|
|
123
|
-
}
|
|
124
|
-
//# sourceMappingURL=safe-origins.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"safe-origins.js","sourceRoot":"","sources":["../../src/react-hooks/safe-origins.ts"],"names":[],"mappings":"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","sourcesContent":["\"use client\";\n\n/**\n * Known Safe wallet domains for iframe origin validation.\n * Includes the main Safe app and chain-specific Safe deployments.\n */\nexport const DEFAULT_SAFE_ORIGINS = [\n // Main Safe domains\n \"https://app.safe.global\",\n \"https://gnosis-safe.io\",\n \"https://safe.global\",\n // Chain-specific Safe deployments\n \"https://safe.berachain.com\",\n \"https://safe.chiliz.com\",\n \"https://safe.lightlink.io\",\n \"https://safe.optimism.io\",\n] as const;\n\nconst ORIGIN_PROTOCOL_PATTERN = /^[a-z]+:\\/\\//i;\nlet safeOrigins = normalizeOrigins(DEFAULT_SAFE_ORIGINS);\nlet safeOriginSet = new Set(safeOrigins);\nconst safeOriginListeners = new Set<() => void>();\nlet safeOriginsConfigured = false;\n\n/** Replace the Safe origins list (one-time configuration). */\nexport function setSafeOrigins(origins: readonly string[]) {\n configureSafeOrigins(normalizeOrigins(origins));\n}\n\n/** Extend the Safe origins list (one-time configuration). */\nexport function extendSafeOrigins(origins: readonly string[]) {\n configureSafeOrigins(normalizeOrigins([...safeOrigins, ...origins]));\n}\n\n/** Read the currently configured Safe origins list. */\nexport function getSafeOrigins(): readonly string[] {\n return [...safeOrigins];\n}\n\n/** Subscribe to Safe origin changes. */\nexport function subscribeSafeOrigins(listener: () => void): () => void {\n safeOriginListeners.add(listener);\n return () => {\n safeOriginListeners.delete(listener);\n };\n}\n\n/** Check if the parent browsing context origin is a Safe domain. */\nexport function isValidSafeOrigin(): boolean {\n const origin = getAncestorOrigin();\n if (!origin) {\n return false;\n }\n return safeOriginSet.has(origin);\n}\n\n/** Check whether the app is running inside an iframe. */\nexport function isHostEmbedded(): boolean {\n return typeof window !== \"undefined\" && window.parent !== window;\n}\n\n/**\n * Get the origin of the parent browsing context.\n * Returns null if it cannot be determined due to cross-origin restrictions.\n */\nfunction getAncestorOrigin(): string | null {\n if (typeof window === \"undefined\") {\n return null;\n }\n\n try {\n return window.parent.location.origin;\n } catch {\n if (window.location.ancestorOrigins?.length) {\n return window.location.ancestorOrigins[0];\n }\n\n if (document.referrer) {\n try {\n return new URL(document.referrer).origin;\n } catch {\n return null;\n }\n }\n\n return null;\n }\n}\n\nfunction updateSafeOrigins(nextOrigins: string[]) {\n if (areSameOrigins(safeOrigins, nextOrigins)) {\n return;\n }\n\n safeOrigins = nextOrigins;\n safeOriginSet = new Set(safeOrigins);\n notifySafeOriginListeners();\n}\n\nfunction configureSafeOrigins(nextOrigins: string[]) {\n if (safeOriginsConfigured) {\n throw new Error(\"Safe origins already configured.\");\n }\n\n safeOriginsConfigured = true;\n updateSafeOrigins(nextOrigins);\n}\n\nfunction notifySafeOriginListeners() {\n for (const listener of safeOriginListeners) {\n listener();\n }\n}\n\nfunction normalizeOrigins(origins: readonly string[]): string[] {\n const normalized: string[] = [];\n const seen = new Set<string>();\n\n for (const origin of origins) {\n const normalizedOrigin = normalizeOrigin(origin);\n if (!normalizedOrigin || seen.has(normalizedOrigin)) {\n continue;\n }\n seen.add(normalizedOrigin);\n normalized.push(normalizedOrigin);\n }\n\n return normalized;\n}\n\nfunction normalizeOrigin(origin: string): string | null {\n const trimmed = origin.trim();\n if (!trimmed) {\n return null;\n }\n\n const candidate = ORIGIN_PROTOCOL_PATTERN.test(trimmed) ? trimmed : `https://${trimmed}`;\n\n try {\n const url = new URL(candidate);\n if (url.protocol !== \"https:\" && url.protocol !== \"http:\") {\n return null;\n }\n return url.origin;\n } catch {\n return null;\n }\n}\n\nfunction areSameOrigins(left: readonly string[], right: readonly string[]): boolean {\n if (left.length !== right.length) {\n return false;\n }\n\n for (let index = 0; index < left.length; index += 1) {\n if (left[index] !== right[index]) {\n return false;\n }\n }\n\n return true;\n}\n"]}
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"use-is-host-safe-multisig.d.ts","sourceRoot":"","sources":["../../src/react-hooks/use-is-host-safe-multisig.ts"],"names":[],"mappings":"AAkBA,wBAAgB,qBAAqB,IAAI,OAAO,CAS/C"}
|
|
@@ -1,16 +0,0 @@
|
|
|
1
|
-
"use client";
|
|
2
|
-
import { useSyncExternalStore } from "react";
|
|
3
|
-
import { isHostEmbedded, isValidSafeOrigin, subscribeSafeOrigins } from "./safe-origins.js";
|
|
4
|
-
import { useSafeContext } from "./use-safe-context.js";
|
|
5
|
-
export function useIsHostSafeMultisig() {
|
|
6
|
-
const isSafeContext = useSafeContext();
|
|
7
|
-
const isSafeHost = useSyncExternalStore(subscribeSafeOrigins, getSafeHostSnapshot, getServerSnapshot);
|
|
8
|
-
return isSafeContext || isSafeHost;
|
|
9
|
-
}
|
|
10
|
-
function getSafeHostSnapshot() {
|
|
11
|
-
return isHostEmbedded() && isValidSafeOrigin();
|
|
12
|
-
}
|
|
13
|
-
function getServerSnapshot() {
|
|
14
|
-
return false;
|
|
15
|
-
}
|
|
16
|
-
//# sourceMappingURL=use-is-host-safe-multisig.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"use-is-host-safe-multisig.js","sourceRoot":"","sources":["../../src/react-hooks/use-is-host-safe-multisig.ts"],"names":[],"mappings":"AAAA,YAAY,CAAC;AAEb,OAAO,EAAE,oBAAoB,EAAE,MAAM,OAAO,CAAC;AAC7C,OAAO,EAAE,cAAc,EAAE,iBAAiB,EAAE,oBAAoB,EAAE,MAAM,mBAAmB,CAAC;AAC5F,OAAO,EAAE,cAAc,EAAE,MAAM,uBAAuB,CAAC;AAcvD,MAAM,UAAU,qBAAqB;IACnC,MAAM,aAAa,GAAG,cAAc,EAAE,CAAC;IACvC,MAAM,UAAU,GAAG,oBAAoB,CACrC,oBAAoB,EACpB,mBAAmB,EACnB,iBAAiB,CAClB,CAAC;IAEF,OAAO,aAAa,IAAI,UAAU,CAAC;AACrC,CAAC;AAED,SAAS,mBAAmB;IAC1B,OAAO,cAAc,EAAE,IAAI,iBAAiB,EAAE,CAAC;AACjD,CAAC;AAED,SAAS,iBAAiB;IACxB,OAAO,KAAK,CAAC;AACf,CAAC","sourcesContent":["\"use client\";\n\nimport { useSyncExternalStore } from \"react\";\nimport { isHostEmbedded, isValidSafeOrigin, subscribeSafeOrigins } from \"./safe-origins.js\";\nimport { useSafeContext } from \"./use-safe-context.js\";\n\n/**\n * Detect if the app is running within a Safe context.\n *\n * This hook returns `true` when either:\n * - Safe Apps SDK confirms the Safe context (async, most reliable), or\n * - The app is embedded in a Safe-origin iframe (sync check).\n *\n * Notes:\n * - This answers “is the host Safe?” and will return true even without a wallet.\n * - For full wallet detection heuristics, use `useIsWalletSafeMultisig`.\n * - For SDK-only checks, use `useSafeContext`.\n */\nexport function useIsHostSafeMultisig(): boolean {\n const isSafeContext = useSafeContext();\n const isSafeHost = useSyncExternalStore(\n subscribeSafeOrigins,\n getSafeHostSnapshot,\n getServerSnapshot\n );\n\n return isSafeContext || isSafeHost;\n}\n\nfunction getSafeHostSnapshot(): boolean {\n return isHostEmbedded() && isValidSafeOrigin();\n}\n\nfunction getServerSnapshot(): boolean {\n return false;\n}\n"]}
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"use-is-wallet-safe-multisig.d.ts","sourceRoot":"","sources":["../../src/react-hooks/use-is-wallet-safe-multisig.ts"],"names":[],"mappings":"AAsBA,wBAAgB,uBAAuB,IAAI,OAAO,CAuBjD"}
|
|
@@ -1,24 +0,0 @@
|
|
|
1
|
-
"use client";
|
|
2
|
-
import { useSyncExternalStore } from "react";
|
|
3
|
-
import { useAccount } from "wagmi";
|
|
4
|
-
import { isHostEmbedded, isValidSafeOrigin, subscribeSafeOrigins } from "./safe-origins.js";
|
|
5
|
-
import { useSafeContext } from "./use-safe-context.js";
|
|
6
|
-
export function useIsWalletSafeMultisig() {
|
|
7
|
-
const { connector, isConnected } = useAccount();
|
|
8
|
-
const isSafeContext = useSafeContext();
|
|
9
|
-
const isSafeIframe = useSyncExternalStore(subscribeSafeOrigins, getSafeIframeSnapshot, getServerSnapshot);
|
|
10
|
-
if (isSafeContext) {
|
|
11
|
-
return true;
|
|
12
|
-
}
|
|
13
|
-
if (isConnected && connector?.id === "safe") {
|
|
14
|
-
return true;
|
|
15
|
-
}
|
|
16
|
-
return isSafeIframe;
|
|
17
|
-
}
|
|
18
|
-
function getSafeIframeSnapshot() {
|
|
19
|
-
return isHostEmbedded() && isValidSafeOrigin();
|
|
20
|
-
}
|
|
21
|
-
function getServerSnapshot() {
|
|
22
|
-
return false;
|
|
23
|
-
}
|
|
24
|
-
//# sourceMappingURL=use-is-wallet-safe-multisig.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"use-is-wallet-safe-multisig.js","sourceRoot":"","sources":["../../src/react-hooks/use-is-wallet-safe-multisig.ts"],"names":[],"mappings":"AAAA,YAAY,CAAC;AAEb,OAAO,EAAE,oBAAoB,EAAE,MAAM,OAAO,CAAC;AAC7C,OAAO,EAAE,UAAU,EAAE,MAAM,OAAO,CAAC;AACnC,OAAO,EAAE,cAAc,EAAE,iBAAiB,EAAE,oBAAoB,EAAE,MAAM,mBAAmB,CAAC;AAC5F,OAAO,EAAE,cAAc,EAAE,MAAM,uBAAuB,CAAC;AAiBvD,MAAM,UAAU,uBAAuB;IACrC,MAAM,EAAE,SAAS,EAAE,WAAW,EAAE,GAAG,UAAU,EAAE,CAAC;IAChD,MAAM,aAAa,GAAG,cAAc,EAAE,CAAC;IAGvC,MAAM,YAAY,GAAG,oBAAoB,CACvC,oBAAoB,EACpB,qBAAqB,EACrB,iBAAiB,CAClB,CAAC;IAGF,IAAI,aAAa,EAAE,CAAC;QAClB,OAAO,IAAI,CAAC;IACd,CAAC;IAGD,IAAI,WAAW,IAAI,SAAS,EAAE,EAAE,KAAK,MAAM,EAAE,CAAC;QAC5C,OAAO,IAAI,CAAC;IACd,CAAC;IAGD,OAAO,YAAY,CAAC;AACtB,CAAC;AAED,SAAS,qBAAqB;IAC5B,OAAO,cAAc,EAAE,IAAI,iBAAiB,EAAE,CAAC;AACjD,CAAC;AAED,SAAS,iBAAiB;IACxB,OAAO,KAAK,CAAC;AACf,CAAC","sourcesContent":["\"use client\";\n\nimport { useSyncExternalStore } from \"react\";\nimport { useAccount } from \"wagmi\";\nimport { isHostEmbedded, isValidSafeOrigin, subscribeSafeOrigins } from \"./safe-origins.js\";\nimport { useSafeContext } from \"./use-safe-context.js\";\n\n/**\n * Detect if the connected wallet is a Safe multisig.\n *\n * Detection strategy (in order of reliability):\n * 1. Safe Apps SDK detection via postMessage (most reliable, works cross-origin)\n * 2. Wagmi connector ID check\n * 3. Iframe origin validation (fallback, may fail cross-origin)\n *\n * @returns true if wallet is a Safe multisig\n *\n * Notes:\n * - This is a broader heuristic: SDK context, connector ID, then host origin.\n * - For just SDK context, use `useSafeContext`.\n * - For a sync host-only check, use `useIsHostSafeMultisig`.\n */\nexport function useIsWalletSafeMultisig(): boolean {\n const { connector, isConnected } = useAccount();\n const isSafeContext = useSafeContext();\n\n // Method 3: Iframe origin fallback (may fail cross-origin)\n const isSafeIframe = useSyncExternalStore(\n subscribeSafeOrigins,\n getSafeIframeSnapshot,\n getServerSnapshot\n );\n\n // Method 1: Safe Apps SDK (most reliable - uses postMessage)\n if (isSafeContext) {\n return true;\n }\n\n // Method 2: Connected via Safe connector\n if (isConnected && connector?.id === \"safe\") {\n return true;\n }\n\n // Method 3: Fallback\n return isSafeIframe;\n}\n\nfunction getSafeIframeSnapshot(): boolean {\n return isHostEmbedded() && isValidSafeOrigin();\n}\n\nfunction getServerSnapshot(): boolean {\n return false;\n}\n"]}
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"use-safe-context.d.ts","sourceRoot":"","sources":["../../src/react-hooks/use-safe-context.ts"],"names":[],"mappings":"AAuCA,wBAAgB,cAAc,IAAI,OAAO,CA6CxC"}
|
|
@@ -1,41 +0,0 @@
|
|
|
1
|
-
"use client";
|
|
2
|
-
import { useEffect, useState } from "react";
|
|
3
|
-
const SAFE_SDK_TIMEOUT = 3000;
|
|
4
|
-
export function useSafeContext() {
|
|
5
|
-
const [isSafe, setIsSafe] = useState(false);
|
|
6
|
-
useEffect(() => {
|
|
7
|
-
if (typeof window === "undefined") {
|
|
8
|
-
return;
|
|
9
|
-
}
|
|
10
|
-
let cancelled = false;
|
|
11
|
-
const timeout = setTimeout(() => {
|
|
12
|
-
if (!cancelled) {
|
|
13
|
-
setIsSafe(false);
|
|
14
|
-
}
|
|
15
|
-
}, SAFE_SDK_TIMEOUT);
|
|
16
|
-
import("@safe-global/safe-apps-sdk")
|
|
17
|
-
.then(({ default: SafeAppsSDK }) => {
|
|
18
|
-
const sdk = new SafeAppsSDK();
|
|
19
|
-
return sdk.safe.getInfo();
|
|
20
|
-
})
|
|
21
|
-
.then(() => {
|
|
22
|
-
if (!cancelled) {
|
|
23
|
-
setIsSafe(true);
|
|
24
|
-
}
|
|
25
|
-
})
|
|
26
|
-
.catch(() => {
|
|
27
|
-
if (!cancelled) {
|
|
28
|
-
setIsSafe(false);
|
|
29
|
-
}
|
|
30
|
-
})
|
|
31
|
-
.finally(() => {
|
|
32
|
-
clearTimeout(timeout);
|
|
33
|
-
});
|
|
34
|
-
return () => {
|
|
35
|
-
cancelled = true;
|
|
36
|
-
clearTimeout(timeout);
|
|
37
|
-
};
|
|
38
|
-
}, []);
|
|
39
|
-
return isSafe;
|
|
40
|
-
}
|
|
41
|
-
//# sourceMappingURL=use-safe-context.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"use-safe-context.js","sourceRoot":"","sources":["../../src/react-hooks/use-safe-context.ts"],"names":[],"mappings":"AAAA,YAAY,CAAC;AAEb,OAAO,EAAE,SAAS,EAAE,QAAQ,EAAE,MAAM,OAAO,CAAC;AAM5C,MAAM,gBAAgB,GAAG,IAAI,CAAC;AA+B9B,MAAM,UAAU,cAAc;IAC5B,MAAM,CAAC,MAAM,EAAE,SAAS,CAAC,GAAG,QAAQ,CAAC,KAAK,CAAC,CAAC;IAE5C,SAAS,CAAC,GAAG,EAAE;QAEb,IAAI,OAAO,MAAM,KAAK,WAAW,EAAE,CAAC;YAClC,OAAO;QACT,CAAC;QAED,IAAI,SAAS,GAAG,KAAK,CAAC;QAGtB,MAAM,OAAO,GAAG,UAAU,CAAC,GAAG,EAAE;YAC9B,IAAI,CAAC,SAAS,EAAE,CAAC;gBACf,SAAS,CAAC,KAAK,CAAC,CAAC;YACnB,CAAC;QACH,CAAC,EAAE,gBAAgB,CAAC,CAAC;QAGrB,MAAM,CAAC,4BAA4B,CAAC;aACjC,IAAI,CAAC,CAAC,EAAE,OAAO,EAAE,WAAW,EAAE,EAAE,EAAE;YACjC,MAAM,GAAG,GAAG,IAAI,WAAW,EAAE,CAAC;YAC9B,OAAO,GAAG,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC;QAC5B,CAAC,CAAC;aACD,IAAI,CAAC,GAAG,EAAE;YACT,IAAI,CAAC,SAAS,EAAE,CAAC;gBACf,SAAS,CAAC,IAAI,CAAC,CAAC;YAClB,CAAC;QACH,CAAC,CAAC;aACD,KAAK,CAAC,GAAG,EAAE;YACV,IAAI,CAAC,SAAS,EAAE,CAAC;gBACf,SAAS,CAAC,KAAK,CAAC,CAAC;YACnB,CAAC;QACH,CAAC,CAAC;aACD,OAAO,CAAC,GAAG,EAAE;YACZ,YAAY,CAAC,OAAO,CAAC,CAAC;QACxB,CAAC,CAAC,CAAC;QAEL,OAAO,GAAG,EAAE;YACV,SAAS,GAAG,IAAI,CAAC;YACjB,YAAY,CAAC,OAAO,CAAC,CAAC;QACxB,CAAC,CAAC;IACJ,CAAC,EAAE,EAAE,CAAC,CAAC;IAEP,OAAO,MAAM,CAAC;AAChB,CAAC","sourcesContent":["\"use client\";\n\nimport { useEffect, useState } from \"react\";\n\n/**\n * Timeout for Safe SDK detection in milliseconds.\n * Safe Apps SDK uses postMessage which can hang if not in Safe context.\n */\nconst SAFE_SDK_TIMEOUT = 3000;\n\n/**\n * Detect if the app is running in a Safe App context using the Safe Apps SDK.\n *\n * This hook asynchronously checks with the Safe Apps SDK to determine if\n * we're running inside a Safe App iframe. More reliable than iframe origin\n * detection because:\n * - Uses postMessage (works cross-origin)\n * - Gets definitive answer from Safe\n * - Includes timeout to prevent hangs\n *\n * @returns true if running in Safe App context, false otherwise\n *\n * Notes:\n * - This only verifies the Safe Apps SDK context (async).\n * - For a host-based sync check, use `useIsHostSafeMultisig`.\n * - For a full wallet detection heuristic, use `useIsWalletSafeMultisig`.\n *\n * @example\n * ```tsx\n * function MyComponent() {\n * const isSafeContext = useSafeContext();\n *\n * if (isSafeContext) {\n * return <p>Running in Safe App</p>;\n * }\n * return <p>Regular browser context</p>;\n * }\n * ```\n */\nexport function useSafeContext(): boolean {\n const [isSafe, setIsSafe] = useState(false);\n\n useEffect(() => {\n // SSR guard\n if (typeof window === \"undefined\") {\n return;\n }\n\n let cancelled = false;\n\n // Timeout fallback - resolve to false if SDK doesn't respond\n const timeout = setTimeout(() => {\n if (!cancelled) {\n setIsSafe(false);\n }\n }, SAFE_SDK_TIMEOUT);\n\n // Dynamically import SDK to keep it optional\n import(\"@safe-global/safe-apps-sdk\")\n .then(({ default: SafeAppsSDK }) => {\n const sdk = new SafeAppsSDK();\n return sdk.safe.getInfo();\n })\n .then(() => {\n if (!cancelled) {\n setIsSafe(true);\n }\n })\n .catch(() => {\n if (!cancelled) {\n setIsSafe(false);\n }\n })\n .finally(() => {\n clearTimeout(timeout);\n });\n\n return () => {\n cancelled = true;\n clearTimeout(timeout);\n };\n }, []);\n\n return isSafe;\n}\n"]}
|
package/dist/safe/adapter.d.ts
DELETED
|
@@ -1,9 +0,0 @@
|
|
|
1
|
-
import { Effect } from "effect";
|
|
2
|
-
import { SafeSdkUnavailableError } from "./errors.js";
|
|
3
|
-
export type SafeAppsSdkConfig = {
|
|
4
|
-
allowedDomains?: RegExp[];
|
|
5
|
-
debug?: boolean;
|
|
6
|
-
};
|
|
7
|
-
export declare const loadSafeSdk: (config?: SafeAppsSdkConfig) => Effect.Effect<import("@safe-global/safe-apps-sdk").default, SafeSdkUnavailableError, never>;
|
|
8
|
-
export type SafeAppsSDKInstance = any;
|
|
9
|
-
//# sourceMappingURL=adapter.d.ts.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"adapter.d.ts","sourceRoot":"","sources":["../../src/safe/adapter.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,MAAM,QAAQ,CAAC;AAChC,OAAO,EAAE,uBAAuB,EAAE,MAAM,aAAa,CAAC;AAGtD,MAAM,MAAM,iBAAiB,GAAG;IAC9B,cAAc,CAAC,EAAE,MAAM,EAAE,CAAC;IAC1B,KAAK,CAAC,EAAE,OAAO,CAAC;CACjB,CAAC;AAGF,eAAO,MAAM,WAAW,GAAI,SAAS,iBAAiB,gGAYlD,CAAC;AAIL,MAAM,MAAM,mBAAmB,GAAG,GAAG,CAAC"}
|
package/dist/safe/adapter.js
DELETED
|
@@ -1,13 +0,0 @@
|
|
|
1
|
-
import { Effect } from "effect";
|
|
2
|
-
import { SafeSdkUnavailableError } from "./errors.js";
|
|
3
|
-
export const loadSafeSdk = (config) => Effect.tryPromise({
|
|
4
|
-
catch: (cause) => new SafeSdkUnavailableError({
|
|
5
|
-
cause,
|
|
6
|
-
message: "Failed to load @safe-global/safe-apps-sdk. Ensure it is installed: bun add @safe-global/safe-apps-sdk",
|
|
7
|
-
}),
|
|
8
|
-
try: async () => {
|
|
9
|
-
const { default: SafeAppsSDK } = await import("@safe-global/safe-apps-sdk");
|
|
10
|
-
return new SafeAppsSDK(config);
|
|
11
|
-
},
|
|
12
|
-
});
|
|
13
|
-
//# sourceMappingURL=adapter.js.map
|
package/dist/safe/adapter.js.map
DELETED
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"adapter.js","sourceRoot":"","sources":["../../src/safe/adapter.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,MAAM,QAAQ,CAAC;AAChC,OAAO,EAAE,uBAAuB,EAAE,MAAM,aAAa,CAAC;AAStD,MAAM,CAAC,MAAM,WAAW,GAAG,CAAC,MAA0B,EAAE,EAAE,CACxD,MAAM,CAAC,UAAU,CAAC;IAChB,KAAK,EAAE,CAAC,KAAK,EAAE,EAAE,CACf,IAAI,uBAAuB,CAAC;QAC1B,KAAK;QACL,OAAO,EACL,uGAAuG;KAC1G,CAAC;IACJ,GAAG,EAAE,KAAK,IAAI,EAAE;QACd,MAAM,EAAE,OAAO,EAAE,WAAW,EAAE,GAAG,MAAM,MAAM,CAAC,4BAA4B,CAAC,CAAC;QAC5E,OAAO,IAAI,WAAW,CAAC,MAAM,CAAC,CAAC;IACjC,CAAC;CACF,CAAC,CAAC","sourcesContent":["import { Effect } from \"effect\";\nimport { SafeSdkUnavailableError } from \"./errors.js\";\n\n/** Configuration for SDK initialization */\nexport type SafeAppsSdkConfig = {\n allowedDomains?: RegExp[];\n debug?: boolean;\n};\n\n/** Load SDK dynamically - keeps the dependency optional at runtime */\nexport const loadSafeSdk = (config?: SafeAppsSdkConfig) =>\n Effect.tryPromise({\n catch: (cause) =>\n new SafeSdkUnavailableError({\n cause,\n message:\n \"Failed to load @safe-global/safe-apps-sdk. Ensure it is installed: bun add @safe-global/safe-apps-sdk\",\n }),\n try: async () => {\n const { default: SafeAppsSDK } = await import(\"@safe-global/safe-apps-sdk\");\n return new SafeAppsSDK(config);\n },\n });\n\n/** Type for the loaded SDK instance - uses any due to optional SDK dependency */\n// biome-ignore lint/suspicious/noExplicitAny: SDK is optional dependency\nexport type SafeAppsSDKInstance = any;\n"]}
|
package/dist/safe/detection.d.ts
DELETED
|
@@ -1,26 +0,0 @@
|
|
|
1
|
-
import { Effect, Schema } from "effect";
|
|
2
|
-
import type { Address, Hex } from "viem";
|
|
3
|
-
import type { ClientNotFoundError } from "../core/index.js";
|
|
4
|
-
import { PublicClientService } from "../core/index.js";
|
|
5
|
-
declare const SafeDetectionError_base: Schema.TaggedErrorClass<SafeDetectionError, "SafeDetectionError", {
|
|
6
|
-
readonly _tag: Schema.tag<"SafeDetectionError">;
|
|
7
|
-
} & {
|
|
8
|
-
address: typeof Schema.String;
|
|
9
|
-
cause: Schema.optional<typeof Schema.Unknown>;
|
|
10
|
-
message: typeof Schema.String;
|
|
11
|
-
}>;
|
|
12
|
-
export declare class SafeDetectionError extends SafeDetectionError_base {
|
|
13
|
-
}
|
|
14
|
-
export type SafeDetectionParams = {
|
|
15
|
-
address: Address;
|
|
16
|
-
chainId: number;
|
|
17
|
-
};
|
|
18
|
-
export type SafeDetectionResult = {
|
|
19
|
-
isSafe: boolean;
|
|
20
|
-
proxyHash: Hex | undefined;
|
|
21
|
-
singletonAddress: Address | undefined;
|
|
22
|
-
singletonHash: Hex | undefined;
|
|
23
|
-
};
|
|
24
|
-
export declare function isSafeMultisig(params: SafeDetectionParams): Effect.Effect<SafeDetectionResult, SafeDetectionError | ClientNotFoundError, PublicClientService>;
|
|
25
|
-
export {};
|
|
26
|
-
//# sourceMappingURL=detection.d.ts.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"detection.d.ts","sourceRoot":"","sources":["../../src/safe/detection.ts"],"names":[],"mappings":"AASA,OAAO,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,QAAQ,CAAC;AACxC,OAAO,KAAK,EAAE,OAAO,EAAE,GAAG,EAAE,MAAM,MAAM,CAAC;AAEzC,OAAO,KAAK,EAAE,mBAAmB,EAAE,MAAM,qBAAqB,CAAC;AAC/D,OAAO,EAAE,mBAAmB,EAAE,MAAM,qBAAqB,CAAC;;;;;;;;AAuD1D,qBAAa,kBAAmB,SAAQ,uBAOvC;CAAG;AAoBJ,MAAM,MAAM,mBAAmB,GAAG;IAEhC,OAAO,EAAE,OAAO,CAAC;IAEjB,OAAO,EAAE,MAAM,CAAC;CACjB,CAAC;AAEF,MAAM,MAAM,mBAAmB,GAAG;IAEhC,MAAM,EAAE,OAAO,CAAC;IAEhB,SAAS,EAAE,GAAG,GAAG,SAAS,CAAC;IAE3B,gBAAgB,EAAE,OAAO,GAAG,SAAS,CAAC;IAEtC,aAAa,EAAE,GAAG,GAAG,SAAS,CAAC;CAChC,CAAC;AA8BF,wBAAgB,cAAc,CAC5B,MAAM,EAAE,mBAAmB,GAC1B,MAAM,CAAC,MAAM,CACd,mBAAmB,EACnB,kBAAkB,GAAG,mBAAmB,EACxC,mBAAmB,CACpB,CAuFA"}
|
package/dist/safe/detection.js
DELETED
|
@@ -1,102 +0,0 @@
|
|
|
1
|
-
import { Effect, Schema } from "effect";
|
|
2
|
-
import { keccak256 } from "viem";
|
|
3
|
-
import { PublicClientService } from "../core/index.js";
|
|
4
|
-
const KNOWN_SAFE_PROXY_HASHES = new Set([
|
|
5
|
-
"0xb89c1b3bdf2cf8827818646bce9a8f6e372885f8c55e5c07acbd307cb133b000",
|
|
6
|
-
"0xd7d408ebcd99b2b70be43e20253d6d92a8ea8fab29bd3be7f55b10032331fb4c",
|
|
7
|
-
]);
|
|
8
|
-
const KNOWN_SAFE_SINGLETON_HASHES = new Set([
|
|
9
|
-
"0xe1f1593df76e69abc2d692792c80f329457551d5e83dde597546a1d58764da80",
|
|
10
|
-
"0x56b8be58b5ad629a621593a2e5e5e8e9a28408dc06e95597497b303902772e45",
|
|
11
|
-
"0x2ae2d1231f0d754a7fa4f5e5d0e5554085e1b500d8e09f95aaaaa3f49c0db922",
|
|
12
|
-
"0xbba688fbdb21ad2bb58bc320638b43d94e7d100f6f3ebaab0a4e4de6304b1c2e",
|
|
13
|
-
"0x21842597390c4c6e3c1239e434a682b054bd9548eee5e9b1d6a4482731023c0f",
|
|
14
|
-
"0x1fe2df852ba3299d6534ef416eefa406e56ced995bca886ab7a553e6d0c5e1c4",
|
|
15
|
-
"0xb1f926978a0f44a2c0ec8fe822418ae969bd8c3f18d61e5103100339894f81ff",
|
|
16
|
-
"0xdda019cbd7c867a533a2a86e5c53434fdc50b13122b5a5ddb4a8df61b31c20f2",
|
|
17
|
-
"0x180193227186ccb85316c94db1f0d156ed932b14712cfaac78901899178572dc",
|
|
18
|
-
]);
|
|
19
|
-
export class SafeDetectionError extends Schema.TaggedError()("SafeDetectionError", {
|
|
20
|
-
address: Schema.String,
|
|
21
|
-
cause: Schema.optional(Schema.Unknown),
|
|
22
|
-
message: Schema.String,
|
|
23
|
-
}) {
|
|
24
|
-
}
|
|
25
|
-
const masterCopyAbi = [
|
|
26
|
-
{
|
|
27
|
-
inputs: [],
|
|
28
|
-
name: "masterCopy",
|
|
29
|
-
outputs: [{ name: "", type: "address" }],
|
|
30
|
-
stateMutability: "view",
|
|
31
|
-
type: "function",
|
|
32
|
-
},
|
|
33
|
-
];
|
|
34
|
-
export function isSafeMultisig(params) {
|
|
35
|
-
return Effect.gen(function* () {
|
|
36
|
-
const { address, chainId } = params;
|
|
37
|
-
const publicClientService = yield* PublicClientService;
|
|
38
|
-
const client = yield* publicClientService.get(chainId);
|
|
39
|
-
const proxyBytecode = yield* Effect.tryPromise({
|
|
40
|
-
catch: (cause) => new SafeDetectionError({
|
|
41
|
-
address,
|
|
42
|
-
cause,
|
|
43
|
-
message: `Failed to get bytecode for ${address}`,
|
|
44
|
-
}),
|
|
45
|
-
try: () => client.getCode({ address }),
|
|
46
|
-
});
|
|
47
|
-
if (!proxyBytecode || proxyBytecode === "0x") {
|
|
48
|
-
return {
|
|
49
|
-
isSafe: false,
|
|
50
|
-
proxyHash: undefined,
|
|
51
|
-
singletonAddress: undefined,
|
|
52
|
-
singletonHash: undefined,
|
|
53
|
-
};
|
|
54
|
-
}
|
|
55
|
-
const proxyHash = keccak256(proxyBytecode);
|
|
56
|
-
if (!KNOWN_SAFE_PROXY_HASHES.has(proxyHash)) {
|
|
57
|
-
return {
|
|
58
|
-
isSafe: false,
|
|
59
|
-
proxyHash,
|
|
60
|
-
singletonAddress: undefined,
|
|
61
|
-
singletonHash: undefined,
|
|
62
|
-
};
|
|
63
|
-
}
|
|
64
|
-
const singletonAddress = yield* Effect.tryPromise({
|
|
65
|
-
catch: (cause) => new SafeDetectionError({
|
|
66
|
-
address,
|
|
67
|
-
cause,
|
|
68
|
-
message: `Failed to call masterCopy() on ${address}`,
|
|
69
|
-
}),
|
|
70
|
-
try: () => client.readContract({
|
|
71
|
-
abi: masterCopyAbi,
|
|
72
|
-
address,
|
|
73
|
-
functionName: "masterCopy",
|
|
74
|
-
}),
|
|
75
|
-
});
|
|
76
|
-
const singletonBytecode = yield* Effect.tryPromise({
|
|
77
|
-
catch: (cause) => new SafeDetectionError({
|
|
78
|
-
address,
|
|
79
|
-
cause,
|
|
80
|
-
message: `Failed to get bytecode for singleton ${singletonAddress}`,
|
|
81
|
-
}),
|
|
82
|
-
try: () => client.getCode({ address: singletonAddress }),
|
|
83
|
-
});
|
|
84
|
-
if (!singletonBytecode || singletonBytecode === "0x") {
|
|
85
|
-
return {
|
|
86
|
-
isSafe: false,
|
|
87
|
-
proxyHash,
|
|
88
|
-
singletonAddress,
|
|
89
|
-
singletonHash: undefined,
|
|
90
|
-
};
|
|
91
|
-
}
|
|
92
|
-
const singletonHash = keccak256(singletonBytecode);
|
|
93
|
-
const isSafe = KNOWN_SAFE_SINGLETON_HASHES.has(singletonHash);
|
|
94
|
-
return {
|
|
95
|
-
isSafe,
|
|
96
|
-
proxyHash,
|
|
97
|
-
singletonAddress,
|
|
98
|
-
singletonHash,
|
|
99
|
-
};
|
|
100
|
-
});
|
|
101
|
-
}
|
|
102
|
-
//# sourceMappingURL=detection.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"detection.js","sourceRoot":"","sources":["../../src/safe/detection.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Safe multisig detection using EXTCODEHASH comparisons.\n *\n * Determines if an address is a Safe multisig by comparing bytecode hashes\n * against known Safe proxy and singleton hashes.\n *\n * @see https://github.com/safe-fndn/safe-smart-account/issues/714\n */\n\nimport { Effect, Schema } from \"effect\";\nimport type { Address, Hex } from \"viem\";\nimport { keccak256 } from \"viem\";\nimport type { ClientNotFoundError } from \"@/src/core/index.js\";\nimport { PublicClientService } from \"@/src/core/index.js\";\n\n// -----------------------------------------------------------------------------\n// Known Safe bytecode hashes\n// -----------------------------------------------------------------------------\n\n/**\n * Known Safe proxy bytecode hashes (keccak256 of deployed bytecode).\n *\n * These are the proxy contracts that delegate to Safe singletons.\n * All Safe accounts use one of these proxy implementations.\n *\n * @see https://github.com/safe-global/safe-deployments\n */\nconst KNOWN_SAFE_PROXY_HASHES: ReadonlySet<Hex> = new Set([\n // SafeProxy v1.3.0\n \"0xb89c1b3bdf2cf8827818646bce9a8f6e372885f8c55e5c07acbd307cb133b000\",\n // SafeProxy v1.4.1\n \"0xd7d408ebcd99b2b70be43e20253d6d92a8ea8fab29bd3be7f55b10032331fb4c\",\n] as Hex[]);\n\n/**\n * Known Safe singleton (implementation) bytecode hashes.\n *\n * These are the actual Safe contract implementations that proxies delegate to.\n *\n * Derivation: `cast keccak $(cast code <address> --rpc-url <mainnet>)`\n *\n * @see https://github.com/safe-global/safe-deployments\n */\nconst KNOWN_SAFE_SINGLETON_HASHES: ReadonlySet<Hex> = new Set([\n // Safe v1.0.0\n \"0xe1f1593df76e69abc2d692792c80f329457551d5e83dde597546a1d58764da80\",\n // Safe v1.1.1\n \"0x56b8be58b5ad629a621593a2e5e5e8e9a28408dc06e95597497b303902772e45\",\n // Safe v1.2.0\n \"0x2ae2d1231f0d754a7fa4f5e5d0e5554085e1b500d8e09f95aaaaa3f49c0db922\",\n // Safe v1.3.0\n \"0xbba688fbdb21ad2bb58bc320638b43d94e7d100f6f3ebaab0a4e4de6304b1c2e\",\n // Safe v1.3.0 L2\n \"0x21842597390c4c6e3c1239e434a682b054bd9548eee5e9b1d6a4482731023c0f\",\n // Safe v1.4.1\n \"0x1fe2df852ba3299d6534ef416eefa406e56ced995bca886ab7a553e6d0c5e1c4\",\n // Safe v1.4.1 L2\n \"0xb1f926978a0f44a2c0ec8fe822418ae969bd8c3f18d61e5103100339894f81ff\",\n // Safe v1.5.0\n \"0xdda019cbd7c867a533a2a86e5c53434fdc50b13122b5a5ddb4a8df61b31c20f2\",\n // Safe v1.5.0 L2\n \"0x180193227186ccb85316c94db1f0d156ed932b14712cfaac78901899178572dc\",\n] as Hex[]);\n\n// -----------------------------------------------------------------------------\n// Errors\n// -----------------------------------------------------------------------------\n\nexport class SafeDetectionError extends Schema.TaggedError<SafeDetectionError>()(\n \"SafeDetectionError\",\n {\n address: Schema.String,\n cause: Schema.optional(Schema.Unknown),\n message: Schema.String,\n }\n) {}\n\n// -----------------------------------------------------------------------------\n// ABI for masterCopy() call\n// -----------------------------------------------------------------------------\n\nconst masterCopyAbi = [\n {\n inputs: [],\n name: \"masterCopy\",\n outputs: [{ name: \"\", type: \"address\" }],\n stateMutability: \"view\",\n type: \"function\",\n },\n] as const;\n\n// -----------------------------------------------------------------------------\n// Detection function\n// -----------------------------------------------------------------------------\n\nexport type SafeDetectionParams = {\n /** The address to check */\n address: Address;\n /** The chain ID to query */\n chainId: number;\n};\n\nexport type SafeDetectionResult = {\n /** Whether the address appears to be a Safe multisig */\n isSafe: boolean;\n /** The proxy bytecode hash (if contract exists) */\n proxyHash: Hex | undefined;\n /** The singleton address (if Safe proxy detected) */\n singletonAddress: Address | undefined;\n /** The singleton bytecode hash (if singleton found) */\n singletonHash: Hex | undefined;\n};\n\n/**\n * Check if an address is a Safe multisig by comparing EXTCODEHASH values.\n *\n * This function:\n * 1. Gets the bytecode of the target address and computes its hash\n * 2. Checks if the hash matches a known Safe proxy\n * 3. If proxy matches, calls `masterCopy()` to get the singleton address\n * 4. Verifies the singleton's bytecode hash matches a known Safe singleton\n *\n * @example\n * ```typescript\n * const result = yield* isSafeMultisig({\n * address: \"0x...\",\n * chainId: 1,\n * });\n *\n * if (result.isSafe) {\n * console.log(\"This is a Safe multisig!\");\n * }\n * ```\n *\n * @remarks\n * This is a heuristic check, not a guarantee. Any contract can technically\n * deploy bytecode that matches Safe's, though this is extremely unlikely\n * in practice. For most use cases, this provides sufficient confidence.\n *\n * @see https://github.com/safe-fndn/safe-smart-account/issues/714\n */\nexport function isSafeMultisig(\n params: SafeDetectionParams\n): Effect.Effect<\n SafeDetectionResult,\n SafeDetectionError | ClientNotFoundError,\n PublicClientService\n> {\n return Effect.gen(function* () {\n const { address, chainId } = params;\n const publicClientService = yield* PublicClientService;\n const client = yield* publicClientService.get(chainId);\n\n // 1. Get proxy bytecode\n const proxyBytecode = yield* Effect.tryPromise({\n catch: (cause) =>\n new SafeDetectionError({\n address,\n cause,\n message: `Failed to get bytecode for ${address}`,\n }),\n try: () => client.getCode({ address }),\n });\n\n // No bytecode = EOA or empty contract\n if (!proxyBytecode || proxyBytecode === \"0x\") {\n return {\n isSafe: false,\n proxyHash: undefined,\n singletonAddress: undefined,\n singletonHash: undefined,\n };\n }\n\n // 2. Compute proxy hash\n const proxyHash = keccak256(proxyBytecode);\n\n // 3. Check if proxy hash matches known Safe proxies\n if (!KNOWN_SAFE_PROXY_HASHES.has(proxyHash)) {\n return {\n isSafe: false,\n proxyHash,\n singletonAddress: undefined,\n singletonHash: undefined,\n };\n }\n\n // 4. Call masterCopy() to get singleton address\n const singletonAddress = yield* Effect.tryPromise({\n catch: (cause) =>\n new SafeDetectionError({\n address,\n cause,\n message: `Failed to call masterCopy() on ${address}`,\n }),\n try: () =>\n client.readContract({\n abi: masterCopyAbi,\n address,\n functionName: \"masterCopy\",\n }),\n });\n\n // 5. Get singleton bytecode\n const singletonBytecode = yield* Effect.tryPromise({\n catch: (cause) =>\n new SafeDetectionError({\n address,\n cause,\n message: `Failed to get bytecode for singleton ${singletonAddress}`,\n }),\n try: () => client.getCode({ address: singletonAddress }),\n });\n\n if (!singletonBytecode || singletonBytecode === \"0x\") {\n return {\n isSafe: false,\n proxyHash,\n singletonAddress,\n singletonHash: undefined,\n };\n }\n\n // 6. Compute and check singleton hash\n const singletonHash = keccak256(singletonBytecode);\n const isSafe = KNOWN_SAFE_SINGLETON_HASHES.has(singletonHash);\n\n return {\n isSafe,\n proxyHash,\n singletonAddress,\n singletonHash,\n };\n });\n}\n"]}
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"detection.test.integration.d.ts","sourceRoot":"","sources":["../../src/safe/detection.test.integration.ts"],"names":[],"mappings":""}
|
|
@@ -1,92 +0,0 @@
|
|
|
1
|
-
import { describe, expect, it } from "@effect/vitest";
|
|
2
|
-
import { Effect } from "effect";
|
|
3
|
-
import { mainnet } from "viem/chains";
|
|
4
|
-
import { makePublicClientLayer } from "../presets/index.js";
|
|
5
|
-
import { isSafeMultisig } from "./detection.js";
|
|
6
|
-
const SAFE_V1_3_0_SINGLETON = "0xd9Db270c1B5E3Bd161E8c8503c55cEABeE709552";
|
|
7
|
-
const SAFE_V1_4_1_SINGLETON = "0x41675C099F32341bf84BFc5382aF534df5C7461a";
|
|
8
|
-
const testLayer = makePublicClientLayer([
|
|
9
|
-
{
|
|
10
|
-
chain: mainnet,
|
|
11
|
-
chainId: 1,
|
|
12
|
-
rpcUrls: ["https://eth.llamarpc.com"],
|
|
13
|
-
},
|
|
14
|
-
]);
|
|
15
|
-
describe("isSafeMultisig", () => {
|
|
16
|
-
describe("Positive Cases - Known Safe Wallets", () => {
|
|
17
|
-
it.effect("detects CoW DAO Safe (v1.3.0)", () => Effect.gen(function* () {
|
|
18
|
-
const result = yield* isSafeMultisig({
|
|
19
|
-
address: "0xcA771eda0c70aA7d053aB1B25004559B918FE662",
|
|
20
|
-
chainId: 1,
|
|
21
|
-
});
|
|
22
|
-
expect(result.isSafe).toBe(true);
|
|
23
|
-
expect(result.proxyHash).toBeDefined();
|
|
24
|
-
expect(result.singletonAddress).toBe(SAFE_V1_3_0_SINGLETON);
|
|
25
|
-
expect(result.singletonHash).toBeDefined();
|
|
26
|
-
}).pipe(Effect.provide(testLayer)));
|
|
27
|
-
it.effect("detects Gnosis DAO Safe (v1.3.0)", () => Effect.gen(function* () {
|
|
28
|
-
const result = yield* isSafeMultisig({
|
|
29
|
-
address: "0x849D52316331967b6fF1198e5E32A0eB168D039d",
|
|
30
|
-
chainId: 1,
|
|
31
|
-
});
|
|
32
|
-
expect(result.isSafe).toBe(true);
|
|
33
|
-
expect(result.proxyHash).toBeDefined();
|
|
34
|
-
expect(result.singletonAddress).toBe(SAFE_V1_3_0_SINGLETON);
|
|
35
|
-
expect(result.singletonHash).toBeDefined();
|
|
36
|
-
}).pipe(Effect.provide(testLayer)));
|
|
37
|
-
it.effect("detects v1.4.1 Safe", () => Effect.gen(function* () {
|
|
38
|
-
const result = yield* isSafeMultisig({
|
|
39
|
-
address: "0x843ed9137c60772b30a71a7fbdb7f302f336ace7",
|
|
40
|
-
chainId: 1,
|
|
41
|
-
});
|
|
42
|
-
expect(result.isSafe).toBe(true);
|
|
43
|
-
expect(result.proxyHash).toBeDefined();
|
|
44
|
-
expect(result.singletonAddress).toBe(SAFE_V1_4_1_SINGLETON);
|
|
45
|
-
expect(result.singletonHash).toBeDefined();
|
|
46
|
-
}).pipe(Effect.provide(testLayer)));
|
|
47
|
-
});
|
|
48
|
-
describe("Negative Cases - Non-Safe Contracts", () => {
|
|
49
|
-
it.effect("identifies non-Safe contract (vitalik.eth with EIP-7702) as non-Safe", () => Effect.gen(function* () {
|
|
50
|
-
const result = yield* isSafeMultisig({
|
|
51
|
-
address: "0xd8dA6BF26964aF9D7eEd9e03E53415D37aA96045",
|
|
52
|
-
chainId: 1,
|
|
53
|
-
});
|
|
54
|
-
expect(result.isSafe).toBe(false);
|
|
55
|
-
expect(result.singletonAddress).toBeUndefined();
|
|
56
|
-
expect(result.singletonHash).toBeUndefined();
|
|
57
|
-
}).pipe(Effect.provide(testLayer)));
|
|
58
|
-
it.effect("identifies Uniswap V2 Router as non-Safe", () => Effect.gen(function* () {
|
|
59
|
-
const result = yield* isSafeMultisig({
|
|
60
|
-
address: "0x7a250d5630B4cF539739dF2C5dAcb4c659F2488D",
|
|
61
|
-
chainId: 1,
|
|
62
|
-
});
|
|
63
|
-
expect(result.isSafe).toBe(false);
|
|
64
|
-
expect(result.proxyHash).toBeDefined();
|
|
65
|
-
expect(result.singletonAddress).toBeUndefined();
|
|
66
|
-
expect(result.singletonHash).toBeUndefined();
|
|
67
|
-
}).pipe(Effect.provide(testLayer)));
|
|
68
|
-
it.effect("identifies USDC as non-Safe", () => Effect.gen(function* () {
|
|
69
|
-
const result = yield* isSafeMultisig({
|
|
70
|
-
address: "0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48",
|
|
71
|
-
chainId: 1,
|
|
72
|
-
});
|
|
73
|
-
expect(result.isSafe).toBe(false);
|
|
74
|
-
expect(result.proxyHash).toBeDefined();
|
|
75
|
-
expect(result.singletonAddress).toBeUndefined();
|
|
76
|
-
expect(result.singletonHash).toBeUndefined();
|
|
77
|
-
}).pipe(Effect.provide(testLayer)));
|
|
78
|
-
});
|
|
79
|
-
describe("Edge Cases", () => {
|
|
80
|
-
it.effect("handles precompile address gracefully", () => Effect.gen(function* () {
|
|
81
|
-
const result = yield* isSafeMultisig({
|
|
82
|
-
address: "0x0000000000000000000000000000000000000001",
|
|
83
|
-
chainId: 1,
|
|
84
|
-
});
|
|
85
|
-
expect(result.isSafe).toBe(false);
|
|
86
|
-
expect(result.proxyHash).toBeUndefined();
|
|
87
|
-
expect(result.singletonAddress).toBeUndefined();
|
|
88
|
-
expect(result.singletonHash).toBeUndefined();
|
|
89
|
-
}).pipe(Effect.provide(testLayer)));
|
|
90
|
-
});
|
|
91
|
-
});
|
|
92
|
-
//# sourceMappingURL=detection.test.integration.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"detection.test.integration.js","sourceRoot":"","sources":["../../src/safe/detection.test.integration.ts"],"names":[],"mappings":"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","sourcesContent":["import { describe, expect, it } from \"@effect/vitest\";\nimport { Effect } from \"effect\";\nimport type { Address } from \"viem\";\nimport { mainnet } from \"viem/chains\";\nimport { makePublicClientLayer } from \"@/src/presets/index.js\";\nimport { isSafeMultisig } from \"./detection.js\";\n\n// Known Safe singleton hashes for verification\nconst SAFE_V1_3_0_SINGLETON = \"0xd9Db270c1B5E3Bd161E8c8503c55cEABeE709552\" as Address;\nconst SAFE_V1_4_1_SINGLETON = \"0x41675C099F32341bf84BFc5382aF534df5C7461a\" as Address;\n\n// Test layer with Ethereum mainnet RPC\nconst testLayer = makePublicClientLayer([\n {\n chain: mainnet,\n chainId: 1,\n rpcUrls: [\"https://eth.llamarpc.com\"],\n },\n]);\n\ndescribe(\"isSafeMultisig\", () => {\n describe(\"Positive Cases - Known Safe Wallets\", () => {\n it.effect(\"detects CoW DAO Safe (v1.3.0)\", () =>\n Effect.gen(function* () {\n const result = yield* isSafeMultisig({\n address: \"0xcA771eda0c70aA7d053aB1B25004559B918FE662\",\n chainId: 1,\n });\n\n expect(result.isSafe).toBe(true);\n expect(result.proxyHash).toBeDefined();\n expect(result.singletonAddress).toBe(SAFE_V1_3_0_SINGLETON);\n expect(result.singletonHash).toBeDefined();\n }).pipe(Effect.provide(testLayer))\n );\n\n it.effect(\"detects Gnosis DAO Safe (v1.3.0)\", () =>\n Effect.gen(function* () {\n const result = yield* isSafeMultisig({\n address: \"0x849D52316331967b6fF1198e5E32A0eB168D039d\",\n chainId: 1,\n });\n\n expect(result.isSafe).toBe(true);\n expect(result.proxyHash).toBeDefined();\n expect(result.singletonAddress).toBe(SAFE_V1_3_0_SINGLETON);\n expect(result.singletonHash).toBeDefined();\n }).pipe(Effect.provide(testLayer))\n );\n\n it.effect(\"detects v1.4.1 Safe\", () =>\n Effect.gen(function* () {\n const result = yield* isSafeMultisig({\n address: \"0x843ed9137c60772b30a71a7fbdb7f302f336ace7\",\n chainId: 1,\n });\n\n expect(result.isSafe).toBe(true);\n expect(result.proxyHash).toBeDefined();\n expect(result.singletonAddress).toBe(SAFE_V1_4_1_SINGLETON);\n expect(result.singletonHash).toBeDefined();\n }).pipe(Effect.provide(testLayer))\n );\n });\n\n describe(\"Negative Cases - Non-Safe Contracts\", () => {\n it.effect(\"identifies non-Safe contract (vitalik.eth with EIP-7702) as non-Safe\", () =>\n Effect.gen(function* () {\n const result = yield* isSafeMultisig({\n address: \"0xd8dA6BF26964aF9D7eEd9e03E53415D37aA96045\",\n chainId: 1,\n });\n\n // vitalik.eth address has bytecode (EIP-7702 delegation) but is not a Safe multisig\n expect(result.isSafe).toBe(false);\n expect(result.singletonAddress).toBeUndefined();\n expect(result.singletonHash).toBeUndefined();\n }).pipe(Effect.provide(testLayer))\n );\n\n it.effect(\"identifies Uniswap V2 Router as non-Safe\", () =>\n Effect.gen(function* () {\n const result = yield* isSafeMultisig({\n address: \"0x7a250d5630B4cF539739dF2C5dAcb4c659F2488D\",\n chainId: 1,\n });\n\n expect(result.isSafe).toBe(false);\n expect(result.proxyHash).toBeDefined();\n expect(result.singletonAddress).toBeUndefined();\n expect(result.singletonHash).toBeUndefined();\n }).pipe(Effect.provide(testLayer))\n );\n\n it.effect(\"identifies USDC as non-Safe\", () =>\n Effect.gen(function* () {\n const result = yield* isSafeMultisig({\n address: \"0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48\",\n chainId: 1,\n });\n\n expect(result.isSafe).toBe(false);\n expect(result.proxyHash).toBeDefined();\n expect(result.singletonAddress).toBeUndefined();\n expect(result.singletonHash).toBeUndefined();\n }).pipe(Effect.provide(testLayer))\n );\n });\n\n describe(\"Edge Cases\", () => {\n it.effect(\"handles precompile address gracefully\", () =>\n Effect.gen(function* () {\n const result = yield* isSafeMultisig({\n address: \"0x0000000000000000000000000000000000000001\",\n chainId: 1,\n });\n\n expect(result.isSafe).toBe(false);\n expect(result.proxyHash).toBeUndefined();\n expect(result.singletonAddress).toBeUndefined();\n expect(result.singletonHash).toBeUndefined();\n }).pipe(Effect.provide(testLayer))\n );\n });\n});\n"]}
|