@@ -395,13 +395,57 @@ impl<T: std::fmt::Display> std::fmt::Display for Move<T> {
395
395
}
396
396
}
397
397
398
- /// Specifies a graph edge/resource that can be "read" by a node.
399
- pub struct View < T > {
398
+ /// Used to generate a default value of a resource, if possible.
399
+ pub trait Gen < T > {
400
+ fn generate ( ) -> Option < T > ;
401
+ }
402
+
403
+ /// Valueless type that represents the ability to generate a resource by
404
+ /// default.
405
+ pub struct SomeDefault ;
406
+
407
+ impl < T : Default > Gen < T > for SomeDefault {
408
+ fn generate ( ) -> Option < T > {
409
+ Some ( T :: default ( ) )
410
+ }
411
+ }
412
+
413
+ /// Valueless type that represents the **inability** to generate a resource by default.
414
+ pub struct NoDefault ;
415
+
416
+ impl < T > Gen < T > for NoDefault {
417
+ fn generate ( ) -> Option < T > {
418
+ None
419
+ }
420
+ }
421
+
422
+ /// Immutably borrowed resource that _may_ be created by default.
423
+ ///
424
+ /// Node functions wrap their parameters in [`View`], [`ViewMut`] or [`Move`].
425
+ ///
426
+ /// `View` has two type parameters:
427
+ /// * `T` - The type of the resource.
428
+ /// * `G` - The method by which the resource can be generated if it doesn't
429
+ /// already exist. By default this is [`SomeDefault`], which denotes creating the
430
+ /// resource using its default instance. Another option is [`NoDefault`] which
431
+ /// fails to generate the resource.
432
+ ///
433
+ /// ```rust
434
+ /// use moongraph::*;
435
+ ///
436
+ /// let mut graph = Graph::default();
437
+ /// let default_number = graph.visit(|u: View<usize>| { *u }).map_err(|e| e.to_string());
438
+ /// assert_eq!(Ok(0), default_number);
439
+ ///
440
+ /// let no_number = graph.visit(|f: View<f32, NoDefault>| *f);
441
+ /// assert!(no_number.is_err());
442
+ /// ```
443
+ pub struct View < T , G : Gen < T > = SomeDefault > {
400
444
inner : Loan ,
401
- _phantom : PhantomData < T > ,
445
+ _phantom : PhantomData < ( T , G ) > ,
402
446
}
403
447
404
- impl < T : Any + Send + Sync > Deref for View < T > {
448
+ impl < T : Any + Send + Sync , G : Gen < T > > Deref for View < T , G > {
405
449
type Target = T ;
406
450
407
451
fn deref ( & self ) -> & Self :: Target {
@@ -410,40 +454,68 @@ impl<T: Any + Send + Sync> Deref for View<T> {
410
454
}
411
455
}
412
456
413
- impl < T : Any + Send + Sync > Edges for View < T > {
457
+ impl < T : Any + Send + Sync , G : Gen < T > > Edges for View < T , G > {
414
458
fn reads ( ) -> Vec < TypeKey > {
415
459
vec ! [ TypeKey :: new:: <T >( ) ]
416
460
}
417
461
418
462
fn construct ( resources : & mut TypeMap ) -> Result < Self , GraphError > {
419
463
let key = TypeKey :: new :: < T > ( ) ;
420
- let inner = resources
421
- . loan ( key)
422
- . context ( ResourceSnafu ) ?
423
- . context ( MissingSnafu {
424
- name : std:: any:: type_name :: < T > ( ) ,
425
- } ) ?;
464
+ let inner = match resources. loan ( key) . context ( ResourceSnafu ) ? {
465
+ Some ( inner) => inner,
466
+ None => {
467
+ let t = G :: generate ( ) . context ( MissingSnafu {
468
+ name : std:: any:: type_name :: < T > ( ) ,
469
+ } ) ?;
470
+ // UNWRAP: safe because we know this type was missing, and no other type
471
+ // is stored with this type's type id.
472
+ let _ = resources. insert_value ( t) . unwrap ( ) ;
473
+ log:: trace!( "generated missing {}" , std:: any:: type_name:: <T >( ) ) ;
474
+ // UNWRAP: safe because we just inserted
475
+ resources. loan ( key) . unwrap ( ) . unwrap ( )
476
+ }
477
+ } ;
426
478
Ok ( View {
427
479
inner,
428
480
_phantom : PhantomData ,
429
481
} )
430
482
}
431
483
}
432
484
433
- impl < T : std:: fmt:: Display + Any + Send + Sync > std:: fmt:: Display for View < T > {
485
+ impl < T : std:: fmt:: Display + Any + Send + Sync , G : Gen < T > > std:: fmt:: Display for View < T , G > {
434
486
fn fmt ( & self , f : & mut std:: fmt:: Formatter < ' _ > ) -> std:: fmt:: Result {
435
487
let t: & T = self . inner . downcast_ref ( ) . unwrap ( ) ;
436
488
t. fmt ( f)
437
489
}
438
490
}
439
491
440
- /// Specifies a graph edge/resource that can be "written" to by a node.
441
- pub struct ViewMut < T > {
492
+ /// A mutably borrowed resource that may be created by default.
493
+ ///
494
+ /// Node functions wrap their parameters in [`View`], [`ViewMut`] or [`Move`].
495
+ ///
496
+ /// `ViewMut` has two type parameters:
497
+ /// * `T` - The type of the resource.
498
+ /// * `G` - The method by which the resource can be generated if it doesn't
499
+ /// already exist. By default this is [`SomeDefault`], which denotes creating
500
+ /// the resource using its default implementation. Another option is
501
+ /// [`NoDefault`] which fails to generate the resource.
502
+ ///
503
+ /// ```rust
504
+ /// use moongraph::*;
505
+ ///
506
+ /// let mut graph = Graph::default();
507
+ /// let default_number = graph.visit(|u: ViewMut<usize>| { *u }).map_err(|e| e.to_string());
508
+ /// assert_eq!(Ok(0), default_number);
509
+ ///
510
+ /// let no_number = graph.visit(|f: ViewMut<f32, NoDefault>| *f);
511
+ /// assert!(no_number.is_err());
512
+ /// ```
513
+ pub struct ViewMut < T , G : Gen < T > = SomeDefault > {
442
514
inner : LoanMut ,
443
- _phantom : PhantomData < T > ,
515
+ _phantom : PhantomData < ( T , G ) > ,
444
516
}
445
517
446
- impl < T : Any + Send + Sync > Deref for ViewMut < T > {
518
+ impl < T : Any + Send + Sync , G : Gen < T > > Deref for ViewMut < T , G > {
447
519
type Target = T ;
448
520
449
521
fn deref ( & self ) -> & Self :: Target {
@@ -452,34 +524,42 @@ impl<T: Any + Send + Sync> Deref for ViewMut<T> {
452
524
}
453
525
}
454
526
455
- impl < T : Any + Send + Sync > DerefMut for ViewMut < T > {
527
+ impl < T : Any + Send + Sync , G : Gen < T > > DerefMut for ViewMut < T , G > {
456
528
fn deref_mut ( & mut self ) -> & mut Self :: Target {
457
529
// UNWRAP: safe because it was constructed with `T`
458
530
self . inner . downcast_mut ( ) . unwrap ( )
459
531
}
460
532
}
461
533
462
- impl < ' a , T : Any + Send + Sync > Edges for ViewMut < T > {
534
+ impl < ' a , T : Any + Send + Sync , G : Gen < T > > Edges for ViewMut < T , G > {
463
535
fn writes ( ) -> Vec < TypeKey > {
464
536
vec ! [ TypeKey :: new:: <T >( ) ]
465
537
}
466
538
467
539
fn construct ( resources : & mut TypeMap ) -> Result < Self , GraphError > {
468
540
let key = TypeKey :: new :: < T > ( ) ;
469
- let inner = resources
470
- . loan_mut ( key)
471
- . context ( ResourceSnafu ) ?
472
- . context ( MissingSnafu {
473
- name : std:: any:: type_name :: < T > ( ) ,
474
- } ) ?;
541
+ let inner = match resources. loan_mut ( key) . context ( ResourceSnafu ) ? {
542
+ Some ( inner) => inner,
543
+ None => {
544
+ let t = G :: generate ( ) . context ( MissingSnafu {
545
+ name : std:: any:: type_name :: < T > ( ) ,
546
+ } ) ?;
547
+ // UNWRAP: safe because we know this type was missing, and no other type
548
+ // is stored with this type's type id.
549
+ let _ = resources. insert_value ( t) . unwrap ( ) ;
550
+ log:: trace!( "generated missing {}" , std:: any:: type_name:: <T >( ) ) ;
551
+ // UNWRAP: safe because we just inserted
552
+ resources. loan_mut ( key) . unwrap ( ) . unwrap ( )
553
+ }
554
+ } ;
475
555
Ok ( ViewMut {
476
556
inner,
477
557
_phantom : PhantomData ,
478
558
} )
479
559
}
480
560
}
481
561
482
- impl < T : std:: fmt:: Display + Any + Send + Sync > std:: fmt:: Display for ViewMut < T > {
562
+ impl < T : std:: fmt:: Display + Any + Send + Sync , G : Gen < T > > std:: fmt:: Display for ViewMut < T , G > {
483
563
fn fmt ( & self , f : & mut std:: fmt:: Formatter < ' _ > ) -> std:: fmt:: Result {
484
564
let t: & T = self . inner . downcast_ref ( ) . unwrap ( ) ;
485
565
t. fmt ( f)
@@ -1090,11 +1170,15 @@ impl Graph {
1090
1170
}
1091
1171
1092
1172
/// Get a reference to a resource in the graph.
1173
+ ///
1174
+ /// If the resource _does not_ exist `Ok(None)` will be returned.
1093
1175
pub fn get_resource < T : Any + Send + Sync > ( & self ) -> Result < Option < & T > , GraphError > {
1094
1176
Ok ( self . resources . get_value ( ) . context ( ResourceSnafu ) ?)
1095
1177
}
1096
1178
1097
1179
/// Get a mutable reference to a resource in the graph.
1180
+ ///
1181
+ /// If the resource _does not_ exist `Ok(None)` will be returned.
1098
1182
pub fn get_resource_mut < T : Any + Send + Sync > ( & mut self ) -> Result < Option < & mut T > , GraphError > {
1099
1183
Ok ( self . resources . get_value_mut ( ) . context ( ResourceSnafu ) ?)
1100
1184
}
@@ -1505,4 +1589,16 @@ mod test {
1505
1589
assert ! ( run_was_all_good, "run was not all good" ) ;
1506
1590
assert_eq ! ( 110.0 , my_num, "local did not run" ) ;
1507
1591
}
1592
+
1593
+ #[ test]
1594
+ // Tests that Gen will generate a default for a missing resource,
1595
+ // and that the result will be stored in the graph.
1596
+ fn can_generate_view_default ( ) {
1597
+ let mut graph = Graph :: default ( ) ;
1598
+ let u = graph. visit ( |u : View < usize > | * u) . unwrap ( ) ;
1599
+ assert_eq ! ( 0 , u) ;
1600
+
1601
+ let my_u = graph. get_resource :: < usize > ( ) . unwrap ( ) ;
1602
+ assert_eq ! ( Some ( 0 ) , my_u. copied( ) ) ;
1603
+ }
1508
1604
}
0 commit comments