1
- use std:: collections:: HashMap ;
2
-
3
1
use axum:: {
4
2
extract:: { Path , Query , State } ,
5
3
routing:: get,
6
4
Json , Router ,
7
5
} ;
8
- use futures:: { StreamExt , TryStreamExt } ;
9
6
use serde:: { Deserialize , Serialize } ;
10
7
11
8
use crate :: server:: { self , Context , Error } ;
@@ -93,9 +90,22 @@ async fn index(
93
90
Ok ( Json ( IndexBody { features } ) )
94
91
}
95
92
93
+ #[ derive( Serialize ) ]
94
+ struct Run {
95
+ id : i32 ,
96
+ count : i32 ,
97
+ }
98
+
99
+ #[ derive( Serialize ) ]
100
+ struct ShowFeature {
101
+ id : i32 ,
102
+ name : String ,
103
+ runs : Vec < Run > ,
104
+ }
105
+
96
106
#[ derive( Serialize ) ]
97
107
struct ShowResponse {
98
- counts : HashMap < String , i32 > ,
108
+ feature : ShowFeature ,
99
109
}
100
110
101
111
/// Shows counts for samples with the given configuration ID and feature name.
@@ -115,30 +125,46 @@ async fn show(
115
125
Path ( ( configuration_id, id) ) : Path < ( i32 , i32 ) > ,
116
126
State ( ctx) : State < Context > ,
117
127
) -> server:: Result < Json < ShowResponse > > {
118
- let counts = sqlx:: query!(
128
+ let rows = sqlx:: query!(
119
129
"
120
130
select
121
- samples.name,
131
+ features.name,
132
+ runs.id,
122
133
counts.value
123
134
from counts
135
+ inner join features
136
+ on features.id = counts.feature_id
124
137
inner join runs
125
138
on runs.id = counts.run_id
126
139
inner join samples
127
140
on samples.id = runs.sample_id
128
- inner join features
129
- on features.id = counts.feature_id
130
- where runs.configuration_id = $1
131
- and features.id = $2
141
+ where counts.feature_id = $1
142
+ and runs.configuration_id = $2
132
143
" ,
133
- configuration_id,
134
144
id,
145
+ configuration_id,
135
146
)
136
- . fetch ( & ctx. pool )
137
- . map ( |result| result. map ( |record| ( record. name , record. value ) ) )
138
- . try_collect ( )
147
+ . fetch_all ( & ctx. pool )
139
148
. await ?;
140
149
141
- Ok ( Json ( ShowResponse { counts } ) )
150
+ if rows. is_empty ( ) {
151
+ return Err ( Error :: NotFound ) ;
152
+ }
153
+
154
+ // SAFETY: `rows` is non-empty.
155
+ let name = rows[ 0 ] . name . clone ( ) ;
156
+
157
+ let runs = rows
158
+ . into_iter ( )
159
+ . map ( |row| Run {
160
+ id : row. id ,
161
+ count : row. value ,
162
+ } )
163
+ . collect ( ) ;
164
+
165
+ Ok ( Json ( ShowResponse {
166
+ feature : ShowFeature { id, name, runs } ,
167
+ } ) )
142
168
}
143
169
144
170
#[ cfg( test) ]
@@ -213,9 +239,16 @@ mod tests {
213
239
assert_eq ! (
214
240
actual,
215
241
json!( {
216
- "counts" : {
217
- "sample1" : 5 ,
218
- "sample2" : 13 ,
242
+ "feature" : {
243
+ "id" : 1 ,
244
+ "name" : "39_feature_1" ,
245
+ "runs" : [ {
246
+ "id" : 1 ,
247
+ "count" : 5 ,
248
+ } , {
249
+ "id" : 2 ,
250
+ "count" : 13 ,
251
+ } ] ,
219
252
}
220
253
} )
221
254
) ;
0 commit comments