Skip to content
New issue

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

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

Already on GitHub? Sign in to your account

Throw when accessing non-draftables in strict mode #715

Open
wants to merge 5 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
132 changes: 132 additions & 0 deletions __tests__/strict-mode.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,132 @@
"use strict"
import produce, {
setStrictMode,
unsafe,
immerable,
enableMapSet
} from "../src/immer"

enableMapSet()

describe("Strict Mode", () => {
class Foo {}

describe("by default", () => {
it("should not throw an error when accessing a non-draftable class instance", () => {
expect.hasAssertions()

produce({instance: new Foo()}, draft => {
expect(() => {
draft.instance.value = 5
}).not.toThrow()
})
})
})

afterAll(() => {
setStrictMode(false)
})

describe("when disabled", () => {
beforeEach(() => {
setStrictMode(false)
})

it("should allow accessing a non-draftable class instance", () => {
expect.hasAssertions()

produce({instance: new Foo()}, draft => {
expect(() => {
draft.instance.value = 5
}).not.toThrow()
})
})

it("should not throw errors when using the `unsafe` function", () => {
expect.hasAssertions()

produce({instance: new Foo()}, draft => {
unsafe(() => {
expect(() => {
draft.instance.value = 5
}).not.toThrow()
})
})
})
})

describe("when enabled", () => {
beforeEach(() => {
setStrictMode(true)
})

it("should throw an error when accessing a non-draftable class instance", () => {
expect.hasAssertions()

produce({instance: new Foo()}, draft => {
expect(() => draft.instance).toThrow()
})
})

it("should allow accessing a non-draftable using the `unsafe` function", () => {
expect.hasAssertions()

produce({instance: new Foo()}, draft => {
unsafe(() => {
expect(() => {
draft.instance.value = 5
}).not.toThrow()
})
})
})

it("should require using unsafe for non-draftables in a different scope", () => {
expect.assertions(2)

produce({instance: new Foo()}, () => {
unsafe(() => {
produce({nested: new Foo()}, nestedDraft => {
expect(() => nestedDraft.nested).toThrow()

unsafe(() => {
expect(() => nestedDraft.nested).not.toThrow()
})
})
})
})
})

describe("with an immerable class", () => {
beforeAll(() => {
Foo[immerable] = true
})

afterAll(() => {
Foo[immerable] = true
})

it("should allow accessing the class instance", () => {
expect.hasAssertions()

produce({instance: new Foo()}, draft => {
expect(() => {
draft.instance.value = 5
}).not.toThrow()
})
})
})

it("should allow accessing draftable properties", () => {
expect(() =>
produce({arr: [], obj: {}, map: new Map(), set: new Set()}, draft => {
draft.arr.push(1)
draft.arr[0] = 1
draft.obj.foo = 5
draft.obj.hasOwnProperty("abc")
draft.map.set("foo", 5)
draft.set.add("foo")
})
).not.toThrow()
})
})
})
27 changes: 26 additions & 1 deletion src/core/immerClass.ts
Original file line number Diff line number Diff line change
Expand Up @@ -37,11 +37,19 @@ export class Immer implements ProducersFns {

autoFreeze_: boolean = true

constructor(config?: {useProxies?: boolean; autoFreeze?: boolean}) {
strictModeEnabled_: boolean = false

constructor(config?: {
useProxies?: boolean
autoFreeze?: boolean
strictMode?: boolean
}) {
if (typeof config?.useProxies === "boolean")
this.setUseProxies(config!.useProxies)
if (typeof config?.autoFreeze === "boolean")
this.setAutoFreeze(config!.autoFreeze)
if (typeof config?.strictMode === "boolean")
this.setStrictMode(config!.strictMode)
this.produce = this.produce.bind(this)
this.produceWithPatches = this.produceWithPatches.bind(this)
}
Expand Down Expand Up @@ -183,6 +191,23 @@ export class Immer implements ProducersFns {
this.useProxies_ = value
}

/**
* Pass true to throw errors when attempting to access a non-draftable reference.
*
* By default, strict mode is disabled.
*/
setStrictMode(value: boolean) {
this.strictModeEnabled_ = value
}

unsafe(callback: () => void) {
const scope = getCurrentScope()

scope.unsafeNonDraftabledAllowed_ = true
callback()
scope.unsafeNonDraftabledAllowed_ = false
}

applyPatches(base: Objectish, patches: Patch[]) {
// If a patch replaces the entire state, take that replacement as base
// before applying patches
Expand Down
14 changes: 13 additions & 1 deletion src/core/proxy.ts
Original file line number Diff line number Diff line change
Expand Up @@ -108,7 +108,19 @@ export const objectTraps: ProxyHandler<ProxyState> = {
return readPropFromProto(state, source, prop)
}
const value = source[prop]
if (state.finalized_ || !isDraftable(value)) {
if (state.finalized_) {
return value
}
if (!isDraftable(value)) {
if (
state.scope_.immer_.strictModeEnabled_ &&
!state.scope_.unsafeNonDraftabledAllowed_ &&
typeof value === "object" &&
value !== null
) {
die(24)
}

return value
}
// Check for existing draft in modified state.
Expand Down
4 changes: 3 additions & 1 deletion src/core/scope.ts
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,7 @@ export interface ImmerScope {
patchListener_?: PatchListener
immer_: Immer
unfinalizedDrafts_: number
unsafeNonDraftabledAllowed_: boolean
}

let currentScope: ImmerScope | undefined
Expand All @@ -42,7 +43,8 @@ function createScope(
// Whenever the modified draft contains a draft from another scope, we
// need to prevent auto-freezing so the unowned draft can be finalized.
canAutoFreeze_: true,
unfinalizedDrafts_: 0
unfinalizedDrafts_: 0,
unsafeNonDraftabledAllowed_: false
}
}

Expand Down
12 changes: 12 additions & 0 deletions src/immer.ts
Original file line number Diff line number Diff line change
Expand Up @@ -67,6 +67,18 @@ export const setAutoFreeze = immer.setAutoFreeze.bind(immer)
*/
export const setUseProxies = immer.setUseProxies.bind(immer)

/**
* Pass true to throw errors when attempting to access a non-draftable reference.
*
* By default, strict mode is disabled.
*/
export const setStrictMode = immer.setStrictMode.bind(immer)

/**
* Allow accessing non-draftable references in strict mode inside the callback.
*/
export const unsafe = immer.unsafe.bind(immer)

/**
* Apply an array of Immer patches to the first argument.
*
Expand Down
2 changes: 2 additions & 0 deletions src/plugins/es5.ts
Original file line number Diff line number Diff line change
Expand Up @@ -31,7 +31,9 @@ export function enableES5() {
) {
if (!isReplaced) {
if (scope.patches_) {
scope.unsafeNonDraftabledAllowed_ = true
markChangesRecursively(scope.drafts_![0])
scope.unsafeNonDraftabledAllowed_ = false
}
// This is faster when we don't care about which attributes changed.
markChangesSweep(scope.drafts_)
Expand Down
12 changes: 12 additions & 0 deletions src/types/index.js.flow
Original file line number Diff line number Diff line change
Expand Up @@ -84,6 +84,18 @@ declare export function setAutoFreeze(autoFreeze: boolean): void
*/
declare export function setUseProxies(useProxies: boolean): void

/**
* Pass true to throw errors when attempting to access a non-draftable reference.
*
* By default, strict mode is disabled.
*/
declare export function setStrictMode(strictMode: boolean): void

/**
* Allow accessing non-draftable references in strict mode inside the callback.
*/
declare export function unsafe(callback: () => void): void

declare export function applyPatches<S>(state: S, patches: Patch[]): S

declare export function original<S>(value: S): S
Expand Down
3 changes: 2 additions & 1 deletion src/utils/errors.ts
Original file line number Diff line number Diff line change
Expand Up @@ -38,7 +38,8 @@ const errors = {
},
23(thing: string) {
return `'original' expects a draft, got: ${thing}`
}
},
24: "Cannot get a non-draftable reference in strict mode. Use the `unsafe` function, add the `immerable` symbol, or disable strict mode"
} as const

export function die(error: keyof typeof errors, ...args: any[]): never {
Expand Down