diff --git a/Cargo.lock b/Cargo.lock
index 27c456d179e..d69d792197f 100644
--- a/Cargo.lock
+++ b/Cargo.lock
@@ -3830,6 +3830,7 @@ dependencies = [
"hash-codec",
"hashbrown 0.16.1",
"hashql-diagnostics",
+ "hashql-macros",
"insta",
"lexical",
"memchr",
@@ -3894,6 +3895,15 @@ dependencies = [
"tracing",
]
+[[package]]
+name = "hashql-macros"
+version = "0.0.0"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "unsynn",
+]
+
[[package]]
name = "hashql-mir"
version = "0.0.0"
@@ -5512,6 +5522,12 @@ dependencies = [
"unsigned-varint 0.7.2",
]
+[[package]]
+name = "mutants"
+version = "0.0.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "bc0287524726960e07b119cebd01678f852f147742ae0d925e6a520dca956126"
+
[[package]]
name = "napi"
version = "2.16.17"
@@ -9890,6 +9906,16 @@ version = "0.8.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "eb066959b24b5196ae73cb057f45598450d2c5f71460e98c49b738086eff9c06"
+[[package]]
+name = "unsynn"
+version = "0.3.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "501a7adf1a4bd9951501e5c66621e972ef8874d787628b7f90e64f936ef7ec0a"
+dependencies = [
+ "mutants",
+ "proc-macro2",
+]
+
[[package]]
name = "untrusted"
version = "0.9.0"
diff --git a/Cargo.toml b/Cargo.toml
index 6c9060ef490..296526c7173 100644
--- a/Cargo.toml
+++ b/Cargo.toml
@@ -97,6 +97,7 @@ hashql-core.path = "libs/@local/hashql/core"
hashql-diagnostics.path = "libs/@local/hashql/diagnostics"
hashql-eval.path = "libs/@local/hashql/eval"
hashql-hir.path = "libs/@local/hashql/hir"
+hashql-macros.path = "libs/@local/hashql/macros"
hashql-mir.path = "libs/@local/hashql/mir"
hashql-syntax-jexpr.path = "libs/@local/hashql/syntax-jexpr"
type-system.path = "libs/@blockprotocol/type-system/rust"
@@ -285,6 +286,7 @@ unicode-ident = { version = "1.0.22", default-features = fa
unicode-normalization = { version = "0.1.25", default-features = false }
unicode-properties = { version = "0.1.4", default-features = false }
unicode-segmentation = { version = "1.12.0", default-features = false }
+unsynn = { version = "0.3.0", default-features = false, features = ["proc_macro2"] }
url = { version = "2.5.7", default-features = false }
utoipa = { version = "4.2.3", default-features = false }
uuid = { version = "1.18.1", default-features = false }
diff --git a/apps/hash-graph/docs/dependency-diagram.mmd b/apps/hash-graph/docs/dependency-diagram.mmd
index 4eacaedd9fb..e1d32bd7d26 100644
--- a/apps/hash-graph/docs/dependency-diagram.mmd
+++ b/apps/hash-graph/docs/dependency-diagram.mmd
@@ -39,20 +39,21 @@ graph TD
27[hashql-diagnostics]
28[hashql-eval]
29[hashql-hir]
- 30[hashql-mir]
- 31[hashql-syntax-jexpr]
- 32[hash-status]
- 33[hash-telemetry]
- 34[hash-temporal-client]
- 35[darwin-kperf]
- 36[darwin-kperf-criterion]
- 37[darwin-kperf-events]
- 38[darwin-kperf-sys]
- 39[error-stack]
- 40[hash-graph-test-data]
+ 30[hashql-macros]
+ 31[hashql-mir]
+ 32[hashql-syntax-jexpr]
+ 33[hash-status]
+ 34[hash-telemetry]
+ 35[hash-temporal-client]
+ 36[darwin-kperf]
+ 37[darwin-kperf-criterion]
+ 38[darwin-kperf-events]
+ 39[darwin-kperf-sys]
+ 40[error-stack]
+ 41[hash-graph-test-data]
0 --> 11
1 --> 10
- 1 -.-> 40
+ 1 -.-> 41
2 -.-> 3
2 --> 23
4 --> 8
@@ -60,25 +61,25 @@ graph TD
4 --> 13
4 --> 19
4 --> 28
- 4 --> 31
+ 4 --> 32
5 --> 1
6 --> 7
- 6 --> 33
+ 6 --> 34
8 -.-> 6
8 --> 15
- 8 --> 32
+ 8 --> 33
9 --> 5
9 --> 14
- 9 --> 34
+ 9 --> 35
10 --> 2
11 --> 4
- 12 --> 32
+ 12 --> 33
13 --> 9
- 14 -.-> 40
- 15 -.-> 40
+ 14 -.-> 41
+ 15 -.-> 41
16 --> 20
17 --> 22
- 17 --> 39
+ 17 --> 40
18 --> 2
18 -.-> 17
18 --> 17
@@ -89,24 +90,25 @@ graph TD
21 --> 18
23 -.-> 22
23 --> 22
- 23 --> 39
+ 23 --> 40
24 -.-> 25
25 --> 28
- 25 --> 30
25 --> 31
- 25 --> 39
+ 25 --> 32
+ 25 --> 40
26 --> 2
26 --> 27
- 26 -.-> 36
+ 26 --> 30
+ 26 -.-> 37
28 --> 9
28 --> 29
29 -.-> 25
- 30 --> 29
- 31 --> 24
- 31 --> 26
- 33 --> 39
- 34 --> 1
- 35 --> 37
- 35 --> 38
- 36 --> 35
- 40 --> 9
+ 31 --> 29
+ 32 --> 24
+ 32 --> 26
+ 34 --> 40
+ 35 --> 1
+ 36 --> 38
+ 36 --> 39
+ 37 --> 36
+ 41 --> 9
diff --git a/libs/@local/graph/api/docs/dependency-diagram.mmd b/libs/@local/graph/api/docs/dependency-diagram.mmd
index f4ec1edf964..c81612f0bd1 100644
--- a/libs/@local/graph/api/docs/dependency-diagram.mmd
+++ b/libs/@local/graph/api/docs/dependency-diagram.mmd
@@ -39,21 +39,22 @@ graph TD
27[hashql-diagnostics]
28[hashql-eval]
29[hashql-hir]
- 30[hashql-mir]
- 31[hashql-syntax-jexpr]
- 32[hash-status]
- 33[hash-telemetry]
- 34[hash-temporal-client]
- 35[darwin-kperf]
- 36[darwin-kperf-criterion]
- 37[darwin-kperf-events]
- 38[darwin-kperf-sys]
- 39[error-stack]
- 40[hash-graph-benches]
- 41[hash-graph-test-data]
+ 30[hashql-macros]
+ 31[hashql-mir]
+ 32[hashql-syntax-jexpr]
+ 33[hash-status]
+ 34[hash-telemetry]
+ 35[hash-temporal-client]
+ 36[darwin-kperf]
+ 37[darwin-kperf-criterion]
+ 38[darwin-kperf-events]
+ 39[darwin-kperf-sys]
+ 40[error-stack]
+ 41[hash-graph-benches]
+ 42[hash-graph-test-data]
0 --> 11
1 --> 10
- 1 -.-> 41
+ 1 -.-> 42
2 -.-> 3
2 --> 23
4 --> 8
@@ -61,25 +62,25 @@ graph TD
4 --> 13
4 --> 19
4 --> 28
- 4 --> 31
+ 4 --> 32
5 --> 1
6 --> 7
- 6 --> 33
+ 6 --> 34
8 -.-> 6
8 --> 15
- 8 --> 32
+ 8 --> 33
9 --> 5
9 --> 14
- 9 --> 34
+ 9 --> 35
10 --> 2
11 --> 4
- 12 --> 32
+ 12 --> 33
13 --> 9
- 14 -.-> 41
- 15 -.-> 41
+ 14 -.-> 42
+ 15 -.-> 42
16 --> 20
17 --> 22
- 17 --> 39
+ 17 --> 40
18 --> 2
18 -.-> 17
18 --> 17
@@ -90,25 +91,26 @@ graph TD
21 --> 18
23 -.-> 22
23 --> 22
- 23 --> 39
+ 23 --> 40
24 -.-> 25
25 --> 28
- 25 --> 30
25 --> 31
- 25 --> 39
+ 25 --> 32
+ 25 --> 40
26 --> 2
26 --> 27
- 26 -.-> 36
+ 26 --> 30
+ 26 -.-> 37
28 --> 9
28 --> 29
29 -.-> 25
- 30 --> 29
- 31 --> 24
- 31 --> 26
- 33 --> 39
- 34 --> 1
- 35 --> 37
- 35 --> 38
- 36 --> 35
- 40 -.-> 4
- 41 --> 9
+ 31 --> 29
+ 32 --> 24
+ 32 --> 26
+ 34 --> 40
+ 35 --> 1
+ 36 --> 38
+ 36 --> 39
+ 37 --> 36
+ 41 -.-> 4
+ 42 --> 9
diff --git a/libs/@local/graph/test-server/docs/dependency-diagram.mmd b/libs/@local/graph/test-server/docs/dependency-diagram.mmd
index 52e1f52912c..87719d33cc2 100644
--- a/libs/@local/graph/test-server/docs/dependency-diagram.mmd
+++ b/libs/@local/graph/test-server/docs/dependency-diagram.mmd
@@ -39,20 +39,21 @@ graph TD
27[hashql-diagnostics]
28[hashql-eval]
29[hashql-hir]
- 30[hashql-mir]
- 31[hashql-syntax-jexpr]
- 32[hash-status]
- 33[hash-telemetry]
- 34[hash-temporal-client]
- 35[darwin-kperf]
- 36[darwin-kperf-criterion]
- 37[darwin-kperf-events]
- 38[darwin-kperf-sys]
- 39[error-stack]
- 40[hash-graph-test-data]
+ 30[hashql-macros]
+ 31[hashql-mir]
+ 32[hashql-syntax-jexpr]
+ 33[hash-status]
+ 34[hash-telemetry]
+ 35[hash-temporal-client]
+ 36[darwin-kperf]
+ 37[darwin-kperf-criterion]
+ 38[darwin-kperf-events]
+ 39[darwin-kperf-sys]
+ 40[error-stack]
+ 41[hash-graph-test-data]
0 --> 11
1 --> 10
- 1 -.-> 40
+ 1 -.-> 41
2 -.-> 3
2 --> 23
4 --> 8
@@ -60,25 +61,25 @@ graph TD
4 --> 13
4 --> 19
4 --> 28
- 4 --> 31
+ 4 --> 32
5 --> 1
6 --> 7
- 6 --> 33
+ 6 --> 34
8 -.-> 6
8 --> 15
- 8 --> 32
+ 8 --> 33
9 --> 5
9 --> 14
- 9 --> 34
+ 9 --> 35
10 --> 2
11 --> 4
- 12 --> 32
+ 12 --> 33
13 --> 9
- 14 -.-> 40
- 15 -.-> 40
+ 14 -.-> 41
+ 15 -.-> 41
16 --> 20
17 --> 22
- 17 --> 39
+ 17 --> 40
18 --> 2
18 -.-> 17
18 --> 17
@@ -89,24 +90,25 @@ graph TD
21 --> 18
23 -.-> 22
23 --> 22
- 23 --> 39
+ 23 --> 40
24 -.-> 25
25 --> 28
- 25 --> 30
25 --> 31
- 25 --> 39
+ 25 --> 32
+ 25 --> 40
26 --> 2
26 --> 27
- 26 -.-> 36
+ 26 --> 30
+ 26 -.-> 37
28 --> 9
28 --> 29
29 -.-> 25
- 30 --> 29
- 31 --> 24
- 31 --> 26
- 33 --> 39
- 34 --> 1
- 35 --> 37
- 35 --> 38
- 36 --> 35
- 40 --> 9
+ 31 --> 29
+ 32 --> 24
+ 32 --> 26
+ 34 --> 40
+ 35 --> 1
+ 36 --> 38
+ 36 --> 39
+ 37 --> 36
+ 41 --> 9
diff --git a/libs/@local/hashql/ast/docs/dependency-diagram.mmd b/libs/@local/hashql/ast/docs/dependency-diagram.mmd
index 2d73a21e90c..c468334f01e 100644
--- a/libs/@local/hashql/ast/docs/dependency-diagram.mmd
+++ b/libs/@local/hashql/ast/docs/dependency-diagram.mmd
@@ -27,50 +27,52 @@ graph TD
15[hashql-diagnostics]
16[hashql-eval]
17[hashql-hir]
- 18[hashql-mir]
- 19[hashql-syntax-jexpr]
- 20[hash-temporal-client]
- 21[darwin-kperf]
- 22[darwin-kperf-criterion]
- 23[darwin-kperf-events]
- 24[darwin-kperf-sys]
- 25[error-stack]
- 26[hash-graph-benches]
- 27[hash-graph-test-data]
+ 18[hashql-macros]
+ 19[hashql-mir]
+ 20[hashql-syntax-jexpr]
+ 21[hash-temporal-client]
+ 22[darwin-kperf]
+ 23[darwin-kperf-criterion]
+ 24[darwin-kperf-events]
+ 25[darwin-kperf-sys]
+ 26[error-stack]
+ 27[hash-graph-benches]
+ 28[hash-graph-test-data]
0 --> 8
1 --> 7
- 1 -.-> 27
+ 1 -.-> 28
2 -.-> 3
2 --> 11
4 --> 16
- 4 --> 19
+ 4 --> 20
5 --> 1
6 --> 5
6 --> 9
- 6 --> 20
+ 6 --> 21
7 --> 2
8 --> 4
- 9 -.-> 27
+ 9 -.-> 28
11 -.-> 10
11 --> 10
- 11 --> 25
+ 11 --> 26
12 -.-> 13
13 --> 16
- 13 --> 18
13 --> 19
- 13 --> 25
+ 13 --> 20
+ 13 --> 26
14 --> 2
14 --> 15
- 14 -.-> 22
+ 14 --> 18
+ 14 -.-> 23
16 --> 6
16 --> 17
17 -.-> 13
- 18 --> 17
- 19 --> 12
- 19 --> 14
- 20 --> 1
- 21 --> 23
- 21 --> 24
- 22 --> 21
- 26 -.-> 4
- 27 --> 6
+ 19 --> 17
+ 20 --> 12
+ 20 --> 14
+ 21 --> 1
+ 22 --> 24
+ 22 --> 25
+ 23 --> 22
+ 27 -.-> 4
+ 28 --> 6
diff --git a/libs/@local/hashql/compiletest/docs/dependency-diagram.mmd b/libs/@local/hashql/compiletest/docs/dependency-diagram.mmd
index 9c1a9805c01..433bed9565b 100644
--- a/libs/@local/hashql/compiletest/docs/dependency-diagram.mmd
+++ b/libs/@local/hashql/compiletest/docs/dependency-diagram.mmd
@@ -27,50 +27,52 @@ graph TD
15[hashql-diagnostics]
16[hashql-eval]
17[hashql-hir]
- 18[hashql-mir]
- 19[hashql-syntax-jexpr]
- 20[hash-temporal-client]
- 21[darwin-kperf]
- 22[darwin-kperf-criterion]
- 23[darwin-kperf-events]
- 24[darwin-kperf-sys]
- 25[error-stack]
- 26[hash-graph-benches]
- 27[hash-graph-test-data]
+ 18[hashql-macros]
+ 19[hashql-mir]
+ 20[hashql-syntax-jexpr]
+ 21[hash-temporal-client]
+ 22[darwin-kperf]
+ 23[darwin-kperf-criterion]
+ 24[darwin-kperf-events]
+ 25[darwin-kperf-sys]
+ 26[error-stack]
+ 27[hash-graph-benches]
+ 28[hash-graph-test-data]
0 --> 8
1 --> 7
- 1 -.-> 27
+ 1 -.-> 28
2 -.-> 3
2 --> 11
4 --> 16
- 4 --> 19
+ 4 --> 20
5 --> 1
6 --> 5
6 --> 9
- 6 --> 20
+ 6 --> 21
7 --> 2
8 --> 4
- 9 -.-> 27
+ 9 -.-> 28
11 -.-> 10
11 --> 10
- 11 --> 25
+ 11 --> 26
12 -.-> 13
13 --> 16
- 13 --> 18
13 --> 19
- 13 --> 25
+ 13 --> 20
+ 13 --> 26
14 --> 2
14 --> 15
- 14 -.-> 22
+ 14 --> 18
+ 14 -.-> 23
16 --> 6
16 --> 17
17 -.-> 13
- 18 --> 17
- 19 --> 12
- 19 --> 14
- 20 --> 1
- 21 --> 23
- 21 --> 24
- 22 --> 21
- 26 -.-> 4
- 27 --> 6
+ 19 --> 17
+ 20 --> 12
+ 20 --> 14
+ 21 --> 1
+ 22 --> 24
+ 22 --> 25
+ 23 --> 22
+ 27 -.-> 4
+ 28 --> 6
diff --git a/libs/@local/hashql/core/Cargo.toml b/libs/@local/hashql/core/Cargo.toml
index 995af13ac17..c1c1f638fc5 100644
--- a/libs/@local/hashql/core/Cargo.toml
+++ b/libs/@local/hashql/core/Cargo.toml
@@ -10,6 +10,7 @@ authors.workspace = true
# Public workspace dependencies
hash-codec = { workspace = true, features = ["numeric"], public = true }
hashql-diagnostics = { workspace = true, public = true }
+hashql-macros = { workspace = true, public = true }
# Public third-party dependencies
anstyle = { workspace = true, public = true }
diff --git a/libs/@local/hashql/core/benches/bit_matrix.rs b/libs/@local/hashql/core/benches/bit_matrix.rs
index 0b22bd1da7f..4e95d82dc8d 100644
--- a/libs/@local/hashql/core/benches/bit_matrix.rs
+++ b/libs/@local/hashql/core/benches/bit_matrix.rs
@@ -4,15 +4,13 @@ use core::hint::black_box;
use codspeed_criterion_compat::{
BatchSize, BenchmarkId, Criterion, criterion_group, criterion_main,
};
-use hashql_core::{
- id::{
- Id as _,
- bit_vec::{BitMatrix, SparseBitMatrix},
- },
+use hashql_core::id::{
+ Id as _,
+ bit_vec::{BitMatrix, SparseBitMatrix},
newtype,
};
-newtype!(struct BenchId(usize is 0..=usize::MAX));
+newtype!(struct BenchId(u64 is 0..=u64::MAX));
// =============================================================================
// Dense BitMatrix
diff --git a/libs/@local/hashql/core/docs/dependency-diagram.mmd b/libs/@local/hashql/core/docs/dependency-diagram.mmd
index 0c5f479e9a8..6615a2d475b 100644
--- a/libs/@local/hashql/core/docs/dependency-diagram.mmd
+++ b/libs/@local/hashql/core/docs/dependency-diagram.mmd
@@ -22,37 +22,39 @@ graph TD
10[hashql-diagnostics]
11[hashql-eval]
12[hashql-hir]
- 13[hashql-mir]
- 14[hashql-syntax-jexpr]
- 15[darwin-kperf]
- 16[darwin-kperf-criterion]
- 17[darwin-kperf-events]
- 18[darwin-kperf-sys]
- 19[error-stack]
- 20[hash-graph-benches]
+ 13[hashql-macros]
+ 14[hashql-mir]
+ 15[hashql-syntax-jexpr]
+ 16[darwin-kperf]
+ 17[darwin-kperf-criterion]
+ 18[darwin-kperf-events]
+ 19[darwin-kperf-sys]
+ 20[error-stack]
+ 21[hash-graph-benches]
0 --> 4
1 -.-> 2
1 --> 6
3 --> 11
- 3 --> 14
+ 3 --> 15
4 --> 3
6 -.-> 5
6 --> 5
- 6 --> 19
+ 6 --> 20
7 -.-> 8
8 --> 11
- 8 --> 13
8 --> 14
- 8 --> 19
+ 8 --> 15
+ 8 --> 20
9 --> 1
9 --> 10
- 9 -.-> 16
+ 9 --> 13
+ 9 -.-> 17
11 --> 12
12 -.-> 8
- 13 --> 12
- 14 --> 7
- 14 --> 9
- 15 --> 17
- 15 --> 18
- 16 --> 15
- 20 -.-> 3
+ 14 --> 12
+ 15 --> 7
+ 15 --> 9
+ 16 --> 18
+ 16 --> 19
+ 17 --> 16
+ 21 -.-> 3
diff --git a/libs/@local/hashql/core/package.json b/libs/@local/hashql/core/package.json
index 43668aa22c1..e8b0af5a925 100644
--- a/libs/@local/hashql/core/package.json
+++ b/libs/@local/hashql/core/package.json
@@ -14,7 +14,8 @@
},
"dependencies": {
"@rust/hash-codec": "workspace:*",
- "@rust/hashql-diagnostics": "workspace:*"
+ "@rust/hashql-diagnostics": "workspace:*",
+ "@rust/hashql-macros": "workspace:*"
},
"devDependencies": {
"@rust/darwin-kperf-criterion": "workspace:*"
diff --git a/libs/@local/hashql/core/src/graph/algorithms/dominators/mod.rs b/libs/@local/hashql/core/src/graph/algorithms/dominators/mod.rs
index c1e5ebf871e..fc78e863dd7 100644
--- a/libs/@local/hashql/core/src/graph/algorithms/dominators/mod.rs
+++ b/libs/@local/hashql/core/src/graph/algorithms/dominators/mod.rs
@@ -41,8 +41,7 @@ pub use self::{
};
use crate::{
graph::{DirectedGraph, Predecessors, Successors},
- id::{Id, IdSlice, IdVec},
- newtype,
+ id::{Id, IdSlice, IdVec, newtype},
};
mod frontier;
@@ -56,7 +55,7 @@ struct PreOrderFrame {
}
newtype!(
- #[steppable]
+ #[id(derive(Step), crate = crate)]
struct PreorderIndex(u32 is 0..=u32::MAX)
);
@@ -444,7 +443,7 @@ struct Time {
finish: u32,
}
-newtype!(struct EdgeIndex(u32 is 0..=u32::MAX));
+newtype!(#[id(crate = crate)] struct EdgeIndex(u32 is 0..=u32::MAX));
fn compute_access_time(
start_node: N,
diff --git a/libs/@local/hashql/core/src/graph/algorithms/tarjan/mod.rs b/libs/@local/hashql/core/src/graph/algorithms/tarjan/mod.rs
index 24eef14a5e4..edda7708ba7 100644
--- a/libs/@local/hashql/core/src/graph/algorithms/tarjan/mod.rs
+++ b/libs/@local/hashql/core/src/graph/algorithms/tarjan/mod.rs
@@ -18,13 +18,12 @@ use crate::{
collections::{FastHashSet, fast_hash_set_in},
graph::{DirectedGraph, EdgeId, Successors},
heap::BumpAllocator,
- id::{HasId, Id, IdSlice, IdVec},
- newtype,
+ id::{HasId, Id, IdSlice, IdVec, newtype},
};
-newtype!(pub struct SccId(u32 is 0..=u32::MAX));
+newtype!(#[id(crate = crate)] pub struct SccId(u32 is 0..=u32::MAX));
-newtype!(struct DiscoveryTime(usize is 0..=usize::MAX));
+newtype!(#[id(crate = crate)] struct DiscoveryTime(u32 is 0..=u32::MAX));
/// Trait for attaching metadata to nodes and strongly connected components during traversal.
///
@@ -408,7 +407,7 @@ where
}
fn iter_edges(&self) -> impl ExactSizeIterator- > + DoubleEndedIterator {
- (0..self.edge_count()).map(EdgeId::new)
+ (0..self.edge_count()).map(EdgeId::from_usize)
}
}
diff --git a/libs/@local/hashql/core/src/graph/algorithms/tarjan/tests.rs b/libs/@local/hashql/core/src/graph/algorithms/tarjan/tests.rs
index e48c68c8839..0c0bb613d04 100644
--- a/libs/@local/hashql/core/src/graph/algorithms/tarjan/tests.rs
+++ b/libs/@local/hashql/core/src/graph/algorithms/tarjan/tests.rs
@@ -17,11 +17,10 @@ use crate::{
DirectedGraph as _, NodeId, Successors as _, algorithms::tarjan::Tarjan, tests::TestGraph,
},
heap::Scratch,
- id::Id as _,
- newtype,
+ id::{Id as _, newtype},
};
-newtype!(struct SccId(usize is 0..=usize::MAX));
+newtype!(#[id(crate = crate)] struct SccId(u32 is 0..=u32::MAX));
type Sccs = StronglyConnectedComponents;
diff --git a/libs/@local/hashql/core/src/graph/linked.rs b/libs/@local/hashql/core/src/graph/linked.rs
index b5469e3100e..569139dbb00 100644
--- a/libs/@local/hashql/core/src/graph/linked.rs
+++ b/libs/@local/hashql/core/src/graph/linked.rs
@@ -62,7 +62,7 @@ use crate::id::{HasId, Id, IdSlice, IdVec};
///
/// Uses the maximum [`EdgeId`] value, which can never be a valid edge ID since
/// edge insertion would overflow before reaching this value.
-const TOMBSTONE: EdgeId = EdgeId(usize::MAX);
+const TOMBSTONE: EdgeId = EdgeId::MAX;
/// A node in a [`LinkedGraph`] with associated data.
///
@@ -319,7 +319,7 @@ impl LinkedGraph {
/// # use hashql_core::graph::LinkedGraph;
/// # use hashql_core::id::{Id, IdVec};
/// #
- /// # hashql_core::id::newtype!(struct MyId(usize is 0..=usize::MAX));
+ /// # hashql_core::id::newtype!(struct MyId(u32 is 0..=u32::MAX));
/// #
/// let mut items: IdVec = IdVec::new();
/// items.push("first");
diff --git a/libs/@local/hashql/core/src/graph/mod.rs b/libs/@local/hashql/core/src/graph/mod.rs
index 81f76cbc2b3..7bb9672b461 100644
--- a/libs/@local/hashql/core/src/graph/mod.rs
+++ b/libs/@local/hashql/core/src/graph/mod.rs
@@ -39,8 +39,8 @@ use self::algorithms::{
pub use self::linked::LinkedGraph;
use crate::id::{HasId, Id, newtype};
-newtype!(pub struct NodeId(usize is 0..=usize::MAX));
-newtype!(pub struct EdgeId(usize is 0..=usize::MAX));
+newtype!(#[id(crate = crate)] pub struct NodeId(u32 is 0..=u32::MAX));
+newtype!(#[id(crate = crate)] pub struct EdgeId(u32 is 0..=u32::MAX));
/// Direction of edge traversal in a directed graph.
///
diff --git a/libs/@local/hashql/core/src/graph/tests.rs b/libs/@local/hashql/core/src/graph/tests.rs
index bdeb92a2400..5abad5dc5bc 100644
--- a/libs/@local/hashql/core/src/graph/tests.rs
+++ b/libs/@local/hashql/core/src/graph/tests.rs
@@ -19,15 +19,15 @@ impl TestGraph {
};
for &(source, target) in edges {
- let source = NodeId::new(source);
- let target = NodeId::new(target);
+ let source = NodeId::from_usize(source);
+ let target = NodeId::from_usize(target);
graph.node_count = graph
.node_count
.max(source.as_usize() + 1)
.max(target.as_usize() + 1);
- let edge_id = EdgeId::new(graph.edge_count);
+ let edge_id = EdgeId::from_usize(graph.edge_count);
graph
.successors
@@ -45,7 +45,7 @@ impl TestGraph {
}
for node in 0..graph.node_count {
- let node = NodeId::new(node);
+ let node = NodeId::from_usize(node);
graph.successors.entry(node).or_default();
graph.predecessors.entry(node).or_default();
@@ -76,11 +76,11 @@ impl DirectedGraph for TestGraph {
}
fn iter_nodes(&self) -> impl ExactSizeIterator
- > + DoubleEndedIterator {
- (0..self.node_count).map(NodeId::new)
+ (0..self.node_count).map(NodeId::from_usize)
}
fn iter_edges(&self) -> impl ExactSizeIterator
- > + DoubleEndedIterator {
- (0..self.edge_count).map(EdgeId::new)
+ (0..self.edge_count).map(EdgeId::from_usize)
}
}
diff --git a/libs/@local/hashql/core/src/id/array.rs b/libs/@local/hashql/core/src/id/array.rs
index b075621fadb..5a15325251c 100644
--- a/libs/@local/hashql/core/src/id/array.rs
+++ b/libs/@local/hashql/core/src/id/array.rs
@@ -19,7 +19,7 @@ use super::{Id, IdSlice};
/// # Examples
///
/// ```
-/// # use hashql_core::{id::{IdArray, Id as _}, newtype};
+/// # use hashql_core::id::{IdArray, Id as _, newtype};
/// # newtype!(struct TargetId(u32 is 0..=3));
/// // Create an array where each element is initialized based on its index
/// let costs = IdArray::::from_fn(|id| id.as_u32() * 10);
@@ -41,7 +41,7 @@ impl IdArray {
/// # Examples
///
/// ```
- /// # use hashql_core::{id::{IdArray, Id as _}, newtype};
+ /// # use hashql_core::id::{IdArray, Id as _, newtype};
/// # newtype!(struct SlotId(u32 is 0..=2));
/// let array = IdArray::::from_raw(["a", "b", "c"]);
/// assert_eq!(array[SlotId::new(1)], "b");
@@ -59,7 +59,7 @@ impl IdArray {
/// # Examples
///
/// ```
- /// # use hashql_core::{id::{IdArray, Id as _}, newtype};
+ /// # use hashql_core::id::{IdArray, Id as _, newtype};
/// # newtype!(struct SlotId(u32 is 0..=2));
/// let array = IdArray::::from_raw([1, 2, 3]);
/// let raw: [i32; 3] = array.into_raw();
@@ -82,7 +82,7 @@ impl IdArray {
/// # Examples
///
/// ```
- /// # use hashql_core::{id::{IdArray, Id as _}, newtype};
+ /// # use hashql_core::id::{IdArray, Id as _, newtype};
/// # newtype!(struct NodeId(u32 is 0..=100));
/// // Create an array where each element equals its index squared
/// let squares = IdArray::::from_fn(|id| id.as_u32() * id.as_u32());
@@ -101,7 +101,7 @@ impl IdArray {
/// # Examples
///
/// ```
- /// # use hashql_core::{id::{IdArray, Id as _}, newtype};
+ /// # use hashql_core::id::{IdArray, Id as _, newtype};
/// # newtype!(struct SlotId(u32 is 0..=4));
/// let array = IdArray::::from_elem(42);
///
@@ -121,7 +121,7 @@ impl IdArray {
/// # Examples
///
/// ```
- /// # use hashql_core::{id::{IdArray, Id as _}, newtype};
+ /// # use hashql_core::id::{IdArray, Id as _, newtype};
/// # newtype!(struct SlotId(u32 is 0..=2));
/// let array =
/// IdArray::::from_raw(["a".to_string(), "b".to_string(), "c".to_string()]);
@@ -138,7 +138,7 @@ impl IdArray {
/// # Examples
///
/// ```
- /// # use hashql_core::{id::{IdArray, Id as _}, newtype};
+ /// # use hashql_core::id::{IdArray, Id as _, newtype};
/// # newtype!(struct SlotId(u32 is 0..=2));
/// let mut array = IdArray::::from_raw([1, 2, 3]);
/// for elem in array.each_mut().into_raw() {
@@ -157,7 +157,7 @@ impl IdArray {
/// # Examples
///
/// ```
- /// # use hashql_core::{id::{IdArray, Id as _}, newtype};
+ /// # use hashql_core::id::{IdArray, Id as _, newtype};
/// # newtype!(struct SlotId(u32 is 0..=2));
/// let array = IdArray::::from_raw([1, 2, 3]);
/// let doubled: IdArray = array.map(|x| x * 2);
@@ -175,7 +175,7 @@ impl IdArray {
/// # Examples
///
/// ```
- /// # use hashql_core::{id::{IdArray, Id as _}, newtype};
+ /// # use hashql_core::id::{IdArray, Id as _, newtype};
/// # newtype!(struct SlotId(u32 is 0..=100));
/// let array = IdArray::::from_raw([10, 20, 30]);
/// let indexed: IdArray =
@@ -202,7 +202,7 @@ impl IdArray {
/// # Examples
///
/// ```
- /// # use hashql_core::{id::{IdArray, Id as _}, newtype};
+ /// # use hashql_core::id::{IdArray, Id as _, newtype};
/// # newtype!(struct SlotId(u32 is 0..=2));
/// let array = IdArray::::from_raw([1, 2, 3]);
/// let slice = array.as_slice();
@@ -220,7 +220,7 @@ impl IdArray {
/// # Examples
///
/// ```
- /// # use hashql_core::{id::{IdArray, Id as _}, newtype};
+ /// # use hashql_core::id::{IdArray, Id as _, newtype};
/// # newtype!(struct SlotId(u32 is 0..=2));
/// let mut array = IdArray::::from_raw([1, 2, 3]);
/// array.as_mut_slice()[SlotId::new(1)] = 42;
@@ -239,7 +239,7 @@ impl IdArray {
/// # Examples
///
/// ```
- /// # use hashql_core::{id::{IdArray, Id as _}, newtype};
+ /// # use hashql_core::id::{IdArray, Id as _, newtype};
/// # newtype!(struct SlotId(u32 is 0..=2));
/// let array = IdArray::::from_raw(["a", "b", "c"]);
/// let pairs: Vec<_> = array.into_iter_enumerated().collect();
diff --git a/libs/@local/hashql/core/src/id/bit_vec/finite.rs b/libs/@local/hashql/core/src/id/bit_vec/finite.rs
index b5d696f5cd6..2490383981b 100644
--- a/libs/@local/hashql/core/src/id/bit_vec/finite.rs
+++ b/libs/@local/hashql/core/src/id/bit_vec/finite.rs
@@ -379,15 +379,16 @@ impl ExactSizeIterator for FiniteBitIter {
#[cfg(test)]
mod tests {
#![expect(clippy::min_ident_chars)]
- use crate::{
- id::{
- Id as _,
- bit_vec::{BitRelations as _, FiniteBitSet},
- },
+ use crate::id::{
+ Id as _,
+ bit_vec::{BitRelations as _, FiniteBitSet},
newtype,
};
- newtype!(struct TestId(u32 is 0..=127));
+ newtype!(
+ #[id(crate = crate)]
+ struct TestId(u32 is 0..=127)
+ );
#[test]
fn new_empty_creates_empty_set() {
diff --git a/libs/@local/hashql/core/src/id/bit_vec/matrix/tests.rs b/libs/@local/hashql/core/src/id/bit_vec/matrix/tests.rs
index 8ed7b38f871..3780c5bdab2 100644
--- a/libs/@local/hashql/core/src/id/bit_vec/matrix/tests.rs
+++ b/libs/@local/hashql/core/src/id/bit_vec/matrix/tests.rs
@@ -1,10 +1,10 @@
use super::{BitMatrix, RowRef, SparseBitMatrix};
-use crate::{
- id::{Id as _, bit_vec::DenseBitSet},
- newtype,
-};
+use crate::id::{Id as _, bit_vec::DenseBitSet, newtype};
-newtype!(struct TestId(usize is 0..=usize::MAX));
+newtype!(
+ #[id(crate = crate)]
+ struct TestId(u32 is 0..=u32::MAX)
+);
fn id(index: usize) -> TestId {
TestId::from_usize(index)
diff --git a/libs/@local/hashql/core/src/id/bit_vec/tests.rs b/libs/@local/hashql/core/src/id/bit_vec/tests.rs
index 35485d167d0..9175da08830 100644
--- a/libs/@local/hashql/core/src/id/bit_vec/tests.rs
+++ b/libs/@local/hashql/core/src/id/bit_vec/tests.rs
@@ -24,15 +24,16 @@ use alloc::rc::Rc;
use core::{marker::PhantomData, ops::RangeBounds};
use super::GrowableBitSet;
-use crate::{
- id::{
- Id as _,
- bit_vec::{BitRelations as _, Chunk, ChunkedBitSet, DenseBitSet, WORD_BITS},
- },
+use crate::id::{
+ Id as _,
+ bit_vec::{BitRelations as _, Chunk, ChunkedBitSet, DenseBitSet, WORD_BITS},
newtype,
};
-newtype!(struct TestId(usize is 0..=usize::MAX));
+newtype!(
+ #[id(crate = crate)]
+ struct TestId(u32 is 0..=u32::MAX)
+);
#[test]
fn new_filled() {
diff --git a/libs/@local/hashql/core/src/id/mod.rs b/libs/@local/hashql/core/src/id/mod.rs
index 02efc9a8871..3a500e8429a 100644
--- a/libs/@local/hashql/core/src/id/mod.rs
+++ b/libs/@local/hashql/core/src/id/mod.rs
@@ -13,6 +13,7 @@ use core::{
};
use ::core::sync::atomic;
+pub use hashql_macros::{Id, define_id as newtype};
pub use self::{
array::IdArray, index::IntoSliceIndex, slice::IdSlice, union_find::IdUnionFind, vec::IdVec,
@@ -165,7 +166,7 @@ pub trait Id:
/// # Examples
///
/// ```
-/// # use hashql_core::{id::{HasId, Id}, newtype};
+/// # use hashql_core::id::{HasId, Id, newtype};
/// # newtype!(struct UserId(u32 is 0..=100));
/// struct User {
/// id: UserId,
@@ -209,262 +210,6 @@ where
}
}
-/// Creates a new ID type with a specified valid range.
-///
-/// This uses the experimental pattern type syntax to define the minimum and maximum values.
-///
-/// # Syntax
-/// ```
-/// hashql_core::id::newtype!(pub struct NodeId(u32 is 0..=0xFFFF_FF00));
-/// ```
-///
-/// This creates a newtype wrapper around [`u32`] with the Id trait fully implemented.
-///
-/// # Optional Attributes
-///
-/// - `#[steppable]` - Implements `core::iter::Step` for the ID type, enabling range iteration
-///
-/// ```
-/// # #![feature(step_trait)]
-/// hashql_core::id::newtype!(
-/// #[steppable]
-/// pub struct NodeId(u32 is 0..=100)
-/// );
-/// ```
-#[macro_export]
-macro_rules! newtype {
- (@internal in_bounds; $value:ident, $type:ty, $min:literal, $max:expr) => {
- $value >= ($min as $type) && $value <= ($max as $type)
- };
-
- (@internal error; $value:ident, $min:literal, $max:expr) => {
- concat!("ID value must be between ", stringify!($min), " and ", stringify!($max))
- };
-
- ($(#[$($attr:tt)*])* $vis:vis struct $name:ident($type:ident is $min:literal..=$max:expr)) => {
- $crate::id::newtype!(@parse_attrs [] [] [] ; $(#[$($attr)*])* ; $vis struct $name($type is $min..=$max));
- };
-
- (@parse_attrs [$($other:tt)*] [$($step:tt)*] [$($display:tt)*]; #[steppable] $(#[$($rest:tt)*])* ; $($tail:tt)*) => {
- $crate::id::newtype!(@parse_attrs [$($other)*] [$($step)* steppable] [$($display)*] ; $(#[$($rest)*])* ; $($tail)*);
- };
-
- (@parse_attrs [$($other:tt)*] [$($step:tt)*] [$($display:tt)*]; #[display = $display_expr:expr] $(#[$($rest:tt)*])* ; $($tail:tt)*) => {
- $crate::id::newtype!(@parse_attrs [$($other)*] [$($step)*] [$($display)* display = $display_expr] ; $(#[$($rest)*])* ; $($tail)*);
- };
-
- (@parse_attrs [$($other:tt)*] [$($step:tt)*] [$($display:tt)*]; #[no_display] $(#[$($rest:tt)*])* ; $($tail:tt)*) => {
- $crate::id::newtype!(@parse_attrs [$($other)*] [$($step)*] [$($display)* no_display] ; $(#[$($rest)*])* ; $($tail)*);
- };
-
- (@parse_attrs [$($other:tt)*] [$($step:tt)*] [$($display:tt)*]; #[$attr:meta] $(#[$($rest:tt)*])* ; $($tail:tt)*) => {
- $crate::id::newtype!(@parse_attrs [$($other)* #[$attr]] [$($step)*] [$($display)*] ; $(#[$($rest)*])* ; $($tail)*);
- };
-
- (@parse_attrs [$($other:tt)*] [$($step:tt)*] [$($display:tt)*]; ; $($tail:tt)*) => {
- $crate::id::newtype!(@impl [$($other)*] [$($step)*] [$($display)*] $($tail)*);
- };
-
- // Implementation
- (@impl [$(#[$attr:meta])*] [$($step:tt)*] [$($display:tt)*] $vis:vis struct $name:ident($type:ident is $min:literal..=$max:expr)) => {
- $(#[$attr])*
- #[derive(Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
- $vis struct $name($type);
-
- #[expect(clippy::allow_attributes)]
- #[allow(dead_code, clippy::checked_conversions)]
- impl $name {
- /// Creates a new ID with the given value.
- ///
- /// # Panics
- ///
- /// When value is outside the valid range of $min..$max.
- #[must_use]
- $vis const fn new(value: $type) -> Self {
- assert!(
- $crate::id::newtype!(@internal in_bounds; value, $type, $min, $max),
- $crate::id::newtype!(@internal error; value, $min, $max)
- );
-
- Self(value)
- }
- }
-
- #[automatically_derived]
- #[expect(clippy::allow_attributes, reason = "automatically generated")]
- #[allow(clippy::cast_possible_truncation, clippy::cast_lossless, clippy::checked_conversions)]
- impl $crate::id::Id for $name {
- const MIN: Self = Self($min);
- const MAX: Self = Self($max);
-
- // fast path that does not go through the default implementation
- fn from_u32(value: u32) -> Self {
- assert!(
- $crate::id::newtype!(@internal in_bounds; value, u32, $min, $max),
- $crate::id::newtype!(@internal error; value, $min, $max)
- );
-
- Self(value as $type)
- }
-
- fn from_u64(value: u64) -> Self {
- assert!(
- $crate::id::newtype!(@internal in_bounds; value, u64, $min, $max),
- $crate::id::newtype!(@internal error; value, $min, $max)
- );
-
- Self(value as $type)
- }
-
- fn from_usize(value: usize) -> Self {
- assert!(
- $crate::id::newtype!(@internal in_bounds; value, usize, $min, $max),
- $crate::id::newtype!(@internal error; value, $min, $max)
- );
-
- Self(value as $type)
- }
-
- #[inline]
- fn as_u32(self) -> u32 {
- self.0 as u32
- }
-
- #[inline]
- fn as_u64(self) -> u64 {
- self.0 as u64
- }
-
- #[inline]
- fn as_usize(self) -> usize {
- self.0 as usize
- }
-
- #[inline]
- fn prev(self) -> ::core::option::Option {
- if self.0 == $min {
- None
- } else {
- Some(Self(self.0 - 1))
- }
- }
- }
-
- #[expect(clippy::allow_attributes, reason = "automatically generated")]
- #[allow(clippy::cast_possible_truncation, clippy::cast_lossless, clippy::checked_conversions)]
- impl ::core::convert::TryFrom for $name {
- type Error = $crate::id::IdError;
-
- fn try_from(value: u32) -> ::core::result::Result {
- if $crate::id::newtype!(@internal in_bounds; value, u32, $min, $max) {
- Ok(Self(value as $type))
- } else {
- Err($crate::id::IdError::OutOfRange {
- value: u64::from(value),
- min: $min as u64,
- max: $max as u64,
- })
- }
- }
- }
-
- #[expect(clippy::allow_attributes, reason = "automatically generated")]
- #[allow(clippy::cast_possible_truncation, clippy::cast_lossless, clippy::checked_conversions)]
- impl ::core::convert::TryFrom for $name {
- type Error = $crate::id::IdError;
-
- fn try_from(value: u64) -> ::core::result::Result {
- if $crate::id::newtype!(@internal in_bounds; value, u64, $min, $max) {
- Ok(Self(value as $type))
- } else {
- Err($crate::id::IdError::OutOfRange {
- value,
- min: $min as u64,
- max: $max as u64,
- })
- }
- }
- }
-
- #[expect(clippy::allow_attributes, reason = "automatically generated")]
- #[allow(clippy::cast_possible_truncation, clippy::cast_lossless, clippy::checked_conversions)]
- impl ::core::convert::TryFrom for $name {
- type Error = $crate::id::IdError;
-
- fn try_from(value: usize) -> ::core::result::Result {
- if $crate::id::newtype!(@internal in_bounds; value, usize, $min, $max) {
- Ok(Self(value as $type))
- } else {
- Err($crate::id::IdError::OutOfRange {
- value: value as u64,
- min: $min as u64,
- max: $max as u64,
- })
- }
- }
- }
-
- impl $crate::id::HasId for $name {
- type Id = $name;
-
- fn id(&self) -> Self::Id {
- *self
- }
- }
-
- $crate::id::newtype!(@maybe_display $name ; $($display)*);
- $crate::id::newtype!(@maybe_step $name ; $($step)*);
- };
-
- // Generate Step implementation if steppable was specified
- (@maybe_step $name:ident ; steppable) => {
- impl ::core::iter::Step for $name {
- #[inline]
- fn steps_between(start: &Self, end: &Self) -> (usize, Option) {
- ::steps_between(
- &$crate::id::Id::as_usize(*start),
- &$crate::id::Id::as_usize(*end),
- )
- }
-
- #[inline]
- fn forward_checked(start: Self, count: usize) -> Option {
- $crate::id::Id::as_usize(start)
- .checked_add(count)
- .map($crate::id::Id::from_usize)
- }
-
- #[inline]
- fn backward_checked(start: Self, count: usize) -> Option {
- $crate::id::Id::as_usize(start)
- .checked_sub(count)
- .map($crate::id::Id::from_usize)
- }
- }
- };
-
- // No Step implementation if steppable was not specified
- (@maybe_step $name:ident ; ) => {};
-
- (@maybe_display $name:ident ; no_display) => {};
-
- (@maybe_display $name:ident ; display = $display:expr) => {
- impl ::core::fmt::Display for $name {
- fn fmt(&self, fmt: &mut ::core::fmt::Formatter<'_>) -> ::core::fmt::Result {
- fmt.write_fmt(format_args!($display, self.0))
- }
- }
- };
-
- (@maybe_display $name:ident ; ) => {
- impl ::core::fmt::Display for $name {
- fn fmt(&self, fmt: &mut ::core::fmt::Formatter<'_>) -> ::core::fmt::Result {
- ::core::fmt::Display::fmt(&self.0, fmt)
- }
- }
- }
-}
-
/// Thread-safe ID generator that produces unique IDs.
///
/// Uses an atomic counter to generate sequential IDs, making it safe to use
@@ -474,7 +219,7 @@ macro_rules! newtype {
/// # Examples
///
/// ```
-/// # use hashql_core::{id::IdProducer, newtype};
+/// # use hashql_core::id::{IdProducer, newtype};
/// # newtype!(struct NodeId(u32 is 0..=1000));
/// let producer = IdProducer::::new();
/// let id1 = producer.next();
@@ -535,7 +280,7 @@ impl Default for IdProducer {
/// # Examples
///
/// ```
-/// # use hashql_core::{id::IdCounter, newtype};
+/// # use hashql_core::id::{IdCounter, newtype};
/// # newtype!(struct NodeId(u32 is 0..=1000));
/// let mut counter = IdCounter::::new();
/// let id1 = counter.next();
@@ -619,7 +364,7 @@ where
/// # Examples
///
/// ```
-/// # use hashql_core::{newtype, newtype_producer};
+/// # use hashql_core::id::{newtype, newtype_producer};
/// # newtype!(struct NodeId(u32 is 0..=1000));
/// newtype_producer!(pub struct NodeIdProducer(NodeId));
/// ```
@@ -635,7 +380,7 @@ macro_rules! newtype_producer {
/// # Examples
///
/// ```
-/// # use hashql_core::{newtype, newtype_counter};
+/// # use hashql_core::id::{newtype, newtype_counter};
/// # newtype!(struct NodeId(u32 is 0..=1000));
/// newtype_counter!(pub struct NodeIdCounter(NodeId));
/// ```
@@ -668,7 +413,7 @@ macro_rules! newtype_counter {
/// ```
/// # #![feature(allocator_api, macro_metavar_expr_concat)]
/// # extern crate alloc;
-/// # use hashql_core::{newtype, newtype_collections};
+/// # use hashql_core::id::{newtype, newtype_collections};
/// # newtype!(struct NodeId(u32 is 0..=1000));
/// newtype_collections!(pub type Node* from NodeId);
/// // Creates: NodeSlice, NodeVec, NodeUnionFind, NodeSet, NodeMap, etc.
@@ -688,7 +433,6 @@ macro_rules! newtype_collections {
};
}
-pub use newtype;
pub use newtype_collections;
pub use newtype_counter;
pub use newtype_producer;
diff --git a/libs/@local/hashql/core/src/id/slice.rs b/libs/@local/hashql/core/src/id/slice.rs
index 2cbefb39a68..2f560b7cf5f 100644
--- a/libs/@local/hashql/core/src/id/slice.rs
+++ b/libs/@local/hashql/core/src/id/slice.rs
@@ -20,7 +20,7 @@ use super::{Id, index::IntoSliceIndex, vec::IdVec};
/// # Examples
///
/// ```
-/// # use hashql_core::{id::{IdSlice, Id as _}, newtype};
+/// # use hashql_core::id::{IdSlice, Id as _, newtype};
/// # newtype!(struct UserId(u32 is 0..=0xFFFF_FF00));
/// let data = [10, 20, 30];
/// let slice = IdSlice::::from_raw(&data);
@@ -303,7 +303,7 @@ where
/// # Examples
///
/// ```
- /// # use hashql_core::{id::{IdVec, Id as _}, newtype};
+ /// # use hashql_core::id::{IdVec, Id as _, newtype};
/// # newtype!(struct MyId(u32 is 0..=0xFFFF_FF00));
/// let mut vec = IdVec::>::new();
/// vec.insert(MyId::from_usize(0), "hello".to_string());
@@ -320,7 +320,7 @@ where
/// # Examples
///
/// ```
- /// # use hashql_core::{id::{IdVec, Id as _}, newtype};
+ /// # use hashql_core::id::{IdVec, Id as _, newtype};
/// # newtype!(struct MyId(u32 is 0..=0xFFFF_FF00));
/// let mut vec = IdVec::>::new();
/// vec.insert(MyId::from_usize(0), "hello".to_string());
@@ -338,7 +338,7 @@ where
/// # Examples
///
/// ```
- /// # use hashql_core::{id::{IdVec, Id as _}, newtype};
+ /// # use hashql_core::id::{IdVec, Id as _, newtype};
/// # newtype!(struct MyId(u32 is 0..=0xFFFF_FF00));
/// let mut vec = IdVec::>::new();
/// vec.insert(MyId::from_usize(0), "hello".to_string());
@@ -358,7 +358,7 @@ where
/// # Examples
///
/// ```
- /// # use hashql_core::{id::{IdVec, Id as _}, newtype};
+ /// # use hashql_core::id::{IdVec, Id as _, newtype};
/// # newtype!(struct MyId(u32 is 0..=0xFFFF_FF00));
/// let mut vec = IdVec::>::new();
/// vec.insert(MyId::from_usize(0), "hello".to_string());
@@ -466,9 +466,12 @@ mod tests {
use core::mem::MaybeUninit;
use super::IdSlice;
- use crate::{id::Id as _, newtype};
+ use crate::id::Id as _;
- newtype!(struct TestId(u32 is 0..=0xFFFF_FF00));
+ hashql_macros::define_id! {
+ #[id(crate = crate)]
+ struct TestId(u32 is 0..=0xFFFF_FF00)
+ }
#[test]
fn from_raw_indexing() {
diff --git a/libs/@local/hashql/core/src/id/union_find.rs b/libs/@local/hashql/core/src/id/union_find.rs
index d3a451d0c6a..5fd5d592a51 100644
--- a/libs/@local/hashql/core/src/id/union_find.rs
+++ b/libs/@local/hashql/core/src/id/union_find.rs
@@ -382,7 +382,10 @@ mod tests {
use super::*;
// Define a test ID type
- crate::id::newtype!(struct TestId(u32 is 0..=1000));
+ crate::id::newtype!(
+ #[id(crate = crate)]
+ struct TestId(u32 is 0..=1000)
+ );
#[test]
fn path_compression() {
diff --git a/libs/@local/hashql/core/src/id/vec.rs b/libs/@local/hashql/core/src/id/vec.rs
index 221d06fe25c..5bed2f54ad0 100644
--- a/libs/@local/hashql/core/src/id/vec.rs
+++ b/libs/@local/hashql/core/src/id/vec.rs
@@ -23,7 +23,7 @@ use crate::heap::{FromIteratorIn, TryCloneIn};
/// # Examples
///
/// ```
-/// # use hashql_core::{id::{IdVec, Id as _}, newtype};
+/// # use hashql_core::id::{IdVec, Id as _, newtype};
/// # newtype!(struct UserId(u32 is 0..=0xFFFF_FF00));
/// let mut users = IdVec::::new();
/// let user_id = users.push("Alice".to_string());
@@ -75,7 +75,7 @@ where
/// # Examples
///
/// ```
- /// # use hashql_core::{id::{IdVec, Id as _}, newtype};
+ /// # use hashql_core::id::{IdVec, Id as _, newtype};
/// # newtype!(struct MyId(u32 is 0..=100));
/// let vec = IdVec::::from_elem(42, 5);
/// assert_eq!(vec.len(), 5);
@@ -165,7 +165,7 @@ where
///
/// ```
/// # #![feature(allocator_api)]
- /// # use hashql_core::{id::{IdVec, Id as _}, newtype};
+ /// # use hashql_core::id::{IdVec, Id as _, newtype};
/// # newtype!(struct MyId(u32 is 0..=100));
/// use std::alloc::Global;
/// let vec = IdVec::::from_elem_in(42, 5, Global);
@@ -188,7 +188,7 @@ where
/// # Examples
///
/// ```
- /// # use hashql_core::{id::{IdVec, Id as _}, newtype};
+ /// # use hashql_core::id::{IdVec, Id as _, newtype};
/// # newtype!(struct MyId(u32 is 0..=100));
/// let domain = IdVec::::from_elem("x".to_string(), 3);
/// let vec = IdVec::::from_domain(0, &domain);
@@ -288,7 +288,7 @@ where
/// # Examples
///
/// ```
- /// # use hashql_core::{id::{IdVec, Id as _}, newtype};
+ /// # use hashql_core::id::{IdVec, Id as _, newtype};
/// # newtype!(struct MyId(u32 is 0..=0xFFFF_FF00));
/// let mut vec = IdVec::::new();
/// let id = vec.push("hello".to_string());
@@ -308,7 +308,7 @@ where
/// # Examples
///
/// ```
- /// # use hashql_core::{id::{IdVec, Id as _}, newtype};
+ /// # use hashql_core::id::{IdVec, Id as _, newtype};
/// # newtype!(struct MyId(u32 is 0..=100));
/// let mut vec = IdVec::::new();
/// let id = vec.push_with(|id| format!("item_{}", id.as_u32()));
@@ -353,7 +353,7 @@ where
/// # Examples
///
/// ```
- /// # use hashql_core::{id::{IdVec, Id as _}, newtype};
+ /// # use hashql_core::id::{IdVec, Id as _, newtype};
/// # newtype!(struct MyId(u32 is 0..=100));
/// let mut vec = IdVec::::new();
/// let value = vec.fill_until(MyId::new(5), || 0);
@@ -442,7 +442,7 @@ where
/// # Examples
///
/// ```
- /// # use hashql_core::{id::{IdVec, Id as _}, newtype};
+ /// # use hashql_core::id::{IdVec, Id as _, newtype};
/// # newtype!(struct MyId(u32 is 0..=0xFFFF_FF00));
/// let mut vec = IdVec::>::new();
/// vec.insert(MyId::from_usize(5), "hello".to_string());
@@ -463,7 +463,7 @@ where
/// # Examples
///
/// ```
- /// # use hashql_core::{id::{IdVec, Id as _}, newtype};
+ /// # use hashql_core::id::{IdVec, Id as _, newtype};
/// # newtype!(struct MyId(u32 is 0..=100));
/// let mut vec = IdVec::>::new();
/// let value = vec.get_or_insert_with(MyId::new(2), || "hello".to_string());
diff --git a/libs/@local/hashql/core/src/intern/map.rs b/libs/@local/hashql/core/src/intern/map.rs
index 463a6a8212c..919557a0c15 100644
--- a/libs/@local/hashql/core/src/intern/map.rs
+++ b/libs/@local/hashql/core/src/intern/map.rs
@@ -149,7 +149,7 @@ where
/// # Examples
///
/// ```
-/// # use hashql_core::{heap::Heap, intern::{InternMap, Decompose, Interned}, id::{HasId, Id}, newtype};
+/// # use hashql_core::{heap::Heap, intern::{InternMap, Decompose, Interned}, id::{HasId, Id, newtype}};
/// # newtype!(struct ValueId(u32 is 0..=0xFFFF_FF00));
/// # #[derive(Debug, PartialEq, Eq, Hash)]
/// # struct Value {
@@ -239,7 +239,7 @@ where
/// # Examples
///
/// ```
- /// # use hashql_core::{heap::Heap, intern::{InternMap, Decompose, Interned}, id::{HasId, Id}, newtype};
+ /// # use hashql_core::{heap::Heap, intern::{InternMap, Decompose, Interned}, id::{HasId, Id, newtype}};
/// # newtype!(struct TypeId(u32 is 0..=0xFFFF_FF00));
/// # #[derive(Debug, PartialEq, Eq, Hash)]
/// # struct TypeInfo {
@@ -270,7 +270,7 @@ where
/// Use case for compiler type systems:
///
/// ```
- /// # use hashql_core::{heap::Heap, intern::{InternMap, Decompose, Interned}, id::{HasId, Id}, newtype};
+ /// # use hashql_core::{heap::Heap, intern::{InternMap, Decompose, Interned}, id::{HasId, Id, newtype}};
/// # newtype!(struct TypeId(u32 is 0..=0xFFFF_FF00));
/// # #[derive(Debug, Copy, Clone, PartialEq, Eq, Hash)]
/// # enum PrimitiveType { I32, I64, F32, F64, Bool, String }
@@ -392,7 +392,7 @@ where
/// # Examples
///
/// ```
- /// # use hashql_core::{heap::Heap, intern::{InternMap, Decompose, Interned}, id::{HasId, Id}, newtype};
+ /// # use hashql_core::{heap::Heap, intern::{InternMap, Decompose, Interned}, id::{HasId, Id, newtype}};
/// # newtype!(struct ValueId(u32 is 0..=0xFFFF_FF00));
/// # #[derive(Debug, PartialEq, Eq, Hash)]
/// # struct Value {
@@ -434,7 +434,7 @@ where
/// # Examples
///
/// ```
- /// # use hashql_core::{heap::Heap, intern::{InternMap, Decompose, Interned, Provisioned}, id::{HasId, Id}, newtype};
+ /// # use hashql_core::{heap::Heap, intern::{InternMap, Decompose, Interned, Provisioned}, id::{HasId, Id, newtype}};
/// # newtype!(struct ValueId(u32 is 0..=0xFFFF_FF00));
/// # #[derive(Debug, PartialEq, Eq, Hash)]
/// # struct Value {
@@ -523,7 +523,7 @@ where
/// Creating a self-referential linked list node:
///
/// ```
- /// # use hashql_core::{heap::Heap, intern::{InternMap, Decompose, Interned, Provisioned}, id::{HasId, Id}, newtype};
+ /// # use hashql_core::{heap::Heap, intern::{InternMap, Decompose, Interned, Provisioned}, id::{HasId, Id, newtype}};
/// # newtype!(struct NodeId(u32 is 0..=0xFFFF_FF00));
/// # #[derive(Debug, Copy, Clone, PartialEq, Eq, Hash)]
/// # struct PartialNode { value: i32, next: Option }
@@ -583,7 +583,7 @@ where
/// # Examples
///
/// ```
- /// # use hashql_core::{heap::Heap, intern::{InternMap, Decompose, Interned}, id::{HasId, Id}, newtype};
+ /// # use hashql_core::{heap::Heap, intern::{InternMap, Decompose, Interned}, id::{HasId, Id, newtype}};
/// # newtype!(struct ValueId(u32 is 0..=0xFFFF_FF00));
/// # #[derive(Debug, PartialEq, Eq, Hash)]
/// # struct Value {
@@ -628,7 +628,7 @@ where
/// # Examples
///
/// ```
- /// # use hashql_core::{heap::Heap, intern::{InternMap, Decompose, Interned}, id::{HasId, Id}, newtype};
+ /// # use hashql_core::{heap::Heap, intern::{InternMap, Decompose, Interned}, id::{HasId, Id, newtype}};
/// # newtype!(struct ValueId(u32 is 0..=0xFFFF_FF00));
/// # #[derive(Debug, PartialEq, Eq, Hash)]
/// # struct Value {
@@ -672,7 +672,7 @@ where
/// # Examples
///
/// ```
- /// # use hashql_core::{heap::Heap, intern::{InternMap, Decompose, Interned}, id::{HasId, Id}, newtype};
+ /// # use hashql_core::{heap::Heap, intern::{InternMap, Decompose, Interned}, id::{HasId, Id, newtype}};
/// # newtype!(struct ValueId(u32 is 0..=0xFFFF_FF00));
/// # #[derive(Debug, PartialEq, Eq, Hash)]
/// # struct Value {
@@ -726,7 +726,7 @@ where
/// # Examples
///
/// ```
- /// # use hashql_core::{heap::Heap, intern::{InternMap, Decompose, Interned}, id::{HasId, Id}, newtype};
+ /// # use hashql_core::{heap::Heap, intern::{InternMap, Decompose, Interned}, id::{HasId, Id, newtype}};
/// # newtype!(struct ValueId(u32 is 0..=0xFFFF_FF00));
/// # #[derive(Debug, PartialEq, Eq, Hash)]
/// # struct Value {
@@ -771,12 +771,14 @@ mod tests {
use crate::{
heap::Heap,
- id::HasId,
+ id::{HasId, newtype},
intern::{Decompose, InternMap, Interned},
- newtype,
};
- newtype!(struct TaggedId(u32 is 0..=0xFFFF_FF00));
+ newtype!(
+ #[id(crate = crate)]
+ struct TaggedId(u32 is 0..=0xFFFF_FF00)
+ );
#[derive(Debug, PartialEq, Eq, Hash)]
struct TaggedValue {
@@ -803,7 +805,10 @@ mod tests {
}
}
- newtype!(struct ListId(u32 is 0..=0xFFFF_FF00));
+ newtype!(
+ #[id(crate = crate)]
+ struct ListId(u32 is 0..=0xFFFF_FF00)
+ );
// A recursive test type that can reference other TestNode instances by ID
#[derive(Debug, PartialEq, Eq, Hash)]
@@ -877,7 +882,7 @@ mod tests {
assert_eq!(retrieved, value);
// Look up a non-existent ID
- let non_existent = map.get(TaggedId(999));
+ let non_existent = map.get(TaggedId::new(999));
assert!(non_existent.is_none());
// Test the index method
diff --git a/libs/@local/hashql/core/src/lib.rs b/libs/@local/hashql/core/src/lib.rs
index 6527eac72a5..3599b4b8442 100644
--- a/libs/@local/hashql/core/src/lib.rs
+++ b/libs/@local/hashql/core/src/lib.rs
@@ -3,7 +3,6 @@
//! ## Workspace dependencies
#![cfg_attr(doc, doc = simple_mermaid::mermaid!("../docs/dependency-diagram.mmd"))]
#![expect(clippy::indexing_slicing)]
-#![recursion_limit = "256"]
#![feature(
// Language Features
arbitrary_self_types,
diff --git a/libs/@local/hashql/core/src/module/mod.rs b/libs/@local/hashql/core/src/module/mod.rs
index 7517079d1d1..f97b1617874 100644
--- a/libs/@local/hashql/core/src/module/mod.rs
+++ b/libs/@local/hashql/core/src/module/mod.rs
@@ -25,14 +25,16 @@ pub use self::{resolver::Reference, universe::Universe};
use crate::{
collections::{FastHashMap, FastHashSet},
heap::Heap,
- id::{HasId, Id as _},
+ id::{HasId, Id as _, newtype},
intern::{Decompose, InternMap, InternSet, Interned, Provisioned},
- newtype,
symbol::Symbol,
r#type::environment::Environment,
};
-newtype!(pub struct ModuleId(u32 is 0..=0xFFFF_FF00));
+newtype! {
+ #[id(crate = crate)]
+ pub struct ModuleId(u32 is 0..=0xFFFF_FF00)
+}
impl ModuleId {
pub const ROOT: Self = Self::MAX;
diff --git a/libs/@local/hashql/core/src/symbol/lookup.rs b/libs/@local/hashql/core/src/symbol/lookup.rs
index 61c6360b0b4..5b9abc41e28 100644
--- a/libs/@local/hashql/core/src/symbol/lookup.rs
+++ b/libs/@local/hashql/core/src/symbol/lookup.rs
@@ -45,7 +45,7 @@ enum SymbolLookupInner<'heap, I> {
/// # Examples
///
/// ```
-/// # use hashql_core::{heap::Heap, symbol::SymbolLookup, newtype, id::Id as _};
+/// # use hashql_core::{heap::Heap, symbol::SymbolLookup, id::{Id as _, newtype}};
/// # newtype!(struct MyId(u32 is 0..=0xFFFF_FF00));
/// # let mut heap = Heap::new();
/// # let symbol = heap.intern_symbol("example");
@@ -86,7 +86,7 @@ where
/// # Examples
///
/// ```
- /// # use hashql_core::{symbol::SymbolLookup, newtype};
+ /// # use hashql_core::{symbol::SymbolLookup, id::newtype};
/// # newtype!(struct MyId(u32 is 0..=0xFFFF_FF00));
/// let table = SymbolLookup::::dense();
/// // Insertions must be sequential: 0, 1, 2, ...
@@ -107,7 +107,7 @@ where
/// # Examples
///
/// ```
- /// # use hashql_core::{symbol::SymbolLookup, newtype};
+ /// # use hashql_core::{symbol::SymbolLookup, id::newtype};
/// # newtype!(struct MyId(u32 is 0..=0xFFFF_FF00));
/// let table = SymbolLookup::::gapped();
/// // Insertions can have gaps: 0, 5, 3, 10, ...
@@ -127,7 +127,7 @@ where
/// # Examples
///
/// ```
- /// # use hashql_core::{symbol::SymbolLookup, newtype};
+ /// # use hashql_core::{symbol::SymbolLookup, id::newtype};
/// # newtype!(struct MyId(u32 is 0..=0xFFFF_FF00));
/// let table = SymbolLookup::::sparse();
/// // Insertions can be in any order: 100, 5, 1000, ...
@@ -155,7 +155,7 @@ where
/// # Examples
///
/// ```
- /// # use hashql_core::{heap::Heap, symbol::SymbolLookup, newtype, id::Id as _};
+ /// # use hashql_core::{heap::Heap, symbol::SymbolLookup, id::{Id as _, newtype}};
/// # newtype!(struct MyId(u32 is 0..=0xFFFF_FF00));
/// # let mut heap = Heap::new();
/// # let symbol = heap.intern_symbol("example");
@@ -167,7 +167,7 @@ where
/// Non-sequential insertions will panic in dense tables:
///
/// ```should_panic
- /// # use hashql_core::{heap::Heap, symbol::SymbolLookup, newtype, id::Id as _};
+ /// # use hashql_core::{heap::Heap, symbol::SymbolLookup, id::{Id as _, newtype}};
/// # newtype!(struct MyId(u32 is 0..=0xFFFF_FF00));
/// # let mut heap = Heap::new();
/// # let symbol = heap.intern_symbol("example");
@@ -203,7 +203,7 @@ where
/// # Examples
///
/// ```
- /// # use hashql_core::{heap::Heap, symbol::SymbolLookup, newtype, id::Id as _};
+ /// # use hashql_core::{heap::Heap, symbol::SymbolLookup, id::{Id as _, newtype}};
/// # newtype!(struct MyId(u32 is 0..=0xFFFF_FF00));
/// # let mut heap = Heap::new();
/// # let symbol = heap.intern_symbol("example");
diff --git a/libs/@local/hashql/core/src/symbol/sym.rs b/libs/@local/hashql/core/src/symbol/sym.rs
index 578e7327372..23a675bb88b 100644
--- a/libs/@local/hashql/core/src/symbol/sym.rs
+++ b/libs/@local/hashql/core/src/symbol/sym.rs
@@ -1,166 +1,7 @@
#![expect(non_upper_case_globals, non_snake_case, clippy::min_ident_chars)]
-use super::{ConstantSymbol, Symbol};
+use super::Symbol;
-/// Generates pre-interned symbols available at compile time.
-///
-/// This macro produces three artifacts from a single symbol table definition:
-///
-/// 1. **`SYMBOLS`** - A static slice of string values for interner pre-population
-/// 2. **Symbol constants** - `Symbol<'static>` constants (e.g., `sym::foo`, `sym::symbol::plus`)
-/// 3. **`LOOKUP`** - A static slice mapping string values to their [`Repr`] for fast lookup
-///
-/// # Syntax
-///
-/// ```text
-/// symbols! {@table;
-/// // Simple symbol: name becomes both the constant and string value
-/// foo,
-///
-/// // Explicit string: use when string differs from identifier
-/// r#true: "true",
-/// input_exists: "$exists",
-///
-/// // Nested module: groups related symbols under a namespace
-/// symbol: {
-/// plus: "+",
-/// minus: "-",
-/// },
-/// }
-/// ```
-///
-/// Each symbol `name` or `name: "value"` generates:
-/// - A constant `name: Symbol<'static>` with auto-generated docs
-/// - A submodule `name` containing `CONST: ConstantSymbol` for pattern matching
-///
-/// Modules create nested namespaces, so `symbol::plus` becomes accessible as `sym::symbol::plus`.
-///
-/// # Internal Rules
-///
-/// The macro uses internal rules (prefixed with `@`) to process the token stream:
-///
-/// - **`@strings`** - Collects all string values into the `SYMBOLS` slice
-/// - **`@consts`** - Generates `Symbol` constants and companion modules with index tracking
-/// - **`@consts @cont`** - Continuation after processing a nested module to resume counting
-/// - **`@lookup`** - Builds the string-to-repr mapping table for runtime lookup
-/// - **`@path`** - Helper to construct module paths (reverses accumulated path segments)
-/// - **`@table`** - Entry point that dispatches to all three generators
-///
-/// Index tracking uses the `${count($count)}` metavariable to assign sequential indices.
-/// Each processed symbol appends `()` to the count accumulator, and `${count(...)}` returns
-/// the number of elements.
-///
-/// [`Repr`]: super::repr::Repr
-macro_rules! symbols {
- (@strings [$($acc:tt)*];) => {
- pub(crate) static SYMBOLS: &[&str] = &[
- $($acc),*
- ];
- };
- (@strings [$($acc:tt)*]; , $($rest:tt)*) => {
- symbols!(@strings [$($acc)*]; $($rest)*);
- };
- (@strings [$($acc:tt)*]; $module:ident : { $($inner:tt)* } $(, $($rest:tt)*)?) => {
- symbols!(@strings [$($acc)*]; $($inner)* $(, $($rest)*)?);
- };
- (@strings [$($acc:tt)*]; $name:ident : $value:literal $(, $($rest:tt)*)?) => {
- symbols!(@strings [$($acc)* $value]; $($($rest)*)?);
- };
- (@strings [$($acc:tt)*]; $name:ident $(, $($rest:tt)*)?) => {
- symbols!(@strings [$($acc)* (stringify!($name))]; $($($rest)*)?);
- };
-
- (@consts @cont [$($count:tt)*] [$($next:tt)*];) => {
- symbols!(@consts [$($count)*]; $($next)*);
- };
- (@consts @cont [$($count:tt)*] [$($next:tt)*]; , $($rest:tt)*) => {
- symbols!(@consts @cont [$($count)*] [$($next)*]; $($rest)*);
- };
- (@consts @cont [$($count:tt)*] [$($next:tt)*]; $name:ident : $value:literal $(, $($rest:tt)*)?) => {
- symbols!(@consts @cont [$($count)* ()] [$($next)*]; $($($rest)*)?);
- };
- (@consts @cont [$($count:tt)*] [$($next:tt)*]; $module:ident : { $($inner:tt)* } $(, $($rest:tt)*)?) => {
- compile_error!("nested modules in modules are not supported");
- };
- (@consts @cont [$($count:tt)*] [$($next:tt)*]; $name:ident $(, $($rest:tt)*)?) => {
- symbols!(@consts @cont [$($count)* ()] [$($next)*]; $($($rest)*)?);
- };
-
- (@consts [$($count:tt)*];) => {};
- (@consts [$($count:tt)*]; , $($rest:tt)*) => {
- symbols!(@consts [$($count)*]; $($rest)*);
- };
- (@consts [$($count:tt)*]; $name:ident : $value:literal $(, $($rest:tt)*)?) => {
- const _: () = { assert!(SYMBOLS[${count($count)}] == $value) };
- #[doc = concat!("The symbol `", $value, "`")]
- pub const $name: Symbol<'static> = Symbol::from_constant($name::CONST);
-
- pub mod $name {
- use super::*;
-
- pub const CONST: ConstantSymbol = ConstantSymbol::new_unchecked(${count($count)});
- }
-
- symbols!(@consts [$($count)* ()]; $($($rest)*)?);
- };
- (@consts [$($count:tt)*]; $module:ident : { $($inner:tt)* } $(, $($rest:tt)*)?) => {
- pub mod $module {
- use super::*;
-
- symbols!(@consts [$($count)*]; $($inner)*);
- }
-
- symbols!(@consts @cont [$($count)*] [$($($rest)*)?]; $($inner)*);
- };
- (@consts [$($count:tt)*]; $name:ident $(, $($rest:tt)*)?) => {
- const _: () = { assert!(SYMBOLS[${count($count)}] == stringify!($name)) };
- #[doc = concat!("The symbol `", stringify!($name), "`")]
- pub const $name: Symbol<'static> = Symbol::from_constant($name::CONST);
-
- pub mod $name {
- use super::*;
-
- pub const CONST: ConstantSymbol = ConstantSymbol::new_unchecked(${count($count)});
- }
-
- symbols!(@consts [$($count)* ()]; $($($rest)*)?);
- };
-
- (@path [] [$($path:ident)*];) => {
- $($path)::*
- };
- (@path [$next:tt $($rest:tt)*] [$($path:tt)*];) => {
- symbols!(@path [$($rest)*] [$next $($path)*];)
- };
-
- (@lookup [$(, $arm:expr => $value:expr)*] [$($path:tt),*];) => {
- pub(crate) static LOOKUP: &[(&'static str, super::repr::Repr)] = &[
- $(($arm, $value.into_repr())),*
- ];
- };
- (@lookup [$($arms:tt)*] [$tail:tt $(, $path:tt)*]; | $($rest:tt)*) => {
- symbols!(@lookup [$($arms)*] [$($path),*]; $($rest)*);
- };
- (@lookup [$($arms:tt)*] [$($path:tt),*]; , $($rest:tt)*) => {
- symbols!(@lookup [$($arms)*] [$($path),*]; $($rest)*);
- };
- (@lookup [$($arms:tt)*] [$($path:tt),*]; $name:ident : $value:literal $(, $($rest:tt)*)?) => {
- symbols!(@lookup [$($arms)*, $value => symbols!(@path [$name $($path)*] [];)] [$($path),*]; $($($rest)*)?);
- };
- (@lookup [$($arms:tt)*] [$($path:tt),*]; $module:ident : { $($inner:tt)* } $(, $($rest:tt)*)?) => {
- symbols!(@lookup [$($arms)*] [$module $(, $path)*]; $($inner)* ,| $($($rest)*)?);
- };
- (@lookup [$($arms:tt)*] [$($path:tt),*]; $name:ident $(, $($rest:tt)*)?) => {
- symbols!(@lookup [$($arms)*, stringify!($name) => symbols!(@path [$name $($path)*] [];)] [$($path),*]; $($($rest)*)?);
- };
-
- (@table; $($items:tt)*) => {
- symbols!(@strings []; $($items)*);
- symbols!(@consts []; $($items)*);
- symbols!(@lookup [] [self]; $($items)*);
- };
-}
-
-symbols! {@table;
+hashql_macros::define_symbols! {
// [tidy] sort alphabetically start
access,
add,
diff --git a/libs/@local/hashql/core/src/type/kind/generic/mod.rs b/libs/@local/hashql/core/src/type/kind/generic/mod.rs
index bfa57660eba..d4a452ddbd3 100644
--- a/libs/@local/hashql/core/src/type/kind/generic/mod.rs
+++ b/libs/@local/hashql/core/src/type/kind/generic/mod.rs
@@ -14,8 +14,9 @@ pub use self::{
use super::TypeKind;
use crate::{
collections::{SmallVec, TinyVec},
+ id::newtype,
intern::Interned,
- newtype, newtype_collections, newtype_producer,
+ newtype_collections, newtype_producer,
pretty::display::DisplayBuilder,
span::SpanId,
symbol::{Ident, Symbol},
@@ -32,6 +33,7 @@ use crate::{
};
newtype!(
+ #[id(crate = crate)]
pub struct GenericArgumentId(u32 is 0..=0xFFFF_FF00)
);
diff --git a/libs/@local/hashql/core/src/type/kind/infer.rs b/libs/@local/hashql/core/src/type/kind/infer.rs
index 4f41bfe4c1d..c42834a3bd1 100644
--- a/libs/@local/hashql/core/src/type/kind/infer.rs
+++ b/libs/@local/hashql/core/src/type/kind/infer.rs
@@ -1,6 +1,7 @@
-use crate::{newtype, newtype_producer};
+use crate::{id::newtype, newtype_producer};
newtype!(
+ #[id(crate = crate)]
pub struct HoleId(u32 is 0..=0xFFFF_FF00)
);
diff --git a/libs/@local/hashql/core/src/type/mod.rs b/libs/@local/hashql/core/src/type/mod.rs
index b408080e103..5a629e933d4 100644
--- a/libs/@local/hashql/core/src/type/mod.rs
+++ b/libs/@local/hashql/core/src/type/mod.rs
@@ -36,6 +36,7 @@ id::newtype!(
/// The value space is restricted to `0..=0xFFFF_FF00`, reserving the last 256 for niches.
/// As real pattern types are an experimental feature in Rust, these can currently only be
/// used by directly modifying and accessing the `TypeId`'s internal value.
+ #[id(crate = crate)]
pub struct TypeId(u32 is 0..=0xFFFF_FF00)
);
@@ -47,7 +48,7 @@ impl TypeId {
/// The uniqueness constraint is not enforced by the type system, but rather just a statistical
/// improbability, considering that 4.294.967.040 types would need to be generated, for a
/// collision to occur.
- pub const PLACEHOLDER: Self = Self(0xFFFF_FF00);
+ pub const PLACEHOLDER: Self = Self::new(0xFFFF_FF00);
}
id::newtype_collections!(pub type TypeId* from TypeId);
diff --git a/libs/@local/hashql/eval/docs/dependency-diagram.mmd b/libs/@local/hashql/eval/docs/dependency-diagram.mmd
index 81a250cb31f..0b1028cfe78 100644
--- a/libs/@local/hashql/eval/docs/dependency-diagram.mmd
+++ b/libs/@local/hashql/eval/docs/dependency-diagram.mmd
@@ -27,50 +27,52 @@ graph TD
16[hashql-eval]
class 16 root
17[hashql-hir]
- 18[hashql-mir]
- 19[hashql-syntax-jexpr]
- 20[hash-temporal-client]
- 21[darwin-kperf]
- 22[darwin-kperf-criterion]
- 23[darwin-kperf-events]
- 24[darwin-kperf-sys]
- 25[error-stack]
- 26[hash-graph-benches]
- 27[hash-graph-test-data]
+ 18[hashql-macros]
+ 19[hashql-mir]
+ 20[hashql-syntax-jexpr]
+ 21[hash-temporal-client]
+ 22[darwin-kperf]
+ 23[darwin-kperf-criterion]
+ 24[darwin-kperf-events]
+ 25[darwin-kperf-sys]
+ 26[error-stack]
+ 27[hash-graph-benches]
+ 28[hash-graph-test-data]
0 --> 8
1 --> 7
- 1 -.-> 27
+ 1 -.-> 28
2 -.-> 3
2 --> 11
4 --> 16
- 4 --> 19
+ 4 --> 20
5 --> 1
6 --> 5
6 --> 9
- 6 --> 20
+ 6 --> 21
7 --> 2
8 --> 4
- 9 -.-> 27
+ 9 -.-> 28
11 -.-> 10
11 --> 10
- 11 --> 25
+ 11 --> 26
12 -.-> 13
13 --> 16
- 13 --> 18
13 --> 19
- 13 --> 25
+ 13 --> 20
+ 13 --> 26
14 --> 2
14 --> 15
- 14 -.-> 22
+ 14 --> 18
+ 14 -.-> 23
16 --> 6
16 --> 17
17 -.-> 13
- 18 --> 17
- 19 --> 12
- 19 --> 14
- 20 --> 1
- 21 --> 23
- 21 --> 24
- 22 --> 21
- 26 -.-> 4
- 27 --> 6
+ 19 --> 17
+ 20 --> 12
+ 20 --> 14
+ 21 --> 1
+ 22 --> 24
+ 22 --> 25
+ 23 --> 22
+ 27 -.-> 4
+ 28 --> 6
diff --git a/libs/@local/hashql/hir/docs/dependency-diagram.mmd b/libs/@local/hashql/hir/docs/dependency-diagram.mmd
index 3bb3fb061a4..01d987d5bcb 100644
--- a/libs/@local/hashql/hir/docs/dependency-diagram.mmd
+++ b/libs/@local/hashql/hir/docs/dependency-diagram.mmd
@@ -27,50 +27,52 @@ graph TD
16[hashql-eval]
17[hashql-hir]
class 17 root
- 18[hashql-mir]
- 19[hashql-syntax-jexpr]
- 20[hash-temporal-client]
- 21[darwin-kperf]
- 22[darwin-kperf-criterion]
- 23[darwin-kperf-events]
- 24[darwin-kperf-sys]
- 25[error-stack]
- 26[hash-graph-benches]
- 27[hash-graph-test-data]
+ 18[hashql-macros]
+ 19[hashql-mir]
+ 20[hashql-syntax-jexpr]
+ 21[hash-temporal-client]
+ 22[darwin-kperf]
+ 23[darwin-kperf-criterion]
+ 24[darwin-kperf-events]
+ 25[darwin-kperf-sys]
+ 26[error-stack]
+ 27[hash-graph-benches]
+ 28[hash-graph-test-data]
0 --> 8
1 --> 7
- 1 -.-> 27
+ 1 -.-> 28
2 -.-> 3
2 --> 11
4 --> 16
- 4 --> 19
+ 4 --> 20
5 --> 1
6 --> 5
6 --> 9
- 6 --> 20
+ 6 --> 21
7 --> 2
8 --> 4
- 9 -.-> 27
+ 9 -.-> 28
11 -.-> 10
11 --> 10
- 11 --> 25
+ 11 --> 26
12 -.-> 13
13 --> 16
- 13 --> 18
13 --> 19
- 13 --> 25
+ 13 --> 20
+ 13 --> 26
14 --> 2
14 --> 15
- 14 -.-> 22
+ 14 --> 18
+ 14 -.-> 23
16 --> 6
16 --> 17
17 -.-> 13
- 18 --> 17
- 19 --> 12
- 19 --> 14
- 20 --> 1
- 21 --> 23
- 21 --> 24
- 22 --> 21
- 26 -.-> 4
- 27 --> 6
+ 19 --> 17
+ 20 --> 12
+ 20 --> 14
+ 21 --> 1
+ 22 --> 24
+ 22 --> 25
+ 23 --> 22
+ 27 -.-> 4
+ 28 --> 6
diff --git a/libs/@local/hashql/macros/Cargo.toml b/libs/@local/hashql/macros/Cargo.toml
new file mode 100644
index 00000000000..a601e45b9e3
--- /dev/null
+++ b/libs/@local/hashql/macros/Cargo.toml
@@ -0,0 +1,25 @@
+[package]
+name = "hashql-macros"
+version.workspace = true
+edition.workspace = true
+license.workspace = true
+publish.workspace = true
+authors.workspace = true
+
+[lib]
+proc-macro = true
+
+[lints]
+workspace = true
+
+[dependencies]
+# Public workspace dependencies
+
+# Public third-party dependencies
+
+# Private workspace dependencies
+
+# Private third-party dependencies
+proc-macro2 = { workspace = true }
+quote = { workspace = true, features = ["proc-macro"] }
+unsynn = { workspace = true }
diff --git a/libs/@local/hashql/macros/LICENSE.md b/libs/@local/hashql/macros/LICENSE.md
new file mode 100644
index 00000000000..8ebfe728d7d
--- /dev/null
+++ b/libs/@local/hashql/macros/LICENSE.md
@@ -0,0 +1,606 @@
+# GNU Affero General Public License
+
+_Version 3, 19 November 2007_
+_Copyright © 2007 Free Software Foundation, Inc. <>_
+
+Everyone is permitted to copy and distribute verbatim copies
+of this license document, but changing it is not allowed.
+
+## Preamble
+
+The GNU Affero General Public License is a free, copyleft license for
+software and other kinds of works, specifically designed to ensure
+cooperation with the community in the case of network server software.
+
+The licenses for most software and other practical works are designed
+to take away your freedom to share and change the works. By contrast,
+our General Public Licenses are intended to guarantee your freedom to
+share and change all versions of a program--to make sure it remains free
+software for all its users.
+
+When we speak of free software, we are referring to freedom, not
+price. Our General Public Licenses are designed to make sure that you
+have the freedom to distribute copies of free software (and charge for
+them if you wish), that you receive source code or can get it if you
+want it, that you can change the software or use pieces of it in new
+free programs, and that you know you can do these things.
+
+Developers that use our General Public Licenses protect your rights
+with two steps: **(1)** assert copyright on the software, and **(2)** offer
+you this License which gives you legal permission to copy, distribute
+and/or modify the software.
+
+A secondary benefit of defending all users' freedom is that
+improvements made in alternate versions of the program, if they
+receive widespread use, become available for other developers to
+incorporate. Many developers of free software are heartened and
+encouraged by the resulting cooperation. However, in the case of
+software used on network servers, this result may fail to come about.
+The GNU General Public License permits making a modified version and
+letting the public access it on a server without ever releasing its
+source code to the public.
+
+The GNU Affero General Public License is designed specifically to
+ensure that, in such cases, the modified source code becomes available
+to the community. It requires the operator of a network server to
+provide the source code of the modified version running there to the
+users of that server. Therefore, public use of a modified version, on
+a publicly accessible server, gives the public access to the source
+code of the modified version.
+
+An older license, called the Affero General Public License and
+published by Affero, was designed to accomplish similar goals. This is
+a different license, not a version of the Affero GPL, but Affero has
+released a new version of the Affero GPL which permits relicensing under
+this license.
+
+The precise terms and conditions for copying, distribution and
+modification follow.
+
+## TERMS AND CONDITIONS
+
+### 0. Definitions
+
+“This License” refers to version 3 of the GNU Affero General Public License.
+
+“Copyright” also means copyright-like laws that apply to other kinds of
+works, such as semiconductor masks.
+
+“The Program” refers to any copyrightable work licensed under this
+License. Each licensee is addressed as “you”. “Licensees” and
+“recipients” may be individuals or organizations.
+
+To “modify” a work means to copy from or adapt all or part of the work
+in a fashion requiring copyright permission, other than the making of an
+exact copy. The resulting work is called a “modified version” of the
+earlier work or a work “based on” the earlier work.
+
+A “covered work” means either the unmodified Program or a work based
+on the Program.
+
+To “propagate” a work means to do anything with it that, without
+permission, would make you directly or secondarily liable for
+infringement under applicable copyright law, except executing it on a
+computer or modifying a private copy. Propagation includes copying,
+distribution (with or without modification), making available to the
+public, and in some countries other activities as well.
+
+To “convey” a work means any kind of propagation that enables other
+parties to make or receive copies. Mere interaction with a user through
+a computer network, with no transfer of a copy, is not conveying.
+
+An interactive user interface displays “Appropriate Legal Notices”
+to the extent that it includes a convenient and prominently visible
+feature that **(1)** displays an appropriate copyright notice, and **(2)**
+tells the user that there is no warranty for the work (except to the
+extent that warranties are provided), that licensees may convey the
+work under this License, and how to view a copy of this License. If
+the interface presents a list of user commands or options, such as a
+menu, a prominent item in the list meets this criterion.
+
+### 1. Source Code
+
+The “source code” for a work means the preferred form of the work
+for making modifications to it. “Object code” means any non-source
+form of a work.
+
+A “Standard Interface” means an interface that either is an official
+standard defined by a recognized standards body, or, in the case of
+interfaces specified for a particular programming language, one that
+is widely used among developers working in that language.
+
+The “System Libraries” of an executable work include anything, other
+than the work as a whole, that **(a)** is included in the normal form of
+packaging a Major Component, but which is not part of that Major
+Component, and **(b)** serves only to enable use of the work with that
+Major Component, or to implement a Standard Interface for which an
+implementation is available to the public in source code form. A
+“Major Component”, in this context, means a major essential component
+(kernel, window system, and so on) of the specific operating system
+(if any) on which the executable work runs, or a compiler used to
+produce the work, or an object code interpreter used to run it.
+
+The “Corresponding Source” for a work in object code form means all
+the source code needed to generate, install, and (for an executable
+work) run the object code and to modify the work, including scripts to
+control those activities. However, it does not include the work's
+System Libraries, or general-purpose tools or generally available free
+programs which are used unmodified in performing those activities but
+which are not part of the work. For example, Corresponding Source
+includes interface definition files associated with source files for
+the work, and the source code for shared libraries and dynamically
+linked subprograms that the work is specifically designed to require,
+such as by intimate data communication or control flow between those
+subprograms and other parts of the work.
+
+The Corresponding Source need not include anything that users
+can regenerate automatically from other parts of the Corresponding
+Source.
+
+The Corresponding Source for a work in source code form is that
+same work.
+
+### 2. Basic Permissions
+
+All rights granted under this License are granted for the term of
+copyright on the Program, and are irrevocable provided the stated
+conditions are met. This License explicitly affirms your unlimited
+permission to run the unmodified Program. The output from running a
+covered work is covered by this License only if the output, given its
+content, constitutes a covered work. This License acknowledges your
+rights of fair use or other equivalent, as provided by copyright law.
+
+You may make, run and propagate covered works that you do not
+convey, without conditions so long as your license otherwise remains
+in force. You may convey covered works to others for the sole purpose
+of having them make modifications exclusively for you, or provide you
+with facilities for running those works, provided that you comply with
+the terms of this License in conveying all material for which you do
+not control copyright. Those thus making or running the covered works
+for you must do so exclusively on your behalf, under your direction
+and control, on terms that prohibit them from making any copies of
+your copyrighted material outside their relationship with you.
+
+Conveying under any other circumstances is permitted solely under
+the conditions stated below. Sublicensing is not allowed; section 10
+makes it unnecessary.
+
+### 3. Protecting Users' Legal Rights From Anti-Circumvention Law
+
+No covered work shall be deemed part of an effective technological
+measure under any applicable law fulfilling obligations under article
+11 of the WIPO copyright treaty adopted on 20 December 1996, or
+similar laws prohibiting or restricting circumvention of such
+measures.
+
+When you convey a covered work, you waive any legal power to forbid
+circumvention of technological measures to the extent such circumvention
+is effected by exercising rights under this License with respect to
+the covered work, and you disclaim any intention to limit operation or
+modification of the work as a means of enforcing, against the work's
+users, your or third parties' legal rights to forbid circumvention of
+technological measures.
+
+### 4. Conveying Verbatim Copies
+
+You may convey verbatim copies of the Program's source code as you
+receive it, in any medium, provided that you conspicuously and
+appropriately publish on each copy an appropriate copyright notice;
+keep intact all notices stating that this License and any
+non-permissive terms added in accord with section 7 apply to the code;
+keep intact all notices of the absence of any warranty; and give all
+recipients a copy of this License along with the Program.
+
+You may charge any price or no price for each copy that you convey,
+and you may offer support or warranty protection for a fee.
+
+### 5. Conveying Modified Source Versions
+
+You may convey a work based on the Program, or the modifications to
+produce it from the Program, in the form of source code under the
+terms of section 4, provided that you also meet all of these conditions:
+
+- **a)** The work must carry prominent notices stating that you modified
+ it, and giving a relevant date.
+- **b)** The work must carry prominent notices stating that it is
+ released under this License and any conditions added under section 7.
+ This requirement modifies the requirement in section 4 to
+ “keep intact all notices”.
+- **c)** You must license the entire work, as a whole, under this
+ License to anyone who comes into possession of a copy. This
+ License will therefore apply, along with any applicable section 7
+ additional terms, to the whole of the work, and all its parts,
+ regardless of how they are packaged. This License gives no
+ permission to license the work in any other way, but it does not
+ invalidate such permission if you have separately received it.
+- **d)** If the work has interactive user interfaces, each must display
+ Appropriate Legal Notices; however, if the Program has interactive
+ interfaces that do not display Appropriate Legal Notices, your
+ work need not make them do so.
+
+A compilation of a covered work with other separate and independent
+works, which are not by their nature extensions of the covered work,
+and which are not combined with it such as to form a larger program,
+in or on a volume of a storage or distribution medium, is called an
+“aggregate” if the compilation and its resulting copyright are not
+used to limit the access or legal rights of the compilation's users
+beyond what the individual works permit. Inclusion of a covered work
+in an aggregate does not cause this License to apply to the other
+parts of the aggregate.
+
+### 6. Conveying Non-Source Forms
+
+You may convey a covered work in object code form under the terms
+of sections 4 and 5, provided that you also convey the
+machine-readable Corresponding Source under the terms of this License,
+in one of these ways:
+
+- **a)** Convey the object code in, or embodied in, a physical product
+ (including a physical distribution medium), accompanied by the
+ Corresponding Source fixed on a durable physical medium
+ customarily used for software interchange.
+- **b)** Convey the object code in, or embodied in, a physical product
+ (including a physical distribution medium), accompanied by a
+ written offer, valid for at least three years and valid for as
+ long as you offer spare parts or customer support for that product
+ model, to give anyone who possesses the object code either **(1)** a
+ copy of the Corresponding Source for all the software in the
+ product that is covered by this License, on a durable physical
+ medium customarily used for software interchange, for a price no
+ more than your reasonable cost of physically performing this
+ conveying of source, or **(2)** access to copy the
+ Corresponding Source from a network server at no charge.
+- **c)** Convey individual copies of the object code with a copy of the
+ written offer to provide the Corresponding Source. This
+ alternative is allowed only occasionally and noncommercially, and
+ only if you received the object code with such an offer, in accord
+ with subsection 6b.
+- **d)** Convey the object code by offering access from a designated
+ place (gratis or for a charge), and offer equivalent access to the
+ Corresponding Source in the same way through the same place at no
+ further charge. You need not require recipients to copy the
+ Corresponding Source along with the object code. If the place to
+ copy the object code is a network server, the Corresponding Source
+ may be on a different server (operated by you or a third party)
+ that supports equivalent copying facilities, provided you maintain
+ clear directions next to the object code saying where to find the
+ Corresponding Source. Regardless of what server hosts the
+ Corresponding Source, you remain obligated to ensure that it is
+ available for as long as needed to satisfy these requirements.
+- **e)** Convey the object code using peer-to-peer transmission, provided
+ you inform other peers where the object code and Corresponding
+ Source of the work are being offered to the general public at no
+ charge under subsection 6d.
+
+A separable portion of the object code, whose source code is excluded
+from the Corresponding Source as a System Library, need not be
+included in conveying the object code work.
+
+A “User Product” is either **(1)** a “consumer product”, which means any
+tangible personal property which is normally used for personal, family,
+or household purposes, or **(2)** anything designed or sold for incorporation
+into a dwelling. In determining whether a product is a consumer product,
+doubtful cases shall be resolved in favor of coverage. For a particular
+product received by a particular user, “normally used” refers to a
+typical or common use of that class of product, regardless of the status
+of the particular user or of the way in which the particular user
+actually uses, or expects or is expected to use, the product. A product
+is a consumer product regardless of whether the product has substantial
+commercial, industrial or non-consumer uses, unless such uses represent
+the only significant mode of use of the product.
+
+“Installation Information” for a User Product means any methods,
+procedures, authorization keys, or other information required to install
+and execute modified versions of a covered work in that User Product from
+a modified version of its Corresponding Source. The information must
+suffice to ensure that the continued functioning of the modified object
+code is in no case prevented or interfered with solely because
+modification has been made.
+
+If you convey an object code work under this section in, or with, or
+specifically for use in, a User Product, and the conveying occurs as
+part of a transaction in which the right of possession and use of the
+User Product is transferred to the recipient in perpetuity or for a
+fixed term (regardless of how the transaction is characterized), the
+Corresponding Source conveyed under this section must be accompanied
+by the Installation Information. But this requirement does not apply
+if neither you nor any third party retains the ability to install
+modified object code on the User Product (for example, the work has
+been installed in ROM).
+
+The requirement to provide Installation Information does not include a
+requirement to continue to provide support service, warranty, or updates
+for a work that has been modified or installed by the recipient, or for
+the User Product in which it has been modified or installed. Access to a
+network may be denied when the modification itself materially and
+adversely affects the operation of the network or violates the rules and
+protocols for communication across the network.
+
+Corresponding Source conveyed, and Installation Information provided,
+in accord with this section must be in a format that is publicly
+documented (and with an implementation available to the public in
+source code form), and must require no special password or key for
+unpacking, reading or copying.
+
+### 7. Additional Terms
+
+“Additional permissions” are terms that supplement the terms of this
+License by making exceptions from one or more of its conditions.
+Additional permissions that are applicable to the entire Program shall
+be treated as though they were included in this License, to the extent
+that they are valid under applicable law. If additional permissions
+apply only to part of the Program, that part may be used separately
+under those permissions, but the entire Program remains governed by
+this License without regard to the additional permissions.
+
+When you convey a copy of a covered work, you may at your option
+remove any additional permissions from that copy, or from any part of
+it. (Additional permissions may be written to require their own
+removal in certain cases when you modify the work.) You may place
+additional permissions on material, added by you to a covered work,
+for which you have or can give appropriate copyright permission.
+
+Notwithstanding any other provision of this License, for material you
+add to a covered work, you may (if authorized by the copyright holders of
+that material) supplement the terms of this License with terms:
+
+- **a)** Disclaiming warranty or limiting liability differently from the
+ terms of sections 15 and 16 of this License; or
+- **b)** Requiring preservation of specified reasonable legal notices or
+ author attributions in that material or in the Appropriate Legal
+ Notices displayed by works containing it; or
+- **c)** Prohibiting misrepresentation of the origin of that material, or
+ requiring that modified versions of such material be marked in
+ reasonable ways as different from the original version; or
+- **d)** Limiting the use for publicity purposes of names of licensors or
+ authors of the material; or
+- **e)** Declining to grant rights under trademark law for use of some
+ trade names, trademarks, or service marks; or
+- **f)** Requiring indemnification of licensors and authors of that
+ material by anyone who conveys the material (or modified versions of
+ it) with contractual assumptions of liability to the recipient, for
+ any liability that these contractual assumptions directly impose on
+ those licensors and authors.
+
+All other non-permissive additional terms are considered “further
+restrictions” within the meaning of section 10. If the Program as you
+received it, or any part of it, contains a notice stating that it is
+governed by this License along with a term that is a further
+restriction, you may remove that term. If a license document contains
+a further restriction but permits relicensing or conveying under this
+License, you may add to a covered work material governed by the terms
+of that license document, provided that the further restriction does
+not survive such relicensing or conveying.
+
+If you add terms to a covered work in accord with this section, you
+must place, in the relevant source files, a statement of the
+additional terms that apply to those files, or a notice indicating
+where to find the applicable terms.
+
+Additional terms, permissive or non-permissive, may be stated in the
+form of a separately written license, or stated as exceptions;
+the above requirements apply either way.
+
+### 8. Termination
+
+You may not propagate or modify a covered work except as expressly
+provided under this License. Any attempt otherwise to propagate or
+modify it is void, and will automatically terminate your rights under
+this License (including any patent licenses granted under the third
+paragraph of section 11).
+
+However, if you cease all violation of this License, then your
+license from a particular copyright holder is reinstated **(a)**
+provisionally, unless and until the copyright holder explicitly and
+finally terminates your license, and **(b)** permanently, if the copyright
+holder fails to notify you of the violation by some reasonable means
+prior to 60 days after the cessation.
+
+Moreover, your license from a particular copyright holder is
+reinstated permanently if the copyright holder notifies you of the
+violation by some reasonable means, this is the first time you have
+received notice of violation of this License (for any work) from that
+copyright holder, and you cure the violation prior to 30 days after
+your receipt of the notice.
+
+Termination of your rights under this section does not terminate the
+licenses of parties who have received copies or rights from you under
+this License. If your rights have been terminated and not permanently
+reinstated, you do not qualify to receive new licenses for the same
+material under section 10.
+
+### 9. Acceptance Not Required for Having Copies
+
+You are not required to accept this License in order to receive or
+run a copy of the Program. Ancillary propagation of a covered work
+occurring solely as a consequence of using peer-to-peer transmission
+to receive a copy likewise does not require acceptance. However,
+nothing other than this License grants you permission to propagate or
+modify any covered work. These actions infringe copyright if you do
+not accept this License. Therefore, by modifying or propagating a
+covered work, you indicate your acceptance of this License to do so.
+
+### 10. Automatic Licensing of Downstream Recipients
+
+Each time you convey a covered work, the recipient automatically
+receives a license from the original licensors, to run, modify and
+propagate that work, subject to this License. You are not responsible
+for enforcing compliance by third parties with this License.
+
+An “entity transaction” is a transaction transferring control of an
+organization, or substantially all assets of one, or subdividing an
+organization, or merging organizations. If propagation of a covered
+work results from an entity transaction, each party to that
+transaction who receives a copy of the work also receives whatever
+licenses to the work the party's predecessor in interest had or could
+give under the previous paragraph, plus a right to possession of the
+Corresponding Source of the work from the predecessor in interest, if
+the predecessor has it or can get it with reasonable efforts.
+
+You may not impose any further restrictions on the exercise of the
+rights granted or affirmed under this License. For example, you may
+not impose a license fee, royalty, or other charge for exercise of
+rights granted under this License, and you may not initiate litigation
+(including a cross-claim or counterclaim in a lawsuit) alleging that
+any patent claim is infringed by making, using, selling, offering for
+sale, or importing the Program or any portion of it.
+
+### 11. Patents
+
+A “contributor” is a copyright holder who authorizes use under this
+License of the Program or a work on which the Program is based. The
+work thus licensed is called the contributor's “contributor version”.
+
+A contributor's “essential patent claims” are all patent claims
+owned or controlled by the contributor, whether already acquired or
+hereafter acquired, that would be infringed by some manner, permitted
+by this License, of making, using, or selling its contributor version,
+but do not include claims that would be infringed only as a
+consequence of further modification of the contributor version. For
+purposes of this definition, “control” includes the right to grant
+patent sublicenses in a manner consistent with the requirements of
+this License.
+
+Each contributor grants you a non-exclusive, worldwide, royalty-free
+patent license under the contributor's essential patent claims, to
+make, use, sell, offer for sale, import and otherwise run, modify and
+propagate the contents of its contributor version.
+
+In the following three paragraphs, a “patent license” is any express
+agreement or commitment, however denominated, not to enforce a patent
+(such as an express permission to practice a patent or covenant not to
+sue for patent infringement). To “grant” such a patent license to a
+party means to make such an agreement or commitment not to enforce a
+patent against the party.
+
+If you convey a covered work, knowingly relying on a patent license,
+and the Corresponding Source of the work is not available for anyone
+to copy, free of charge and under the terms of this License, through a
+publicly available network server or other readily accessible means,
+then you must either **(1)** cause the Corresponding Source to be so
+available, or **(2)** arrange to deprive yourself of the benefit of the
+patent license for this particular work, or **(3)** arrange, in a manner
+consistent with the requirements of this License, to extend the patent
+license to downstream recipients. “Knowingly relying” means you have
+actual knowledge that, but for the patent license, your conveying the
+covered work in a country, or your recipient's use of the covered work
+in a country, would infringe one or more identifiable patents in that
+country that you have reason to believe are valid.
+
+If, pursuant to or in connection with a single transaction or
+arrangement, you convey, or propagate by procuring conveyance of, a
+covered work, and grant a patent license to some of the parties
+receiving the covered work authorizing them to use, propagate, modify
+or convey a specific copy of the covered work, then the patent license
+you grant is automatically extended to all recipients of the covered
+work and works based on it.
+
+A patent license is “discriminatory” if it does not include within
+the scope of its coverage, prohibits the exercise of, or is
+conditioned on the non-exercise of one or more of the rights that are
+specifically granted under this License. You may not convey a covered
+work if you are a party to an arrangement with a third party that is
+in the business of distributing software, under which you make payment
+to the third party based on the extent of your activity of conveying
+the work, and under which the third party grants, to any of the
+parties who would receive the covered work from you, a discriminatory
+patent license **(a)** in connection with copies of the covered work
+conveyed by you (or copies made from those copies), or **(b)** primarily
+for and in connection with specific products or compilations that
+contain the covered work, unless you entered into that arrangement,
+or that patent license was granted, prior to 28 March 2007.
+
+Nothing in this License shall be construed as excluding or limiting
+any implied license or other defenses to infringement that may
+otherwise be available to you under applicable patent law.
+
+### 12. No Surrender of Others' Freedom
+
+If conditions are imposed on you (whether by court order, agreement or
+otherwise) that contradict the conditions of this License, they do not
+excuse you from the conditions of this License. If you cannot convey a
+covered work so as to satisfy simultaneously your obligations under this
+License and any other pertinent obligations, then as a consequence you may
+not convey it at all. For example, if you agree to terms that obligate you
+to collect a royalty for further conveying from those to whom you convey
+the Program, the only way you could satisfy both those terms and this
+License would be to refrain entirely from conveying the Program.
+
+### 13. Remote Network Interaction; Use with the GNU General Public License
+
+Notwithstanding any other provision of this License, if you modify the
+Program, your modified version must prominently offer all users
+interacting with it remotely through a computer network (if your version
+supports such interaction) an opportunity to receive the Corresponding
+Source of your version by providing access to the Corresponding Source
+from a network server at no charge, through some standard or customary
+means of facilitating copying of software. This Corresponding Source
+shall include the Corresponding Source for any work covered by version 3
+of the GNU General Public License that is incorporated pursuant to the
+following paragraph.
+
+Notwithstanding any other provision of this License, you have
+permission to link or combine any covered work with a work licensed
+under version 3 of the GNU General Public License into a single
+combined work, and to convey the resulting work. The terms of this
+License will continue to apply to the part which is the covered work,
+but the work with which it is combined will remain governed by version
+3 of the GNU General Public License.
+
+### 14. Revised Versions of this License
+
+The Free Software Foundation may publish revised and/or new versions of
+the GNU Affero General Public License from time to time. Such new versions
+will be similar in spirit to the present version, but may differ in detail to
+address new problems or concerns.
+
+Each version is given a distinguishing version number. If the
+Program specifies that a certain numbered version of the GNU Affero General
+Public License “or any later version” applies to it, you have the
+option of following the terms and conditions either of that numbered
+version or of any later version published by the Free Software
+Foundation. If the Program does not specify a version number of the
+GNU Affero General Public License, you may choose any version ever published
+by the Free Software Foundation.
+
+If the Program specifies that a proxy can decide which future
+versions of the GNU Affero General Public License can be used, that proxy's
+public statement of acceptance of a version permanently authorizes you
+to choose that version for the Program.
+
+Later license versions may give you additional or different
+permissions. However, no additional obligations are imposed on any
+author or copyright holder as a result of your choosing to follow a
+later version.
+
+### 15. Disclaimer of Warranty
+
+THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY
+APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT
+HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM “AS IS” WITHOUT WARRANTY
+OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO,
+THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM
+IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF
+ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
+
+### 16. Limitation of Liability
+
+IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
+WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS
+THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY
+GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE
+USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF
+DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD
+PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS),
+EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF
+SUCH DAMAGES.
+
+### 17. Interpretation of Sections 15 and 16
+
+If the disclaimer of warranty and limitation of liability provided
+above cannot be given local legal effect according to their terms,
+reviewing courts shall apply local law that most closely approximates
+an absolute waiver of all civil liability in connection with the
+Program, unless a warranty or assumption of liability accompanies a
+copy of the Program in return for a fee.
diff --git a/libs/@local/hashql/macros/package.json b/libs/@local/hashql/macros/package.json
new file mode 100644
index 00000000000..fc812410670
--- /dev/null
+++ b/libs/@local/hashql/macros/package.json
@@ -0,0 +1,10 @@
+{
+ "name": "@rust/hashql-macros",
+ "version": "0.0.0-private",
+ "private": true,
+ "license": "AGPL-3",
+ "scripts": {
+ "fix:clippy": "just clippy --fix",
+ "lint:clippy": "just clippy"
+ }
+}
diff --git a/libs/@local/hashql/macros/src/grammar.rs b/libs/@local/hashql/macros/src/grammar.rs
new file mode 100644
index 00000000000..358f34c5b3f
--- /dev/null
+++ b/libs/@local/hashql/macros/src/grammar.rs
@@ -0,0 +1,44 @@
+#![expect(clippy::result_large_err)]
+use unsynn::{
+ BracketGroupContaining, Cons, Either, Except, Gt, Ident, Lt, Many, ParenthesisGroupContaining,
+ PathSep, PathSepDelimited, Pound, TokenTree, keyword, unsynn,
+};
+
+keyword! {
+ pub KPub = "pub";
+ pub KStruct = "struct";
+ pub KEnum = "enum";
+ pub KIn = "in";
+ pub KId = "id";
+ pub KDerive = "derive";
+ pub KDisplay = "display";
+ pub KStep = "Step";
+ pub KIs = "is";
+ pub KCrate = "crate";
+ pub KConst = "const";
+ pub KU8 = "u8";
+ pub KU16 = "u16";
+ pub KU32 = "u32";
+ pub KU64 = "u64";
+ pub KU128 = "u128";
+}
+
+pub(crate) type VerbatimUntil = Many, AngleTokenTree>>;
+pub(crate) type ModPath = Cons