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
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
|
//! Test fixture management for dual-mode testing
//!
//! This module provides a TestContext abstraction that manages prerequisite events
//! differently based on the audit mode:
//!
//! - **CI Mode (Isolated)**: Creates fresh events for each test, ensuring complete isolation
//! - **Production Mode (Shared)**: Reuses shared fixtures to minimize event publication
//!
//! # Example
//!
//! ```no_run
//! use grasp_audit::*;
//!
//! # async fn example() -> anyhow::Result<()> {
//! let config = AuditConfig::ci();
//! let client = AuditClient::new("ws://localhost:7000", config).await?;
//! let ctx = TestContext::new(&client);
//!
//! // Request a fixture - behavior depends on mode
//! let repo = ctx.get_fixture(FixtureKind::ValidRepo).await?;
//! # Ok(())
//! # }
//! ```
use crate::{AuditClient, AuditMode};
use anyhow::{Context, Result};
use nostr_sdk::prelude::Event;
use std::collections::HashMap;
use std::sync::{Arc, Mutex};
/// Types of test fixtures available
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
pub enum FixtureKind {
/// Basic repository announcement (kind 30617)
ValidRepo,
/// Repository with one issue (kind 1621)
RepoWithIssue,
/// Repository with issue and comment (kind 1111)
RepoWithComment,
/// Repository state announcement (kind 30618)
RepoState,
}
/// Context mode for fixture management
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
pub enum ContextMode {
/// Create fresh fixtures for each request (test isolation)
Isolated,
/// Reuse shared fixtures across requests (minimal events)
Shared,
}
impl From<AuditMode> for ContextMode {
fn from(mode: AuditMode) -> Self {
match mode {
AuditMode::CI => ContextMode::Isolated,
AuditMode::Production => ContextMode::Shared,
}
}
}
/// Test context for managing prerequisite events
///
/// The TestContext provides mode-aware fixture management:
/// - In Isolated mode: Creates fresh events for each test
/// - In Shared mode: Caches and reuses events across tests
///
/// # Example
///
/// ```no_run
/// # use grasp_audit::*;
/// # async fn example() -> anyhow::Result<()> {
/// let config = AuditConfig::ci();
/// let client = AuditClient::new("ws://localhost:7000", config).await?;
/// let ctx = TestContext::new(&client);
///
/// // Get a repository fixture
/// let repo = ctx.get_fixture(FixtureKind::ValidRepo).await?;
///
/// // In CI mode: Creates new repo
/// // In Production mode: Returns cached repo
/// # Ok(())
/// # }
/// ```
pub struct TestContext<'a> {
client: &'a AuditClient,
mode: ContextMode,
cache: Arc<Mutex<HashMap<FixtureKind, Event>>>,
}
impl<'a> TestContext<'a> {
/// Create a new test context
///
/// The context mode is automatically determined from the client's audit config.
pub fn new(client: &'a AuditClient) -> Self {
let mode = ContextMode::from(client.config.mode);
Self {
client,
mode,
cache: Arc::new(Mutex::new(HashMap::new())),
}
}
/// Create a test context with explicit mode override
///
/// This is useful for testing the context itself or for advanced use cases
/// where you want to override the default mode behavior.
pub fn with_mode(client: &'a AuditClient, mode: ContextMode) -> Self {
Self {
client,
mode,
cache: Arc::new(Mutex::new(HashMap::new())),
}
}
/// Get a fixture, creating it if needed based on mode
///
/// # Behavior
///
/// - **Isolated mode**: Always creates a fresh fixture
/// - **Shared mode**: Returns cached fixture or creates and caches if not present
///
/// # Example
///
/// ```no_run
/// # use grasp_audit::*;
/// # async fn example(ctx: &TestContext<'_>) -> anyhow::Result<()> {
/// let repo = ctx.get_fixture(FixtureKind::ValidRepo).await?;
/// # Ok(())
/// # }
/// ```
pub async fn get_fixture(&self, kind: FixtureKind) -> Result<Event> {
match self.mode {
ContextMode::Isolated => self.create_fresh(kind).await,
ContextMode::Shared => self.get_or_create_shared(kind).await,
}
}
/// Get the underlying client for direct access
///
/// This allows tests to use the client directly when needed while still
/// benefiting from the TestContext for fixture management.
pub fn client(&self) -> &'a AuditClient {
self.client
}
/// Get the current context mode
pub fn mode(&self) -> ContextMode {
self.mode
}
/// Create a fresh fixture (always creates new)
async fn create_fresh(&self, kind: FixtureKind) -> Result<Event> {
let event = self.build_fixture(kind).await
.with_context(|| format!("Failed to build {:?} fixture", kind))?;
self.client.send_event(event.clone()).await
.with_context(|| format!("Failed to send {:?} fixture event to relay", kind))?;
Ok(event)
}
/// Get or create a shared fixture (caches for reuse)
async fn get_or_create_shared(&self, kind: FixtureKind) -> Result<Event> {
// Check cache first
{
let cache = self.cache.lock().unwrap();
if let Some(event) = cache.get(&kind) {
return Ok(event.clone());
}
}
// Not in cache, create it
let event = self.build_fixture(kind).await
.with_context(|| format!("Failed to build {:?} fixture for shared cache", kind))?;
self.client.send_event(event.clone()).await
.with_context(|| format!("Failed to send {:?} fixture event to relay (shared cache)", kind))?;
// Store in cache
{
let mut cache = self.cache.lock().unwrap();
cache.insert(kind, event.clone());
}
Ok(event)
}
/// Build a fixture event (doesn't send it)
async fn build_fixture(&self, kind: FixtureKind) -> Result<Event> {
match kind {
FixtureKind::ValidRepo => {
let test_name = format!("fixture-{:?}-{}", kind, &uuid::Uuid::new_v4().to_string()[..8]);
self.client.create_repo_announcement(&test_name).await
}
FixtureKind::RepoWithIssue => {
use nostr_sdk::prelude::*;
// First create and send repo
let test_name = format!("fixture-{:?}-{}", FixtureKind::ValidRepo, &uuid::Uuid::new_v4().to_string()[..8]);
let repo = self.client.create_repo_announcement(&test_name).await?;
self.client.send_event(repo.clone()).await?;
// Then create issue referencing it - this will have 'a' tag to repo
// Note: We build the issue but DON'T send it here - the caller will send it
let issue = self.client.create_issue(
&repo,
"Test Issue",
"Issue content for testing",
vec![],
)?;
// Return the issue - tests can extract repo reference from its 'a' tag
// The caller (create_fresh/get_or_create_shared) will send this event
Ok(issue)
}
FixtureKind::RepoWithComment => {
// First create repo with issue
let test_name = format!("fixture-{:?}-{}", FixtureKind::ValidRepo, &uuid::Uuid::new_v4().to_string()[..8]);
let repo = self.client.create_repo_announcement(&test_name).await?;
self.client.send_event(repo.clone()).await?;
let issue = self.client.create_issue(
&repo,
"Test Issue",
"Issue content",
vec![],
)?;
self.client.send_event(issue.clone()).await?;
// Then create comment on issue
self.client.create_comment(
&issue,
"Test comment",
vec![],
)
}
FixtureKind::RepoState => {
use nostr_sdk::prelude::*;
// First create repo announcement
let test_name = format!("fixture-{:?}-{}", FixtureKind::ValidRepo, &uuid::Uuid::new_v4().to_string()[..8]);
let repo = self.client.create_repo_announcement(&test_name).await?;
self.client.send_event(repo.clone()).await?;
// Extract repo_id from repo announcement
let repo_id = repo.tags.iter()
.find(|t| t.kind() == TagKind::d())
.and_then(|t| t.content())
.ok_or_else(|| anyhow::anyhow!("Missing d tag in repo announcement"))?
.to_string();
// Create state announcement
self.client.event_builder(Kind::Custom(30618), "")
.tag(Tag::identifier(&repo_id))
.tag(Tag::custom(TagKind::custom("refs/heads/main"), vec![
"abc123def456789012345678901234567890abcd"
]))
.tag(Tag::custom(TagKind::custom("HEAD"), vec![
"ref: refs/heads/main"
]))
.build(self.client.keys())
.map_err(|e| anyhow::anyhow!("Failed to build state announcement: {}", e))
}
}
}
/// Clear the fixture cache
///
/// This is useful for tests that want to ensure fresh fixtures
/// even in shared mode.
pub fn clear_cache(&self) {
let mut cache = self.cache.lock().unwrap();
cache.clear();
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::AuditConfig;
#[test]
fn test_context_mode_from_audit_mode() {
assert_eq!(ContextMode::from(AuditMode::CI), ContextMode::Isolated);
assert_eq!(ContextMode::from(AuditMode::Production), ContextMode::Shared);
}
#[test]
fn test_fixture_kind_hash() {
use std::collections::HashSet;
let mut set = HashSet::new();
set.insert(FixtureKind::ValidRepo);
set.insert(FixtureKind::RepoWithIssue);
assert!(set.contains(&FixtureKind::ValidRepo));
assert!(!set.contains(&FixtureKind::RepoWithComment));
}
#[tokio::test]
async fn test_context_creation() {
let config = AuditConfig::ci();
let client = crate::AuditClient::new_test(config);
let ctx = TestContext::new(&client);
assert_eq!(ctx.mode(), ContextMode::Isolated);
let ctx = TestContext::with_mode(&client, ContextMode::Shared);
assert_eq!(ctx.mode(), ContextMode::Shared);
}
}
|