Skip to content

Commit 46d2c01

Browse files
tim-smarteffect-bot
authored andcommitted
add RcMap.touch, for reseting the idle timeout for an item (#4281)
1 parent 71285c7 commit 46d2c01

File tree

4 files changed

+80
-2
lines changed

4 files changed

+80
-2
lines changed

.changeset/great-buses-perform.md

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
---
2+
"effect": minor
3+
---
4+
5+
add RcMap.touch, for reseting the idle timeout for an item

packages/effect/src/RcMap.ts

+9
Original file line numberDiff line numberDiff line change
@@ -118,3 +118,12 @@ export const invalidate: {
118118
<K>(key: K): <A, E>(self: RcMap<K, A, E>) => Effect.Effect<void>
119119
<K, A, E>(self: RcMap<K, A, E>, key: K): Effect.Effect<void>
120120
} = internal.invalidate
121+
122+
/**
123+
* @since 3.13.0
124+
* @category combinators
125+
*/
126+
export const touch: {
127+
<K>(key: K): <A, E>(self: RcMap<K, A, E>) => Effect.Effect<void>
128+
<K, A, E>(self: RcMap<K, A, E>, key: K): Effect.Effect<void>
129+
} = internal.touch

packages/effect/src/internal/rcMap.ts

+35-2
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
import type * as Cause from "../Cause.js"
2+
import type { Clock } from "../Clock.js"
23
import * as Context from "../Context.js"
34
import type * as Deferred from "../Deferred.js"
45
import * as Duration from "../Duration.js"
@@ -34,6 +35,7 @@ declare namespace State {
3435
readonly scope: Scope.CloseableScope
3536
readonly finalizer: Effect<void>
3637
fiber: RuntimeFiber<void, never> | undefined
38+
expiresAt: number
3739
refCount: number
3840
}
3941
}
@@ -168,6 +170,7 @@ const acquire = core.fnUntraced(function*<K, A, E>(self: RcMapImpl<K, A, E>, key
168170
scope,
169171
finalizer: undefined as any,
170172
fiber: undefined,
173+
expiresAt: 0,
171174
refCount: 1
172175
}
173176
;(entry as any).finalizer = release(self, key, entry)
@@ -178,7 +181,7 @@ const acquire = core.fnUntraced(function*<K, A, E>(self: RcMapImpl<K, A, E>, key
178181
})
179182

180183
const release = <K, A, E>(self: RcMapImpl<K, A, E>, key: K, entry: State.Entry<A, E>) =>
181-
core.suspend(() => {
184+
coreEffect.clockWith((clock) => {
182185
entry.refCount--
183186
if (entry.refCount > 0) {
184187
return core.void
@@ -193,7 +196,10 @@ const release = <K, A, E>(self: RcMapImpl<K, A, E>, key: K, entry: State.Entry<A
193196
return core.scopeClose(entry.scope, core.exitVoid)
194197
}
195198

196-
return coreEffect.sleep(self.idleTimeToLive).pipe(
199+
entry.expiresAt = clock.unsafeCurrentTimeMillis() + Duration.toMillis(self.idleTimeToLive)
200+
201+
return clock.sleep(self.idleTimeToLive).pipe(
202+
core.zipRight(waitUntilExpired(entry, clock)),
197203
core.interruptible,
198204
core.zipRight(core.suspend(() => {
199205
if (self.state._tag === "Open" && entry.refCount === 0) {
@@ -213,6 +219,16 @@ const release = <K, A, E>(self: RcMapImpl<K, A, E>, key: K, entry: State.Entry<A
213219
)
214220
})
215221

222+
const waitUntilExpired = <A, E>(entry: State.Entry<A, E>, clock: Clock) =>
223+
core.suspend(function loop(): Effect<void> {
224+
const now = clock.unsafeCurrentTimeMillis()
225+
const remaining = entry.expiresAt - now
226+
if (remaining <= 0) {
227+
return core.void
228+
}
229+
return core.flatMap(clock.sleep(Duration.millis(remaining)), loop)
230+
})
231+
216232
/** @internal */
217233
export const keys = <K, A, E>(self: RcMap.RcMap<K, A, E>): Effect<Array<K>> => {
218234
const impl = self as RcMapImpl<K, A, E>
@@ -239,3 +255,20 @@ export const invalidate: {
239255
if (entry.fiber) yield* core.interruptFiber(entry.fiber)
240256
})
241257
)
258+
259+
/** @internal */
260+
export const touch: {
261+
<K>(key: K): <A, E>(self: RcMap.RcMap<K, A, E>) => Effect<void>
262+
<K, A, E>(self: RcMap.RcMap<K, A, E>, key: K): Effect<void>
263+
} = dual(
264+
2,
265+
<K, A, E>(self_: RcMap.RcMap<K, A, E>, key: K) =>
266+
coreEffect.clockWith((clock) => {
267+
const self = self_ as RcMapImpl<K, A, E>
268+
if (!self.idleTimeToLive || self.state._tag === "Closed") return core.void
269+
const o = MutableHashMap.get(self.state.map, key)
270+
if (o._tag === "None") return core.void
271+
o.value.expiresAt = clock.unsafeCurrentTimeMillis() + Duration.toMillis(self.idleTimeToLive)
272+
return core.void
273+
})
274+
)

packages/effect/test/RcMap.test.ts

+31
Original file line numberDiff line numberDiff line change
@@ -97,6 +97,37 @@ describe("RcMap", () => {
9797
deepStrictEqual(released, ["foo", "bar", "baz"])
9898
}))
9999

100+
it.scoped(".touch", () =>
101+
Effect.gen(function*() {
102+
const acquired: Array<string> = []
103+
const released: Array<string> = []
104+
const map = yield* RcMap.make({
105+
lookup: (key: string) =>
106+
Effect.acquireRelease(
107+
Effect.sync(() => {
108+
acquired.push(key)
109+
return key
110+
}),
111+
() => Effect.sync(() => released.push(key))
112+
),
113+
idleTimeToLive: 1000
114+
})
115+
116+
assert.deepStrictEqual(acquired, [])
117+
assert.strictEqual(yield* Effect.scoped(RcMap.get(map, "foo")), "foo")
118+
assert.deepStrictEqual(acquired, ["foo"])
119+
assert.deepStrictEqual(released, [])
120+
121+
yield* TestClock.adjust(500)
122+
assert.deepStrictEqual(released, [])
123+
124+
yield* RcMap.touch(map, "foo")
125+
yield* TestClock.adjust(500)
126+
assert.deepStrictEqual(released, [])
127+
yield* TestClock.adjust(500)
128+
assert.deepStrictEqual(released, ["foo"])
129+
}))
130+
100131
it.scoped("capacity", () =>
101132
Effect.gen(function*() {
102133
const map = yield* RcMap.make({

0 commit comments

Comments
 (0)