refactor: working on constructor injection 175 pass 2 fail

This commit is contained in:
Patrick Stewart 2024-12-30 01:27:56 -07:00
parent 22ade7f699
commit ab9ebde517
3 changed files with 67 additions and 420 deletions

View file

@ -299,90 +299,20 @@ class Container {
if (contextualConcrete != null) { if (contextualConcrete != null) {
dynamic instance; dynamic instance;
if (contextualConcrete is Function) { if (contextualConcrete is Function) {
// Remove current type from stack to avoid circular dependency // Keep current type on stack for dependency resolution
_buildStack.removeLast();
try {
instance = contextualConcrete(this); instance = contextualConcrete(this);
} finally {
_buildStack.add(t2);
}
} else if (contextualConcrete is Type) { } else if (contextualConcrete is Type) {
// For Type bindings, we need to use reflection to create the instance // For Type bindings, we need to use reflection to create the instance
_buildStack.removeLast(); // Remove current type from stack instance = make(contextualConcrete);
try {
var reflectedType = reflector.reflectType(contextualConcrete);
if (reflectedType is ReflectedClass) {
bool isDefault(String name) {
return name.isEmpty || name == reflectedType.name;
}
var constructor = reflectedType.constructors.firstWhere(
(c) => isDefault(c.name),
orElse: (() => throw BindingResolutionException(
'${reflectedType.name} has no default constructor, and therefore cannot be instantiated.')));
var positional = [];
var named = <String, Object>{};
for (var param in constructor.parameters) {
// Check for parameter override
var override = getParameterOverride(param.name);
if (override != null) {
if (param.isNamed) {
named[param.name] = override;
} else {
positional.add(override);
}
continue;
}
// No override, resolve normally
var value = make(param.type.reflectedType);
if (param.isNamed) {
named[param.name] = value;
} else {
positional.add(value);
}
}
try {
dynamic tempInstance;
try {
tempInstance = reflectedType.newInstance(
isDefault(constructor.name) ? '' : constructor.name,
positional,
named, []).reflectee;
} on AbstractClassInstantiationError {
throw BindingResolutionException(
'Cannot instantiate abstract class ${reflectedType.name}');
}
tempInstance = _applyExtenders(t2, tempInstance);
_fireResolvingCallbacks(tempInstance);
_fireAfterResolvingCallbacks(tempInstance);
return tempInstance as T;
} catch (e) {
if (e is AbstractClassInstantiationError) {
throw BindingResolutionException(
'Cannot instantiate abstract class ${reflectedType.name}');
}
rethrow;
}
}
} finally {
_buildStack.add(t2); // Add it back
}
} }
if (instance != null) { if (instance != null) {
instance = _applyExtenders(t2, instance); instance = _applyExtenders(t2, instance);
var typedInstance = instance as T; _fireResolvingCallbacks(instance);
_fireResolvingCallbacks(typedInstance); _fireAfterResolvingCallbacks(instance);
_fireAfterResolvingCallbacks(typedInstance); return instance as T;
return typedInstance;
} }
} }
// Check for contextual binding in parent classes // Check for contextual binding in parent classes
var parentContextual = _getContextualConcreteFromParent(t2); var parentContextual = _getContextualConcreteFromParent(t2);
if (parentContextual != null) { if (parentContextual != null) {
@ -485,8 +415,8 @@ class Container {
} }
} }
var positional = []; // var positional = [];
var named = <String, Object>{}; // var named = <String, Object>{};
if (reflectedType is ReflectedClass) { if (reflectedType is ReflectedClass) {
bool isDefault(String name) { bool isDefault(String name) {
@ -498,6 +428,17 @@ class Container {
orElse: (() => throw BindingResolutionException( orElse: (() => throw BindingResolutionException(
'${reflectedType.name} has no default constructor, and therefore cannot be instantiated.'))); '${reflectedType.name} has no default constructor, and therefore cannot be instantiated.')));
// Check if we can resolve all constructor parameters
for (var param in constructor.parameters) {
var paramType = param.type.reflectedType;
if (param.isRequired &&
!has(paramType) &&
reflector.reflectType(paramType) == null) {
throw BindingResolutionException(
'No binding was found for required parameter $paramType in ${reflectedType.name}');
}
}
// Add current type to build stack before resolving parameters // Add current type to build stack before resolving parameters
_buildStack.add(t2); _buildStack.add(t2);
try { try {
@ -505,7 +446,7 @@ class Container {
var named = <String, Object>{}; var named = <String, Object>{};
for (var param in constructor.parameters) { for (var param in constructor.parameters) {
// Check for parameter override // Check for parameter override first
var override = getParameterOverride(param.name); var override = getParameterOverride(param.name);
if (override != null) { if (override != null) {
if (param.isNamed) { if (param.isNamed) {
@ -516,27 +457,26 @@ class Container {
continue; continue;
} }
// Skip optional parameters if we can't resolve them // Try to resolve the parameter
if (!param.isRequired) {
try { try {
var value = make(param.type.reflectedType); var value = make(param.type.reflectedType);
if (param.isNamed) { if (param.isNamed) {
// For named parameters, only add if we successfully resolved
named[param.name] = value; named[param.name] = value;
} else { } else {
// For positional parameters, handle errors
positional.add(value); positional.add(value);
} }
} catch (e) { } catch (e) {
// Skip optional parameter if we can't resolve it // Only rethrow for required parameters
continue; if (param.isRequired) {
rethrow;
} }
} else { // For optional positional parameters, add null
// Required parameter, must resolve if (!param.isNamed) {
var value = make(param.type.reflectedType); positional.add(null);
if (param.isNamed) {
named[param.name] = value;
} else {
positional.add(value);
} }
// Skip optional named parameters that can't be resolved
} }
} }
@ -1267,7 +1207,15 @@ class Container {
/// var logger = container.makeWith<Logger>({'level': 'debug'}); /// var logger = container.makeWith<Logger>({'level': 'debug'});
/// ``` /// ```
T makeWith<T>(Map<String, dynamic> parameters, [Type? type]) { T makeWith<T>(Map<String, dynamic> parameters, [Type? type]) {
return withParameters(parameters, () => make<T>(type)); // Push parameters onto stack
_parameterStack.add(parameters);
try {
// Make instance with parameters
return make<T>(type);
} finally {
// Always pop parameters from stack
_parameterStack.removeLast();
}
} }
/// Create a factory binding for deferred resolution /// Create a factory binding for deferred resolution

View file

@ -33,9 +33,17 @@ class ContextualBindingBuilder {
/// Bind directly to a concrete implementation /// Bind directly to a concrete implementation
void to(Type implementation) { void to(Type implementation) {
for (var concreteType in concrete) { for (var abstractType in concrete) {
container.addContextualBinding( // Add contextual binding with a factory function
concreteType, concreteType, implementation); container.addContextualBinding(abstractType, abstractType, (c) {
// Get parameter overrides
var overrides = c.getParameterOverride('name');
if (overrides != null) {
return c.withParameters(
{'name': overrides}, () => c.make(implementation));
}
return c.make(implementation);
});
} }
} }
} }
@ -70,331 +78,22 @@ class ContextualImplementationBuilder {
} }
} }
/// Bind to a concrete implementation type
// void to(Type implementation) {
// for (var concreteType in concrete) {
// // Add contextual binding with implementation as the value
// container.addContextualBinding(
// concreteType, concreteType, implementation);
// // Also add a contextual binding with implementation as the key
// container.addContextualBinding(
// implementation, concreteType, implementation);
// // Also register a singleton for direct resolution
// container.registerSingleton(container.make(implementation),
// as: concreteType);
// }
// }
// void to(Type implementation) {
// for (var concreteType in concrete) {
// // Register a factory that creates the implementation
// container.registerFactory(
// (c) => c.withParameters({}, () {
// var reflectedType = c.reflector.reflectType(implementation);
// if (reflectedType is ReflectedClass) {
// bool isDefault(String name) {
// return name.isEmpty || name == reflectedType.name;
// }
// var constructor = reflectedType.constructors.firstWhere(
// (c) => isDefault(c.name),
// orElse: (() => throw BindingResolutionException(
// '${reflectedType.name} has no default constructor')));
// return reflectedType.newInstance(
// isDefault(constructor.name) ? '' : constructor.name,
// [],
// {},
// []).reflectee;
// }
// throw BindingResolutionException(
// '$implementation is not a class, and therefore cannot be instantiated.');
// }),
// as: concreteType);
// }
// }
// void to(Type implementation) {
// for (var concreteType in concrete) {
// // Keep existing contextual binding
// container.addContextualBinding(
// concreteType, concreteType, implementation);
// // Add factory that returns implementation instance
// container.registerFactory((c) {
// // Create implementation instance
// var reflectedType = c.reflector.reflectType(implementation);
// if (reflectedType is ReflectedClass) {
// bool isDefault(String name) {
// return name.isEmpty || name == reflectedType.name;
// }
// var constructor = reflectedType.constructors.firstWhere(
// (c) => isDefault(c.name),
// orElse: (() => throw BindingResolutionException(
// '${reflectedType.name} has no default constructor')));
// // Get all parameter overrides
// var positional = [];
// var named = <String, Object>{};
// for (var param in constructor.parameters) {
// var override = c.getParameterOverride(param.name);
// if (override != null) {
// if (param.isNamed) {
// named[param.name] = override;
// } else {
// positional.add(override);
// }
// }
// }
// return reflectedType.newInstance(
// isDefault(constructor.name) ? '' : constructor.name,
// positional,
// named, []).reflectee;
// }
// return c.make(implementation);
// }, as: concreteType);
// }
// }
// void to(Type implementation) {
// for (var concreteType in concrete) {
// // Register a factory that returns implementation instance
// container.registerFactory((c) {
// // Get parameter overrides
// var overrides = c.getParameterOverride('name');
// if (overrides != null) {
// return c.withParameters({'name': overrides}, () {
// // Create a new instance of the implementation
// var reflectedType = c.reflector.reflectType(implementation);
// if (reflectedType is ReflectedClass) {
// bool isDefault(String name) {
// return name.isEmpty || name == reflectedType.name;
// }
// var constructor = reflectedType.constructors.firstWhere(
// (c) => isDefault(c.name),
// orElse: (() => throw BindingResolutionException(
// '${reflectedType.name} has no default constructor')));
// return reflectedType.newInstance(
// isDefault(constructor.name) ? '' : constructor.name,
// [],
// {'name': overrides},
// []).reflectee;
// }
// throw BindingResolutionException(
// '$implementation is not a class, and therefore cannot be instantiated.');
// });
// }
// // Create a new instance of the implementation
// var reflectedType = c.reflector.reflectType(implementation);
// if (reflectedType is ReflectedClass) {
// bool isDefault(String name) {
// return name.isEmpty || name == reflectedType.name;
// }
// var constructor = reflectedType.constructors.firstWhere(
// (c) => isDefault(c.name),
// orElse: (() => throw BindingResolutionException(
// '${reflectedType.name} has no default constructor')));
// return reflectedType.newInstance(
// isDefault(constructor.name) ? '' : constructor.name,
// [],
// {},
// []).reflectee;
// }
// throw BindingResolutionException(
// '$implementation is not a class, and therefore cannot be instantiated.');
// }, as: concreteType);
// }
// }
// void to(Type implementation) {
// for (var concreteType in concrete) {
// // Add contextual binding with a factory function
// container.addContextualBinding(
// concreteType, concreteType, (c) => c.make(implementation));
// }
// }
// void to(Type implementation) {
// for (var concreteType in concrete) {
// // Add contextual binding with a factory function that handles both cases
// container.addContextualBinding(concreteType, concreteType, (c) {
// try {
// return c.make(implementation);
// } catch (e) {
// if (e.toString().contains('Cannot instantiate abstract class')) {
// return null;
// }
// rethrow;
// }
// });
// }
// }
// void to(Type implementation) {
// for (var concreteType in concrete) {
// // Add contextual binding with a factory function
// container.addContextualBinding(implementation, concreteType, (c) {
// // Get parameter overrides first
// var overrides = c.getParameterOverride('name');
// if (overrides != null) {
// return c.withParameters({'name': overrides}, () {
// return c.make(implementation);
// });
// }
// return c.make(implementation);
// });
// }
// }
// void to(Type implementation) {
// for (var abstractType in concrete) {
// // Register a factory that returns implementation instance
// container.registerFactory((c) {
// // Get parameter overrides first
// var overrides = c.getParameterOverride('name');
// if (overrides != null) {
// return c.withParameters({'name': overrides}, () {
// // Create a new instance of the implementation
// var reflectedType = c.reflector.reflectType(implementation);
// if (reflectedType is ReflectedClass) {
// bool isDefault(String name) {
// return name.isEmpty || name == reflectedType.name;
// }
// var constructor = reflectedType.constructors.firstWhere(
// (c) => isDefault(c.name),
// orElse: (() => throw BindingResolutionException(
// '${reflectedType.name} has no default constructor')));
// return reflectedType.newInstance(
// isDefault(constructor.name) ? '' : constructor.name,
// [],
// {'name': overrides},
// []).reflectee;
// }
// throw BindingResolutionException(
// '$implementation is not a class, and therefore cannot be instantiated.');
// });
// }
// // Create a new instance of the implementation
// var reflectedType = c.reflector.reflectType(implementation);
// if (reflectedType is ReflectedClass) {
// bool isDefault(String name) {
// return name.isEmpty || name == reflectedType.name;
// }
// var constructor = reflectedType.constructors.firstWhere(
// (c) => isDefault(c.name),
// orElse: (() => throw BindingResolutionException(
// '${reflectedType.name} has no default constructor')));
// return reflectedType.newInstance(
// isDefault(constructor.name) ? '' : constructor.name,
// [],
// {},
// []).reflectee;
// }
// throw BindingResolutionException(
// '$implementation is not a class, and therefore cannot be instantiated.');
// }, as: abstractType);
// }
// }
// void to(Type implementation) {
// for (var abstractType in concrete) {
// // Add contextual binding from abstract to implementation
// container.addContextualBinding(
// abstractType, abstractType, implementation);
// // Register a factory that handles parameter overrides
// container.registerFactory((c) {
// // Get parameter overrides
// var nameOverride = c.getParameterOverride('name');
// if (nameOverride != null) {
// return c.withParameters(
// {'name': nameOverride}, () => c.make(implementation));
// }
// return c.make(implementation);
// }, as: abstractType);
// }
// }
// void to(Type implementation) {
// for (var abstractType in concrete) {
// // Add contextual binding with a factory function
// container.addContextualBinding(abstractType, abstractType, (c) {
// // Get parameter overrides first
// var nameOverride = c.getParameterOverride('name');
// if (nameOverride != null) {
// return c.withParameters({'name': nameOverride}, () {
// var reflectedType = c.reflector.reflectType(implementation);
// if (reflectedType is ReflectedClass) {
// bool isDefault(String name) {
// return name.isEmpty || name == reflectedType.name;
// }
// var constructor = reflectedType.constructors.firstWhere(
// (c) => isDefault(c.name),
// orElse: (() => throw BindingResolutionException(
// '${reflectedType.name} has no default constructor')));
// return reflectedType.newInstance(
// isDefault(constructor.name) ? '' : constructor.name,
// [],
// {'name': nameOverride},
// []).reflectee;
// }
// throw BindingResolutionException(
// '$implementation is not a class, and therefore cannot be instantiated.');
// });
// }
// // Create implementation instance
// var reflectedType = c.reflector.reflectType(implementation);
// if (reflectedType is ReflectedClass) {
// bool isDefault(String name) {
// return name.isEmpty || name == reflectedType.name;
// }
// var constructor = reflectedType.constructors.firstWhere(
// (c) => isDefault(c.name),
// orElse: (() => throw BindingResolutionException(
// '${reflectedType.name} has no default constructor')));
// return reflectedType.newInstance(
// isDefault(constructor.name) ? '' : constructor.name,
// [],
// {},
// []).reflectee;
// }
// throw BindingResolutionException(
// '$implementation is not a class, and therefore cannot be instantiated.');
// });
// }
// }
// void to(Type implementation) {
// for (var abstractType in concrete) {
// // Register a factory that handles parameter overrides
// container.registerFactory((c) => c.make(implementation),
// as: abstractType);
// }
// }
void to(Type implementation) { void to(Type implementation) {
for (var abstractType in concrete) { for (var abstractType in concrete) {
// Add contextual binding with a factory function // Add contextual binding with a factory function
container.addContextualBinding( container.addContextualBinding(abstractType, abstractType, (c) {
abstractType, abstractType, (c) => c.make(implementation)); // Get parameter overrides
var overrides = c.getParameterOverride('name');
if (overrides != null) {
// Create instance with named parameters
var reflectedType = c.reflector.reflectType(implementation);
if (reflectedType is ReflectedClass) {
return reflectedType
.newInstance('', [], {'name': overrides}, []).reflectee;
}
}
return c.make(implementation);
});
} }
} }

View file