-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathArtifactStore.php
More file actions
113 lines (102 loc) · 3.4 KB
/
ArtifactStore.php
File metadata and controls
113 lines (102 loc) · 3.4 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
<?php
declare(strict_types=1);
namespace Arcp\Runtime;
use Arcp\Clock\ClockInterface;
use Arcp\Clock\SystemClock;
use Arcp\Errors\InvalidArgumentException;
use Arcp\Errors\NotFoundException;
use Arcp\Ids\ArtifactId;
use Arcp\Messages\Artifacts\ArtifactRef;
/**
* In-memory artifact store keyed by session (RFC §16). Implements the
* inline-base64 path for v0.1; sidecar binary frames are deferred.
*
* Retention: every put is stamped with `expires_at`. The store does not
* spawn its own sweep fiber; the runtime calls {@see sweep()} on a
* periodic timer.
*
* @phpstan-type StoredArtifact array{
* ref: ArtifactRef,
* bytes: string,
* session_key: string,
* }
*/
final class ArtifactStore
{
/** @var array<string, StoredArtifact> indexed by artifact id */
private array $artifacts = [];
public function __construct(
private readonly ClockInterface $clock = new SystemClock(),
public readonly int $defaultRetentionSeconds = 86400,
public readonly int $maxRetentionSeconds = 604800,
) {
}
public function put(Session $session, ArtifactBlob $blob): ArtifactRef
{
$sessionId = (string) (
$session->sessionId
?? throw new InvalidArgumentException('session has no id')
);
$retention = $blob->retentionSeconds ?? $this->defaultRetentionSeconds;
$retention = min($retention, $this->maxRetentionSeconds);
$id = ArtifactId::random();
$expiresAt = $this->clock->now()->modify('+' . $retention . ' seconds');
$ref = new ArtifactRef(
artifactId: $id,
uri: 'arcp://session/' . $sessionId . '/artifact/' . $id->value,
mediaType: $blob->mediaType,
size: \strlen($blob->bytes),
sha256: hash('sha256', $blob->bytes),
expiresAt: $expiresAt,
);
$this->artifacts[(string) $id] = [
'ref' => $ref,
'bytes' => $blob->bytes,
'session_key' => $sessionId,
];
return $ref;
}
public function fetch(ArtifactId $id): string
{
$row = $this->artifacts[(string) $id]
?? throw new NotFoundException(\sprintf('artifact %s not found', $id));
$expiresAt = $row['ref']->expiresAt;
if ($expiresAt !== null && $expiresAt <= $this->clock->now()) {
unset($this->artifacts[(string) $id]);
throw new NotFoundException(\sprintf('artifact %s expired', $id));
}
return $row['bytes'];
}
public function ref(ArtifactId $id): ArtifactRef
{
$row = $this->artifacts[(string) $id]
?? throw new NotFoundException(\sprintf('artifact %s not found', $id));
return $row['ref'];
}
public function release(ArtifactId $id): bool
{
if (!isset($this->artifacts[(string) $id])) {
return false;
}
unset($this->artifacts[(string) $id]);
return true;
}
/** Remove every expired artifact. */
public function sweep(): int
{
$now = $this->clock->now();
$removed = 0;
foreach ($this->artifacts as $key => $row) {
$exp = $row['ref']->expiresAt;
if ($exp !== null && $exp <= $now) {
unset($this->artifacts[$key]);
++$removed;
}
}
return $removed;
}
public function count(): int
{
return \count($this->artifacts);
}
}