diff --git a/404.html b/404.html index e196df49..01c4d813 100644 --- a/404.html +++ b/404.html @@ -10,13 +10,13 @@ - +
Skip to main content

Page Not Found

We could not find what you were looking for.

Please contact the owner of the site that linked you to the original URL and let them know their link is broken.

- + \ No newline at end of file diff --git a/assets/js/026a065b.c7fa72ce.js b/assets/js/026a065b.c7fa72ce.js deleted file mode 100644 index 70d8ba3c..00000000 --- a/assets/js/026a065b.c7fa72ce.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkwebsite=self.webpackChunkwebsite||[]).push([[340],{3905:(e,n,t)=>{t.d(n,{Zo:()=>u,kt:()=>b});var o=t(7294);function a(e,n,t){return n in e?Object.defineProperty(e,n,{value:t,enumerable:!0,configurable:!0,writable:!0}):e[n]=t,e}function r(e,n){var t=Object.keys(e);if(Object.getOwnPropertySymbols){var o=Object.getOwnPropertySymbols(e);n&&(o=o.filter((function(n){return Object.getOwnPropertyDescriptor(e,n).enumerable}))),t.push.apply(t,o)}return t}function i(e){for(var n=1;n=0||(a[t]=e[t]);return a}(e,n);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);for(o=0;o=0||Object.prototype.propertyIsEnumerable.call(e,t)&&(a[t]=e[t])}return a}var s=o.createContext({}),c=function(e){var n=o.useContext(s),t=n;return e&&(t="function"==typeof e?e(n):i(i({},n),e)),t},u=function(e){var n=c(e.components);return o.createElement(s.Provider,{value:n},e.children)},p="mdxType",d={inlineCode:"code",wrapper:function(e){var n=e.children;return o.createElement(o.Fragment,{},n)}},g=o.forwardRef((function(e,n){var t=e.components,a=e.mdxType,r=e.originalType,s=e.parentName,u=l(e,["components","mdxType","originalType","parentName"]),p=c(t),g=a,b=p["".concat(s,".").concat(g)]||p[g]||d[g]||r;return t?o.createElement(b,i(i({ref:n},u),{},{components:t})):o.createElement(b,i({ref:n},u))}));function b(e,n){var t=arguments,a=n&&n.mdxType;if("string"==typeof e||a){var r=t.length,i=new Array(r);i[0]=g;var l={};for(var s in n)hasOwnProperty.call(n,s)&&(l[s]=n[s]);l.originalType=e,l[p]="string"==typeof e?e:a,i[1]=l;for(var c=2;c{t.d(n,{Z:()=>i});var o=t(7294),a=t(6010);const r="tabItem_Ymn6";function i(e){let{children:n,hidden:t,className:i}=e;return o.createElement("div",{role:"tabpanel",className:(0,a.Z)(r,i),hidden:t},n)}},4866:(e,n,t)=>{t.d(n,{Z:()=>N});var o=t(7462),a=t(7294),r=t(6010),i=t(2466),l=t(6550),s=t(1980),c=t(7392),u=t(12);function p(e){return function(e){return a.Children.map(e,(e=>{if(!e||(0,a.isValidElement)(e)&&function(e){const{props:n}=e;return!!n&&"object"==typeof n&&"value"in n}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}(e).map((e=>{let{props:{value:n,label:t,attributes:o,default:a}}=e;return{value:n,label:t,attributes:o,default:a}}))}function d(e){const{values:n,children:t}=e;return(0,a.useMemo)((()=>{const e=n??p(t);return function(e){const n=(0,c.l)(e,((e,n)=>e.value===n.value));if(n.length>0)throw new Error(`Docusaurus error: Duplicate values "${n.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[n,t])}function g(e){let{value:n,tabValues:t}=e;return t.some((e=>e.value===n))}function b(e){let{queryString:n=!1,groupId:t}=e;const o=(0,l.k6)(),r=function(e){let{queryString:n=!1,groupId:t}=e;if("string"==typeof n)return n;if(!1===n)return null;if(!0===n&&!t)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return t??null}({queryString:n,groupId:t});return[(0,s._X)(r),(0,a.useCallback)((e=>{if(!r)return;const n=new URLSearchParams(o.location.search);n.set(r,e),o.replace({...o.location,search:n.toString()})}),[r,o])]}function m(e){const{defaultValue:n,queryString:t=!1,groupId:o}=e,r=d(e),[i,l]=(0,a.useState)((()=>function(e){let{defaultValue:n,tabValues:t}=e;if(0===t.length)throw new Error("Docusaurus error: the component requires at least one children component");if(n){if(!g({value:n,tabValues:t}))throw new Error(`Docusaurus error: The has a defaultValue "${n}" but none of its children has the corresponding value. Available values are: ${t.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return n}const o=t.find((e=>e.default))??t[0];if(!o)throw new Error("Unexpected error: 0 tabValues");return o.value}({defaultValue:n,tabValues:r}))),[s,c]=b({queryString:t,groupId:o}),[p,m]=function(e){let{groupId:n}=e;const t=function(e){return e?`docusaurus.tab.${e}`:null}(n),[o,r]=(0,u.Nk)(t);return[o,(0,a.useCallback)((e=>{t&&r.set(e)}),[t,r])]}({groupId:o}),k=(()=>{const e=s??p;return g({value:e,tabValues:r})?e:null})();(0,a.useLayoutEffect)((()=>{k&&l(k)}),[k]);return{selectedValue:i,selectValue:(0,a.useCallback)((e=>{if(!g({value:e,tabValues:r}))throw new Error(`Can't select invalid tab value=${e}`);l(e),c(e),m(e)}),[c,m,r]),tabValues:r}}var k=t(2389);const h="tabList__CuJ",y="tabItem_LNqP";function v(e){let{className:n,block:t,selectedValue:l,selectValue:s,tabValues:c}=e;const u=[],{blockElementScrollPositionUntilNextRender:p}=(0,i.o5)(),d=e=>{const n=e.currentTarget,t=u.indexOf(n),o=c[t].value;o!==l&&(p(n),s(o))},g=e=>{let n=null;switch(e.key){case"Enter":d(e);break;case"ArrowRight":{const t=u.indexOf(e.currentTarget)+1;n=u[t]??u[0];break}case"ArrowLeft":{const t=u.indexOf(e.currentTarget)-1;n=u[t]??u[u.length-1];break}}n?.focus()};return a.createElement("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,r.Z)("tabs",{"tabs--block":t},n)},c.map((e=>{let{value:n,label:t,attributes:i}=e;return a.createElement("li",(0,o.Z)({role:"tab",tabIndex:l===n?0:-1,"aria-selected":l===n,key:n,ref:e=>u.push(e),onKeyDown:g,onClick:d},i,{className:(0,r.Z)("tabs__item",y,i?.className,{"tabs__item--active":l===n})}),t??n)})))}function f(e){let{lazy:n,children:t,selectedValue:o}=e;const r=(Array.isArray(t)?t:[t]).filter(Boolean);if(n){const e=r.find((e=>e.props.value===o));return e?(0,a.cloneElement)(e,{className:"margin-top--md"}):null}return a.createElement("div",{className:"margin-top--md"},r.map(((e,n)=>(0,a.cloneElement)(e,{key:n,hidden:e.props.value!==o}))))}function I(e){const n=m(e);return a.createElement("div",{className:(0,r.Z)("tabs-container",h)},a.createElement(v,(0,o.Z)({},e,n)),a.createElement(f,(0,o.Z)({},e,n)))}function N(e){const n=(0,k.Z)();return a.createElement(I,(0,o.Z)({key:String(n)},e))}},8846:(e,n,t)=>{t.d(n,{Z:()=>l});var o=t(7294);const a="codeDescContainer_ie8f",r="desc_jyqI",i="example_eYlF";function l(e){let{children:n}=e,t=o.Children.toArray(n).filter((e=>e));return o.createElement("div",{className:a},o.createElement("div",{className:r},t[0]),o.createElement("div",{className:i},t[1]))}},385:(e,n,t)=>{t.r(n),t.d(n,{assets:()=>p,contentTitle:()=>c,default:()=>b,frontMatter:()=>s,metadata:()=>u,toc:()=>d});var o=t(7462),a=(t(7294),t(3905)),r=t(8846),i=t(4866),l=t(5162);const s={},c="Service resolution",u={unversionedId:"guides/service-resolution",id:"guides/service-resolution",title:"Service resolution",description:"When you have all your components registered and configured adequately, you can resolve them from the container or a scope by requesting their service type.",source:"@site/docs/guides/service-resolution.md",sourceDirName:"guides",slug:"/guides/service-resolution",permalink:"/stashbox/docs/guides/service-resolution",draft:!1,editUrl:"https://github.com/z4kn4fein/stashbox/edit/master/docs/docs/guides/service-resolution.md",tags:[],version:"current",lastUpdatedBy:"dependabot[bot]",lastUpdatedAt:1697644420,formattedLastUpdatedAt:"Oct 18, 2023",frontMatter:{},sidebar:"docs",previous:{title:"Advanced registration",permalink:"/stashbox/docs/guides/advanced-registration"},next:{title:"Lifetimes",permalink:"/stashbox/docs/guides/lifetimes"}},p={},d=[{value:"Injection patterns",id:"injection-patterns",level:2},{value:"Attributes",id:"attributes",level:2},{value:"Dependency binding",id:"dependency-binding",level:2},{value:"Conventional resolution",id:"conventional-resolution",level:2},{value:"Conditional resolution",id:"conditional-resolution",level:2},{value:"Optional resolution",id:"optional-resolution",level:2},{value:"Dependency overrides",id:"dependency-overrides",level:2},{value:"Activation",id:"activation",level:2},{value:"Build-up",id:"build-up",level:3}],g={toc:d};function b(e){let{components:n,...t}=e;return(0,a.kt)("wrapper",(0,o.Z)({},g,t,{components:n,mdxType:"MDXLayout"}),(0,a.kt)("h1",{id:"service-resolution"},"Service resolution"),(0,a.kt)("p",null,"When you have all your components registered and configured adequately, you can resolve them from the container or a ",(0,a.kt)("a",{parentName:"p",href:"/docs/guides/scopes"},"scope")," by requesting their ",(0,a.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type"),"."),(0,a.kt)("p",null,"During a service's resolution, the container walks through the entire ",(0,a.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#resolution-tree"},"resolution tree")," and instantiates all dependencies required for the service construction.\nWhen the container encounters any violations of ",(0,a.kt)("a",{parentName:"p",href:"/docs/diagnostics/validation#resolution-validation"},"these rules")," ",(0,a.kt)("em",{parentName:"p"},"(circular dependencies, missing required services, lifetime misconfigurations)")," during the walkthrough, it lets you know that something is wrong by throwing a specific exception."),(0,a.kt)("h2",{id:"injection-patterns"},"Injection patterns"),(0,a.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,(0,a.kt)("strong",{parentName:"p"},"Constructor injection")," is the ",(0,a.kt)("em",{parentName:"p"},"primary dependency injection pattern"),". It encourages the organization of dependencies to a single place - the constructor."),(0,a.kt)("p",null,"Stashbox, by default, uses the constructor that has the most parameters it knows how to resolve. This behavior is configurable through ",(0,a.kt)("a",{parentName:"p",href:"/docs/configuration/registration-configuration#constructor-selection"},"constructor selection"),"."),(0,a.kt)("p",null,(0,a.kt)("a",{parentName:"p",href:"/docs/configuration/registration-configuration#property-field-injection"},"Property/field injection")," is also supported in cases where constructor injection is not applicable."),(0,a.kt)("admonition",{type:"info"},(0,a.kt)("p",{parentName:"admonition"},(0,a.kt)("a",{parentName:"p",href:"/docs/configuration/container-configuration#constructor-selection"},"Constructor selection")," and ",(0,a.kt)("a",{parentName:"p",href:"/docs/configuration/container-configuration#auto-member-injection"},"property/field injection")," is also configurable container-wide."))),(0,a.kt)("div",null,(0,a.kt)(i.Z,{mdxType:"Tabs"},(0,a.kt)(l.Z,{value:"Constructor injection",label:"Constructor injection",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"class DbBackup : IJob\n{\n private readonly ILogger logger;\n private readonly IEventBroadcaster eventBroadcaster;\n\n public DbBackup(ILogger logger, IEventBroadcaster eventBroadcaster)\n {\n this.logger = logger;\n this.eventBroadcaster = eventBroadcaster;\n }\n}\n\ncontainer.Register();\ncontainer.Register();\n\ncontainer.Register();\n\n// resolution using the available constructor.\nIJob job = container.Resolve();\n"))),(0,a.kt)(l.Z,{value:"Property/field injection",label:"Property/field injection",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"class DbBackup : IJob\n{\n public ILogger Logger { get; set; }\n public IEventBroadcaster EventBroadcaster { get; set; }\n\n public DbBackup() \n { }\n}\n\ncontainer.Register();\ncontainer.Register();\n\n// registration of service with auto member injection.\ncontainer.Register(options => \n options.WithAutoMemberInjection());\n\n// resolution will inject the properties.\nIJob job = container.Resolve();\n")))))),(0,a.kt)("admonition",{type:"caution"},(0,a.kt)("p",{parentName:"admonition"},"It's a common mistake to use the ",(0,a.kt)("em",{parentName:"p"},"property/field injection")," only to disencumber the constructor from having too many parameters. That's a code smell and also violates the ",(0,a.kt)("a",{parentName:"p",href:"https://en.wikipedia.org/wiki/Single-responsibility_principle"},"Single-responsibility principle"),". If you recognize these conditions, you should consider splitting your class into multiple smaller units rather than adding an extra property-injected dependency. ")),(0,a.kt)("h2",{id:"attributes"},"Attributes"),(0,a.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"Attributes can give you control over how Stashbox selects dependencies for a service's resolution."),(0,a.kt)("p",null,(0,a.kt)("strong",{parentName:"p"},"Dependency attribute"),": "),(0,a.kt)("ul",null,(0,a.kt)("li",{parentName:"ul"},(0,a.kt)("p",{parentName:"li"},(0,a.kt)("strong",{parentName:"p"},"On a constructor/method parameter"),": used with the ",(0,a.kt)("em",{parentName:"p"},"name")," property, it works as a marker for ",(0,a.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#named-resolution"},"named resolution"),".")),(0,a.kt)("li",{parentName:"ul"},(0,a.kt)("p",{parentName:"li"},(0,a.kt)("strong",{parentName:"p"},"On a property/field"),": first, it enables ",(0,a.kt)("em",{parentName:"p"},"auto-injection")," on the marked property/field (even if it wasn't configured at registration explicitly), and just as with the method parameter, it allows ",(0,a.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#named-resolution"},"named resolution"),"."))),(0,a.kt)("p",null,(0,a.kt)("strong",{parentName:"p"},"InjectionMethod attribute"),": marks a method to be called when the requested service is instantiated.")),(0,a.kt)("div",null,(0,a.kt)(i.Z,{mdxType:"Tabs"},(0,a.kt)(l.Z,{value:"Constructor",label:"Constructor",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},'class DbBackup : IJob\n{\n private readonly ILogger logger;\n\n public DbBackup([Dependency("Console")]ILogger logger)\n {\n this.logger = logger;\n }\n}\n\ncontainer.Register("Console");\ncontainer.Register("File");\n\ncontainer.Register();\n\n// the container will resolve DbBackup with ConsoleLogger.\nIJob job = container.Resolve();\n'))),(0,a.kt)(l.Z,{value:"Property/field",label:"Property/field",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},'class DbBackup : IJob\n{\n [Dependency("Console")]\n public ILogger Logger { get; set; }\n\n public DbBackup() \n { }\n}\n\ncontainer.Register("Console");\ncontainer.Register("File");\n\ncontainer.Register();\n\n// the container will resolve DbBackup with ConsoleLogger.\nIJob job = container.Resolve();\n'))),(0,a.kt)(l.Z,{value:"Method",label:"Method",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},'class DbBackup : IJob\n{\n [InjectionMethod]\n public void Initialize([Dependency("Console")]ILogger logger)\n {\n this.logger.Log("Initializing.");\n }\n}\n\ncontainer.Register("Console");\ncontainer.Register("File");\n\ncontainer.Register();\n\n// the container will call DbBackup\'s Initialize method.\nIJob job = container.Resolve();\n')))))),(0,a.kt)("admonition",{type:"caution"},(0,a.kt)("p",{parentName:"admonition"},"Attributes provide a more straightforward configuration, but using them also tightens the bond between your application and Stashbox. If you consider this an issue, the same functionality is available on the ",(0,a.kt)("em",{parentName:"p"},"registration API")," as ",(0,a.kt)("a",{parentName:"p",href:"/docs/guides/service-resolution#dependency-binding"},"dependency binding"),".")),(0,a.kt)("h2",{id:"dependency-binding"},"Dependency binding"),(0,a.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"The same dependency configuration as attributes is available on the registration configuration API."),(0,a.kt)("ul",null,(0,a.kt)("li",{parentName:"ul"},(0,a.kt)("p",{parentName:"li"},(0,a.kt)("strong",{parentName:"p"},"Bind to parameter"),": it has the same functionality as the ",(0,a.kt)("a",{parentName:"p",href:"/docs/guides/service-resolution#attributes"},"Dependency attribute")," on a constructor or method parameter, enabling the ",(0,a.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#named-resolution"},"named resolution"),".")),(0,a.kt)("li",{parentName:"ul"},(0,a.kt)("p",{parentName:"li"},(0,a.kt)("strong",{parentName:"p"},"Bind to property/field"),": it has the same functionality as the ",(0,a.kt)("a",{parentName:"p",href:"/docs/guides/service-resolution#attributes"},"Dependency attribute"),", enabling the injection of the given property/field."))),(0,a.kt)("admonition",{type:"info"},(0,a.kt)("p",{parentName:"admonition"},"There are further dependency binding options ",(0,a.kt)("a",{parentName:"p",href:"/docs/configuration/registration-configuration#dependency-configuration"},"available")," on the registration configuration API."))),(0,a.kt)("div",null,(0,a.kt)(i.Z,{mdxType:"Tabs"},(0,a.kt)(l.Z,{value:"Bind to parameter",label:"Bind to parameter",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},'class DbBackup : IJob\n{\n public DbBackup(ILogger logger)\n { }\n}\n\ncontainer.Register("Console");\ncontainer.Register("File");\n\n// registration of service with the dependency binding.\ncontainer.Register(options => options\n .WithDependencyBinding("logger", "Console"));\n\n// the container will resolve DbBackup with ConsoleLogger.\nIJob job = container.Resolve();\n'))),(0,a.kt)(l.Z,{value:"Bind to property / field",label:"Bind to property / field",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},'class DbBackup : IJob\n{\n public ILogger Logger { get; set; }\n}\n\ncontainer.Register("Console");\ncontainer.Register("File");\n\n// registration of service with the member injection.\ncontainer.Register(options => options\n .WithDependencyBinding("Logger", "Console"));\n\n// the container will resolve DbBackup with ConsoleLogger.\nIJob job = container.Resolve();\n')))))),(0,a.kt)("h2",{id:"conventional-resolution"},"Conventional resolution"),(0,a.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"When you enable conventional resolution, the container treats member and method parameter names as their dependency identifier. "),(0,a.kt)("p",null,"It's like an implicit dependency binding on every class member."),(0,a.kt)("p",null,"First, you have to enable conventional resolution through the configuration of the container: "),(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .TreatParameterAndMemberNameAsDependencyName());\n")),(0,a.kt)("admonition",{type:"note"},(0,a.kt)("p",{parentName:"admonition"},"The container will attempt a ",(0,a.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#named-resolution"},"named resolution")," on each dependency based on their parameter or property/field name."))),(0,a.kt)("div",null,(0,a.kt)(i.Z,{mdxType:"Tabs"},(0,a.kt)(l.Z,{value:"Parameters",label:"Parameters",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},'class DbBackup : IJob\n{\n public DbBackup(\n // the parameter name identifies the dependency.\n ILogger consoleLogger)\n { }\n}\n\ncontainer.Register("consoleLogger");\ncontainer.Register("fileLogger");\n\ncontainer.Register();\n\n// the container will resolve DbBackup with ConsoleLogger.\nIJob job = container.Resolve();\n'))),(0,a.kt)(l.Z,{value:"Properties / fields",label:"Properties / fields",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},'class DbBackup : IJob\n{\n // the property name identifies the dependency.\n public ILogger ConsoleLogger { get; set; }\n}\n\ncontainer.Register("ConsoleLogger");\ncontainer.Register("FileLogger");\n\n// registration of service with auto member injection.\ncontainer.Register(options => options\n .WithAutoMemberInjection());\n\n// the container will resolve DbBackup with ConsoleLogger.\nIJob job = container.Resolve();\n')))))),(0,a.kt)("h2",{id:"conditional-resolution"},"Conditional resolution"),(0,a.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"Stashbox can resolve a particular dependency based on its context. This context is typically the reflected type of dependency, its usage, and the type it gets injected into."),(0,a.kt)("ul",null,(0,a.kt)("li",{parentName:"ul"},(0,a.kt)("p",{parentName:"li"},(0,a.kt)("strong",{parentName:"p"},"Attribute"),": you can filter on constructor, method, property, or field attributes to select the desired dependency for your service. In contrast to the ",(0,a.kt)("inlineCode",{parentName:"p"},"Dependency")," attribute, this configuration doesn't tie your application to Stashbox because you use your own attributes.")),(0,a.kt)("li",{parentName:"ul"},(0,a.kt)("p",{parentName:"li"},(0,a.kt)("strong",{parentName:"p"},"Parent type"),": you can filter on what type the given service is injected into.")),(0,a.kt)("li",{parentName:"ul"},(0,a.kt)("p",{parentName:"li"},(0,a.kt)("strong",{parentName:"p"},"Resolution path"),": similar to the parent type and attribute condition but extended with inheritance. You can set that the given service is only usable in a type's resolution path. This means that each direct and sub-dependency of the selected type must use the provided service as a dependency.")),(0,a.kt)("li",{parentName:"ul"},(0,a.kt)("p",{parentName:"li"},(0,a.kt)("strong",{parentName:"p"},"Custom"),": with this, you can build your own selection logic based on the given contextual type information.")))),(0,a.kt)("div",null,(0,a.kt)(i.Z,{mdxType:"Tabs"},(0,a.kt)(l.Z,{value:"Attribute",label:"Attribute",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"class ConsoleAttribute : Attribute { }\n\nclass DbBackup : IJob\n{\n public DbBackup([Console]ILogger logger)\n { }\n}\n\ncontainer.Register(options => options\n // resolve only when the injected parameter, \n // property or field has the 'Console' attribute\n .WhenHas());\n\ncontainer.Register();\n\n// the container will resolve DbBackup with ConsoleLogger.\nIJob job = container.Resolve();\n"))),(0,a.kt)(l.Z,{value:"Parent",label:"Parent",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"class DbBackup : IJob\n{\n public DbBackup(ILogger logger)\n { }\n}\n\ncontainer.Register(options => options\n // inject only when we are \n // currently resolving DbBackup OR StorageCleanup.\n .WhenDependantIs()\n .WhenDependantIs());\n\ncontainer.Register();\n\n// the container will resolve DbBackup with ConsoleLogger.\nIJob job = container.Resolve();\n"))),(0,a.kt)(l.Z,{value:"Path",label:"Path",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"class DbBackup : IJob\n{\n public DbBackup(IStorage storage)\n { }\n}\n\nclass FileStorage : IStorage\n{\n public FileStorage(ILogger logger) \n { }\n}\n\ncontainer.Register(options => options\n // inject only when we are in the\n // resolution path of DbBackup\n .WhenInResolutionPathOf());\n\ncontainer.Register();\ncontainer.Register();\n\n// the container will select ConsoleLogger for FileStorage\n// because they are injected into DbBackup.\nIJob job = container.Resolve();\n"))),(0,a.kt)(l.Z,{value:"Custom",label:"Custom",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"class DbBackup : IJob\n{\n public DbBackup(ILogger logger)\n { }\n}\n\ncontainer.Register(options => options\n // inject only when we are \n // currently resolving DbBackup.\n .When(typeInfo => typeInfo.ParentType.Equals(typeof(DbBackup))));\n\ncontainer.Register();\n\n// the container will resolve DbBackup with ConsoleLogger.\nIJob job = container.Resolve();\n"))),(0,a.kt)(l.Z,{value:"Collection",label:"Collection",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"class DbJobsExecutor : IJobsExecutor\n{\n public DbBackup(IEnumerable jobs)\n { }\n}\n\ncontainer.Register(options => options\n .WhenDependantIs());\ncontainer.Register(options => options\n .WhenDependantIs());\nontainer.Register();\n\ncontainer.Register();\n\n// jobsExecutor will get DbBackup and DbCleanup within a collection.\nIJobsExecutor jobsExecutor = container.Resolve();\n")))))),(0,a.kt)("p",null,"The specified conditions are behaving like filters when a ",(0,a.kt)("strong",{parentName:"p"},"collection")," is requested."),(0,a.kt)("p",null,"When you use the same conditional option multiple times, the container will evaluate them ",(0,a.kt)("strong",{parentName:"p"},"with OR")," logical operator."),(0,a.kt)("admonition",{type:"tip"},(0,a.kt)("p",{parentName:"admonition"},(0,a.kt)("a",{parentName:"p",href:"/docs/configuration/registration-configuration#conditions"},"Here")," you can find each condition related registration option.")),(0,a.kt)("h2",{id:"optional-resolution"},"Optional resolution"),(0,a.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"In cases where it's not guaranteed that a service is resolvable, either because it's not registered or any of its dependencies are missing, you can attempt an optional resolution using the ",(0,a.kt)("inlineCode",{parentName:"p"},"ResolveOrDefault()")," method. "),(0,a.kt)("p",null,"When the resolution attempt fails, it will return ",(0,a.kt)("inlineCode",{parentName:"p"},"null")," (or ",(0,a.kt)("inlineCode",{parentName:"p"},"default")," in case of value types).")),(0,a.kt)("div",null,(0,a.kt)(i.Z,{groupId:"generic-runtime-apis",mdxType:"Tabs"},(0,a.kt)(l.Z,{value:"Generic API",label:"Generic API",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"// returns null when the resolution fails.\nIJob job = container.ResolveOrDefault();\n\n// throws ResolutionFailedException when the resolution fails.\nIJob job = container.Resolve();\n"))),(0,a.kt)(l.Z,{value:"Runtime type API",label:"Runtime type API",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"// returns null when the resolution fails.\nobject job = container.ResolveOrDefault(typeof(IJob));\n\n// throws ResolutionFailedException when the resolution fails.\nobject job = container.Resolve(typeof(IJob));\n")))))),(0,a.kt)("h2",{id:"dependency-overrides"},"Dependency overrides"),(0,a.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"At resolution time, you can override a service's dependencies by passing an ",(0,a.kt)("inlineCode",{parentName:"p"},"object[]")," to the ",(0,a.kt)("inlineCode",{parentName:"p"},"Resolve()")," method."),(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"class DbBackup : IJob\n{\n public DbBackup(ILogger logger)\n { }\n}\n"))),(0,a.kt)("div",null,(0,a.kt)(i.Z,{groupId:"generic-runtime-apis",mdxType:"Tabs"},(0,a.kt)(l.Z,{value:"Generic API",label:"Generic API",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"DbBackup backup = container.Resolve( \n dependencyOverrides: new object[] \n { \n new ConsoleLogger() \n });\n"))),(0,a.kt)(l.Z,{value:"Runtime type API",label:"Runtime type API",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"object backup = container.Resolve(typeof(DbBackup),\n dependencyOverrides: new object[] \n { \n new ConsoleLogger() \n });\n")))))),(0,a.kt)("h2",{id:"activation"},"Activation"),(0,a.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"You can use the container's ",(0,a.kt)("inlineCode",{parentName:"p"},".Activate()")," method when you only want to build up an instance from a type on the fly without registration."),(0,a.kt)("p",null,"It allows dependency overriding with ",(0,a.kt)("inlineCode",{parentName:"p"},"object")," arguments and performs property/field/method injection (when configured)."),(0,a.kt)("p",null,"It works like ",(0,a.kt)("inlineCode",{parentName:"p"},"Activator.CreateInstance()")," except that Stashbox supplies the dependencies.")),(0,a.kt)("div",null,(0,a.kt)(i.Z,{groupId:"generic-runtime-apis",mdxType:"Tabs"},(0,a.kt)(l.Z,{value:"Generic API",label:"Generic API",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"// use dependency injected by container.\nDbBackup backup = container.Activate();\n\n// override the injected dependency.\nDbBackup backup = container.Activate(new ConsoleLogger());\n"))),(0,a.kt)(l.Z,{value:"Runtime type API",label:"Runtime type API",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"// use dependency injected by container.\nobject backup = container.Activate(typeof(DbBackup));\n\n// override the injected dependency.\nobject backup = container.Activate(typeof(DbBackup), new ConsoleLogger());\n")))))),(0,a.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("h3",{id:"build-up"},"Build-up"),(0,a.kt)("p",null,"With the ",(0,a.kt)("inlineCode",{parentName:"p"},".BuildUp()")," method, you can do the same ",(0,a.kt)("em",{parentName:"p"},"on the fly")," post-processing (property/field/method injection) on already constructed instances. "),(0,a.kt)("admonition",{type:"caution"},(0,a.kt)("p",{parentName:"admonition"},(0,a.kt)("inlineCode",{parentName:"p"},".BuildUp()")," won't register the given instance into the container."))),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"class DbBackup : IJob\n{\n public ILogger Logger { get; set; }\n}\n\nDbBackup backup = new DbBackup();\n// the container fills the Logger property.\ncontainer.BuildUp(backup); \n")))))}b.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/026a065b.d6d30dca.js b/assets/js/026a065b.d6d30dca.js new file mode 100644 index 00000000..6b5b4616 --- /dev/null +++ b/assets/js/026a065b.d6d30dca.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkwebsite=self.webpackChunkwebsite||[]).push([[340],{3905:(e,n,t)=>{t.d(n,{Zo:()=>u,kt:()=>b});var o=t(7294);function a(e,n,t){return n in e?Object.defineProperty(e,n,{value:t,enumerable:!0,configurable:!0,writable:!0}):e[n]=t,e}function r(e,n){var t=Object.keys(e);if(Object.getOwnPropertySymbols){var o=Object.getOwnPropertySymbols(e);n&&(o=o.filter((function(n){return Object.getOwnPropertyDescriptor(e,n).enumerable}))),t.push.apply(t,o)}return t}function i(e){for(var n=1;n=0||(a[t]=e[t]);return a}(e,n);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);for(o=0;o=0||Object.prototype.propertyIsEnumerable.call(e,t)&&(a[t]=e[t])}return a}var s=o.createContext({}),c=function(e){var n=o.useContext(s),t=n;return e&&(t="function"==typeof e?e(n):i(i({},n),e)),t},u=function(e){var n=c(e.components);return o.createElement(s.Provider,{value:n},e.children)},p="mdxType",d={inlineCode:"code",wrapper:function(e){var n=e.children;return o.createElement(o.Fragment,{},n)}},g=o.forwardRef((function(e,n){var t=e.components,a=e.mdxType,r=e.originalType,s=e.parentName,u=l(e,["components","mdxType","originalType","parentName"]),p=c(t),g=a,b=p["".concat(s,".").concat(g)]||p[g]||d[g]||r;return t?o.createElement(b,i(i({ref:n},u),{},{components:t})):o.createElement(b,i({ref:n},u))}));function b(e,n){var t=arguments,a=n&&n.mdxType;if("string"==typeof e||a){var r=t.length,i=new Array(r);i[0]=g;var l={};for(var s in n)hasOwnProperty.call(n,s)&&(l[s]=n[s]);l.originalType=e,l[p]="string"==typeof e?e:a,i[1]=l;for(var c=2;c{t.d(n,{Z:()=>i});var o=t(7294),a=t(6010);const r="tabItem_Ymn6";function i(e){let{children:n,hidden:t,className:i}=e;return o.createElement("div",{role:"tabpanel",className:(0,a.Z)(r,i),hidden:t},n)}},4866:(e,n,t)=>{t.d(n,{Z:()=>N});var o=t(7462),a=t(7294),r=t(6010),i=t(2466),l=t(6550),s=t(1980),c=t(7392),u=t(12);function p(e){return function(e){return a.Children.map(e,(e=>{if(!e||(0,a.isValidElement)(e)&&function(e){const{props:n}=e;return!!n&&"object"==typeof n&&"value"in n}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}(e).map((e=>{let{props:{value:n,label:t,attributes:o,default:a}}=e;return{value:n,label:t,attributes:o,default:a}}))}function d(e){const{values:n,children:t}=e;return(0,a.useMemo)((()=>{const e=n??p(t);return function(e){const n=(0,c.l)(e,((e,n)=>e.value===n.value));if(n.length>0)throw new Error(`Docusaurus error: Duplicate values "${n.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[n,t])}function g(e){let{value:n,tabValues:t}=e;return t.some((e=>e.value===n))}function b(e){let{queryString:n=!1,groupId:t}=e;const o=(0,l.k6)(),r=function(e){let{queryString:n=!1,groupId:t}=e;if("string"==typeof n)return n;if(!1===n)return null;if(!0===n&&!t)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return t??null}({queryString:n,groupId:t});return[(0,s._X)(r),(0,a.useCallback)((e=>{if(!r)return;const n=new URLSearchParams(o.location.search);n.set(r,e),o.replace({...o.location,search:n.toString()})}),[r,o])]}function m(e){const{defaultValue:n,queryString:t=!1,groupId:o}=e,r=d(e),[i,l]=(0,a.useState)((()=>function(e){let{defaultValue:n,tabValues:t}=e;if(0===t.length)throw new Error("Docusaurus error: the component requires at least one children component");if(n){if(!g({value:n,tabValues:t}))throw new Error(`Docusaurus error: The has a defaultValue "${n}" but none of its children has the corresponding value. Available values are: ${t.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return n}const o=t.find((e=>e.default))??t[0];if(!o)throw new Error("Unexpected error: 0 tabValues");return o.value}({defaultValue:n,tabValues:r}))),[s,c]=b({queryString:t,groupId:o}),[p,m]=function(e){let{groupId:n}=e;const t=function(e){return e?`docusaurus.tab.${e}`:null}(n),[o,r]=(0,u.Nk)(t);return[o,(0,a.useCallback)((e=>{t&&r.set(e)}),[t,r])]}({groupId:o}),k=(()=>{const e=s??p;return g({value:e,tabValues:r})?e:null})();(0,a.useLayoutEffect)((()=>{k&&l(k)}),[k]);return{selectedValue:i,selectValue:(0,a.useCallback)((e=>{if(!g({value:e,tabValues:r}))throw new Error(`Can't select invalid tab value=${e}`);l(e),c(e),m(e)}),[c,m,r]),tabValues:r}}var k=t(2389);const h="tabList__CuJ",y="tabItem_LNqP";function v(e){let{className:n,block:t,selectedValue:l,selectValue:s,tabValues:c}=e;const u=[],{blockElementScrollPositionUntilNextRender:p}=(0,i.o5)(),d=e=>{const n=e.currentTarget,t=u.indexOf(n),o=c[t].value;o!==l&&(p(n),s(o))},g=e=>{let n=null;switch(e.key){case"Enter":d(e);break;case"ArrowRight":{const t=u.indexOf(e.currentTarget)+1;n=u[t]??u[0];break}case"ArrowLeft":{const t=u.indexOf(e.currentTarget)-1;n=u[t]??u[u.length-1];break}}n?.focus()};return a.createElement("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,r.Z)("tabs",{"tabs--block":t},n)},c.map((e=>{let{value:n,label:t,attributes:i}=e;return a.createElement("li",(0,o.Z)({role:"tab",tabIndex:l===n?0:-1,"aria-selected":l===n,key:n,ref:e=>u.push(e),onKeyDown:g,onClick:d},i,{className:(0,r.Z)("tabs__item",y,i?.className,{"tabs__item--active":l===n})}),t??n)})))}function f(e){let{lazy:n,children:t,selectedValue:o}=e;const r=(Array.isArray(t)?t:[t]).filter(Boolean);if(n){const e=r.find((e=>e.props.value===o));return e?(0,a.cloneElement)(e,{className:"margin-top--md"}):null}return a.createElement("div",{className:"margin-top--md"},r.map(((e,n)=>(0,a.cloneElement)(e,{key:n,hidden:e.props.value!==o}))))}function I(e){const n=m(e);return a.createElement("div",{className:(0,r.Z)("tabs-container",h)},a.createElement(v,(0,o.Z)({},e,n)),a.createElement(f,(0,o.Z)({},e,n)))}function N(e){const n=(0,k.Z)();return a.createElement(I,(0,o.Z)({key:String(n)},e))}},8846:(e,n,t)=>{t.d(n,{Z:()=>l});var o=t(7294);const a="codeDescContainer_ie8f",r="desc_jyqI",i="example_eYlF";function l(e){let{children:n}=e,t=o.Children.toArray(n).filter((e=>e));return o.createElement("div",{className:a},o.createElement("div",{className:r},t[0]),o.createElement("div",{className:i},t[1]))}},385:(e,n,t)=>{t.r(n),t.d(n,{assets:()=>p,contentTitle:()=>c,default:()=>b,frontMatter:()=>s,metadata:()=>u,toc:()=>d});var o=t(7462),a=(t(7294),t(3905)),r=t(8846),i=t(4866),l=t(5162);const s={},c="Service resolution",u={unversionedId:"guides/service-resolution",id:"guides/service-resolution",title:"Service resolution",description:"When you have all your components registered and configured adequately, you can resolve them from the container or a scope by requesting their service type.",source:"@site/docs/guides/service-resolution.md",sourceDirName:"guides",slug:"/guides/service-resolution",permalink:"/stashbox/docs/guides/service-resolution",draft:!1,editUrl:"https://github.com/z4kn4fein/stashbox/edit/master/docs/docs/guides/service-resolution.md",tags:[],version:"current",lastUpdatedBy:"dependabot[bot]",lastUpdatedAt:1699538162,formattedLastUpdatedAt:"Nov 9, 2023",frontMatter:{},sidebar:"docs",previous:{title:"Advanced registration",permalink:"/stashbox/docs/guides/advanced-registration"},next:{title:"Lifetimes",permalink:"/stashbox/docs/guides/lifetimes"}},p={},d=[{value:"Injection patterns",id:"injection-patterns",level:2},{value:"Attributes",id:"attributes",level:2},{value:"Dependency binding",id:"dependency-binding",level:2},{value:"Conventional resolution",id:"conventional-resolution",level:2},{value:"Conditional resolution",id:"conditional-resolution",level:2},{value:"Optional resolution",id:"optional-resolution",level:2},{value:"Dependency overrides",id:"dependency-overrides",level:2},{value:"Activation",id:"activation",level:2},{value:"Build-up",id:"build-up",level:3}],g={toc:d};function b(e){let{components:n,...t}=e;return(0,a.kt)("wrapper",(0,o.Z)({},g,t,{components:n,mdxType:"MDXLayout"}),(0,a.kt)("h1",{id:"service-resolution"},"Service resolution"),(0,a.kt)("p",null,"When you have all your components registered and configured adequately, you can resolve them from the container or a ",(0,a.kt)("a",{parentName:"p",href:"/docs/guides/scopes"},"scope")," by requesting their ",(0,a.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type"),"."),(0,a.kt)("p",null,"During a service's resolution, the container walks through the entire ",(0,a.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#resolution-tree"},"resolution tree")," and instantiates all dependencies required for the service construction.\nWhen the container encounters any violations of ",(0,a.kt)("a",{parentName:"p",href:"/docs/diagnostics/validation#resolution-validation"},"these rules")," ",(0,a.kt)("em",{parentName:"p"},"(circular dependencies, missing required services, lifetime misconfigurations)")," during the walkthrough, it lets you know that something is wrong by throwing a specific exception."),(0,a.kt)("h2",{id:"injection-patterns"},"Injection patterns"),(0,a.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,(0,a.kt)("strong",{parentName:"p"},"Constructor injection")," is the ",(0,a.kt)("em",{parentName:"p"},"primary dependency injection pattern"),". It encourages the organization of dependencies to a single place - the constructor."),(0,a.kt)("p",null,"Stashbox, by default, uses the constructor that has the most parameters it knows how to resolve. This behavior is configurable through ",(0,a.kt)("a",{parentName:"p",href:"/docs/configuration/registration-configuration#constructor-selection"},"constructor selection"),"."),(0,a.kt)("p",null,(0,a.kt)("a",{parentName:"p",href:"/docs/configuration/registration-configuration#property-field-injection"},"Property/field injection")," is also supported in cases where constructor injection is not applicable."),(0,a.kt)("admonition",{type:"info"},(0,a.kt)("p",{parentName:"admonition"},(0,a.kt)("a",{parentName:"p",href:"/docs/configuration/container-configuration#constructor-selection"},"Constructor selection")," and ",(0,a.kt)("a",{parentName:"p",href:"/docs/configuration/container-configuration#auto-member-injection"},"property/field injection")," is also configurable container-wide."))),(0,a.kt)("div",null,(0,a.kt)(i.Z,{mdxType:"Tabs"},(0,a.kt)(l.Z,{value:"Constructor injection",label:"Constructor injection",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"class DbBackup : IJob\n{\n private readonly ILogger logger;\n private readonly IEventBroadcaster eventBroadcaster;\n\n public DbBackup(ILogger logger, IEventBroadcaster eventBroadcaster)\n {\n this.logger = logger;\n this.eventBroadcaster = eventBroadcaster;\n }\n}\n\ncontainer.Register();\ncontainer.Register();\n\ncontainer.Register();\n\n// resolution using the available constructor.\nIJob job = container.Resolve();\n"))),(0,a.kt)(l.Z,{value:"Property/field injection",label:"Property/field injection",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"class DbBackup : IJob\n{\n public ILogger Logger { get; set; }\n public IEventBroadcaster EventBroadcaster { get; set; }\n\n public DbBackup() \n { }\n}\n\ncontainer.Register();\ncontainer.Register();\n\n// registration of service with auto member injection.\ncontainer.Register(options => \n options.WithAutoMemberInjection());\n\n// resolution will inject the properties.\nIJob job = container.Resolve();\n")))))),(0,a.kt)("admonition",{type:"caution"},(0,a.kt)("p",{parentName:"admonition"},"It's a common mistake to use the ",(0,a.kt)("em",{parentName:"p"},"property/field injection")," only to disencumber the constructor from having too many parameters. That's a code smell and also violates the ",(0,a.kt)("a",{parentName:"p",href:"https://en.wikipedia.org/wiki/Single-responsibility_principle"},"Single-responsibility principle"),". If you recognize these conditions, you should consider splitting your class into multiple smaller units rather than adding an extra property-injected dependency. ")),(0,a.kt)("h2",{id:"attributes"},"Attributes"),(0,a.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"Attributes can give you control over how Stashbox selects dependencies for a service's resolution."),(0,a.kt)("p",null,(0,a.kt)("strong",{parentName:"p"},"Dependency attribute"),": "),(0,a.kt)("ul",null,(0,a.kt)("li",{parentName:"ul"},(0,a.kt)("p",{parentName:"li"},(0,a.kt)("strong",{parentName:"p"},"On a constructor/method parameter"),": used with the ",(0,a.kt)("em",{parentName:"p"},"name")," property, it works as a marker for ",(0,a.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#named-resolution"},"named resolution"),".")),(0,a.kt)("li",{parentName:"ul"},(0,a.kt)("p",{parentName:"li"},(0,a.kt)("strong",{parentName:"p"},"On a property/field"),": first, it enables ",(0,a.kt)("em",{parentName:"p"},"auto-injection")," on the marked property/field (even if it wasn't configured at registration explicitly), and just as with the method parameter, it allows ",(0,a.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#named-resolution"},"named resolution"),"."))),(0,a.kt)("p",null,(0,a.kt)("strong",{parentName:"p"},"InjectionMethod attribute"),": marks a method to be called when the requested service is instantiated.")),(0,a.kt)("div",null,(0,a.kt)(i.Z,{mdxType:"Tabs"},(0,a.kt)(l.Z,{value:"Constructor",label:"Constructor",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},'class DbBackup : IJob\n{\n private readonly ILogger logger;\n\n public DbBackup([Dependency("Console")]ILogger logger)\n {\n this.logger = logger;\n }\n}\n\ncontainer.Register("Console");\ncontainer.Register("File");\n\ncontainer.Register();\n\n// the container will resolve DbBackup with ConsoleLogger.\nIJob job = container.Resolve();\n'))),(0,a.kt)(l.Z,{value:"Property/field",label:"Property/field",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},'class DbBackup : IJob\n{\n [Dependency("Console")]\n public ILogger Logger { get; set; }\n\n public DbBackup() \n { }\n}\n\ncontainer.Register("Console");\ncontainer.Register("File");\n\ncontainer.Register();\n\n// the container will resolve DbBackup with ConsoleLogger.\nIJob job = container.Resolve();\n'))),(0,a.kt)(l.Z,{value:"Method",label:"Method",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},'class DbBackup : IJob\n{\n [InjectionMethod]\n public void Initialize([Dependency("Console")]ILogger logger)\n {\n this.logger.Log("Initializing.");\n }\n}\n\ncontainer.Register("Console");\ncontainer.Register("File");\n\ncontainer.Register();\n\n// the container will call DbBackup\'s Initialize method.\nIJob job = container.Resolve();\n')))))),(0,a.kt)("admonition",{type:"caution"},(0,a.kt)("p",{parentName:"admonition"},"Attributes provide a more straightforward configuration, but using them also tightens the bond between your application and Stashbox. If you consider this an issue, the same functionality is available on the ",(0,a.kt)("em",{parentName:"p"},"registration API")," as ",(0,a.kt)("a",{parentName:"p",href:"/docs/guides/service-resolution#dependency-binding"},"dependency binding"),".")),(0,a.kt)("h2",{id:"dependency-binding"},"Dependency binding"),(0,a.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"The same dependency configuration as attributes is available on the registration configuration API."),(0,a.kt)("ul",null,(0,a.kt)("li",{parentName:"ul"},(0,a.kt)("p",{parentName:"li"},(0,a.kt)("strong",{parentName:"p"},"Bind to parameter"),": it has the same functionality as the ",(0,a.kt)("a",{parentName:"p",href:"/docs/guides/service-resolution#attributes"},"Dependency attribute")," on a constructor or method parameter, enabling the ",(0,a.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#named-resolution"},"named resolution"),".")),(0,a.kt)("li",{parentName:"ul"},(0,a.kt)("p",{parentName:"li"},(0,a.kt)("strong",{parentName:"p"},"Bind to property/field"),": it has the same functionality as the ",(0,a.kt)("a",{parentName:"p",href:"/docs/guides/service-resolution#attributes"},"Dependency attribute"),", enabling the injection of the given property/field."))),(0,a.kt)("admonition",{type:"info"},(0,a.kt)("p",{parentName:"admonition"},"There are further dependency binding options ",(0,a.kt)("a",{parentName:"p",href:"/docs/configuration/registration-configuration#dependency-configuration"},"available")," on the registration configuration API."))),(0,a.kt)("div",null,(0,a.kt)(i.Z,{mdxType:"Tabs"},(0,a.kt)(l.Z,{value:"Bind to parameter",label:"Bind to parameter",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},'class DbBackup : IJob\n{\n public DbBackup(ILogger logger)\n { }\n}\n\ncontainer.Register("Console");\ncontainer.Register("File");\n\n// registration of service with the dependency binding.\ncontainer.Register(options => options\n .WithDependencyBinding("logger", "Console"));\n\n// the container will resolve DbBackup with ConsoleLogger.\nIJob job = container.Resolve();\n'))),(0,a.kt)(l.Z,{value:"Bind to property / field",label:"Bind to property / field",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},'class DbBackup : IJob\n{\n public ILogger Logger { get; set; }\n}\n\ncontainer.Register("Console");\ncontainer.Register("File");\n\n// registration of service with the member injection.\ncontainer.Register(options => options\n .WithDependencyBinding("Logger", "Console"));\n\n// the container will resolve DbBackup with ConsoleLogger.\nIJob job = container.Resolve();\n')))))),(0,a.kt)("h2",{id:"conventional-resolution"},"Conventional resolution"),(0,a.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"When you enable conventional resolution, the container treats member and method parameter names as their dependency identifier. "),(0,a.kt)("p",null,"It's like an implicit dependency binding on every class member."),(0,a.kt)("p",null,"First, you have to enable conventional resolution through the configuration of the container: "),(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .TreatParameterAndMemberNameAsDependencyName());\n")),(0,a.kt)("admonition",{type:"note"},(0,a.kt)("p",{parentName:"admonition"},"The container will attempt a ",(0,a.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#named-resolution"},"named resolution")," on each dependency based on their parameter or property/field name."))),(0,a.kt)("div",null,(0,a.kt)(i.Z,{mdxType:"Tabs"},(0,a.kt)(l.Z,{value:"Parameters",label:"Parameters",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},'class DbBackup : IJob\n{\n public DbBackup(\n // the parameter name identifies the dependency.\n ILogger consoleLogger)\n { }\n}\n\ncontainer.Register("consoleLogger");\ncontainer.Register("fileLogger");\n\ncontainer.Register();\n\n// the container will resolve DbBackup with ConsoleLogger.\nIJob job = container.Resolve();\n'))),(0,a.kt)(l.Z,{value:"Properties / fields",label:"Properties / fields",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},'class DbBackup : IJob\n{\n // the property name identifies the dependency.\n public ILogger ConsoleLogger { get; set; }\n}\n\ncontainer.Register("ConsoleLogger");\ncontainer.Register("FileLogger");\n\n// registration of service with auto member injection.\ncontainer.Register(options => options\n .WithAutoMemberInjection());\n\n// the container will resolve DbBackup with ConsoleLogger.\nIJob job = container.Resolve();\n')))))),(0,a.kt)("h2",{id:"conditional-resolution"},"Conditional resolution"),(0,a.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"Stashbox can resolve a particular dependency based on its context. This context is typically the reflected type of dependency, its usage, and the type it gets injected into."),(0,a.kt)("ul",null,(0,a.kt)("li",{parentName:"ul"},(0,a.kt)("p",{parentName:"li"},(0,a.kt)("strong",{parentName:"p"},"Attribute"),": you can filter on constructor, method, property, or field attributes to select the desired dependency for your service. In contrast to the ",(0,a.kt)("inlineCode",{parentName:"p"},"Dependency")," attribute, this configuration doesn't tie your application to Stashbox because you use your own attributes.")),(0,a.kt)("li",{parentName:"ul"},(0,a.kt)("p",{parentName:"li"},(0,a.kt)("strong",{parentName:"p"},"Parent type"),": you can filter on what type the given service is injected into.")),(0,a.kt)("li",{parentName:"ul"},(0,a.kt)("p",{parentName:"li"},(0,a.kt)("strong",{parentName:"p"},"Resolution path"),": similar to the parent type and attribute condition but extended with inheritance. You can set that the given service is only usable in a type's resolution path. This means that each direct and sub-dependency of the selected type must use the provided service as a dependency.")),(0,a.kt)("li",{parentName:"ul"},(0,a.kt)("p",{parentName:"li"},(0,a.kt)("strong",{parentName:"p"},"Custom"),": with this, you can build your own selection logic based on the given contextual type information.")))),(0,a.kt)("div",null,(0,a.kt)(i.Z,{mdxType:"Tabs"},(0,a.kt)(l.Z,{value:"Attribute",label:"Attribute",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"class ConsoleAttribute : Attribute { }\n\nclass DbBackup : IJob\n{\n public DbBackup([Console]ILogger logger)\n { }\n}\n\ncontainer.Register(options => options\n // resolve only when the injected parameter, \n // property or field has the 'Console' attribute\n .WhenHas());\n\ncontainer.Register();\n\n// the container will resolve DbBackup with ConsoleLogger.\nIJob job = container.Resolve();\n"))),(0,a.kt)(l.Z,{value:"Parent",label:"Parent",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"class DbBackup : IJob\n{\n public DbBackup(ILogger logger)\n { }\n}\n\ncontainer.Register(options => options\n // inject only when we are \n // currently resolving DbBackup OR StorageCleanup.\n .WhenDependantIs()\n .WhenDependantIs());\n\ncontainer.Register();\n\n// the container will resolve DbBackup with ConsoleLogger.\nIJob job = container.Resolve();\n"))),(0,a.kt)(l.Z,{value:"Path",label:"Path",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"class DbBackup : IJob\n{\n public DbBackup(IStorage storage)\n { }\n}\n\nclass FileStorage : IStorage\n{\n public FileStorage(ILogger logger) \n { }\n}\n\ncontainer.Register(options => options\n // inject only when we are in the\n // resolution path of DbBackup\n .WhenInResolutionPathOf());\n\ncontainer.Register();\ncontainer.Register();\n\n// the container will select ConsoleLogger for FileStorage\n// because they are injected into DbBackup.\nIJob job = container.Resolve();\n"))),(0,a.kt)(l.Z,{value:"Custom",label:"Custom",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"class DbBackup : IJob\n{\n public DbBackup(ILogger logger)\n { }\n}\n\ncontainer.Register(options => options\n // inject only when we are \n // currently resolving DbBackup.\n .When(typeInfo => typeInfo.ParentType.Equals(typeof(DbBackup))));\n\ncontainer.Register();\n\n// the container will resolve DbBackup with ConsoleLogger.\nIJob job = container.Resolve();\n"))),(0,a.kt)(l.Z,{value:"Collection",label:"Collection",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"class DbJobsExecutor : IJobsExecutor\n{\n public DbBackup(IEnumerable jobs)\n { }\n}\n\ncontainer.Register(options => options\n .WhenDependantIs());\ncontainer.Register(options => options\n .WhenDependantIs());\nontainer.Register();\n\ncontainer.Register();\n\n// jobsExecutor will get DbBackup and DbCleanup within a collection.\nIJobsExecutor jobsExecutor = container.Resolve();\n")))))),(0,a.kt)("p",null,"The specified conditions are behaving like filters when a ",(0,a.kt)("strong",{parentName:"p"},"collection")," is requested."),(0,a.kt)("p",null,"When you use the same conditional option multiple times, the container will evaluate them ",(0,a.kt)("strong",{parentName:"p"},"with OR")," logical operator."),(0,a.kt)("admonition",{type:"tip"},(0,a.kt)("p",{parentName:"admonition"},(0,a.kt)("a",{parentName:"p",href:"/docs/configuration/registration-configuration#conditions"},"Here")," you can find each condition related registration option.")),(0,a.kt)("h2",{id:"optional-resolution"},"Optional resolution"),(0,a.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"In cases where it's not guaranteed that a service is resolvable, either because it's not registered or any of its dependencies are missing, you can attempt an optional resolution using the ",(0,a.kt)("inlineCode",{parentName:"p"},"ResolveOrDefault()")," method. "),(0,a.kt)("p",null,"When the resolution attempt fails, it will return ",(0,a.kt)("inlineCode",{parentName:"p"},"null")," (or ",(0,a.kt)("inlineCode",{parentName:"p"},"default")," in case of value types).")),(0,a.kt)("div",null,(0,a.kt)(i.Z,{groupId:"generic-runtime-apis",mdxType:"Tabs"},(0,a.kt)(l.Z,{value:"Generic API",label:"Generic API",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"// returns null when the resolution fails.\nIJob job = container.ResolveOrDefault();\n\n// throws ResolutionFailedException when the resolution fails.\nIJob job = container.Resolve();\n"))),(0,a.kt)(l.Z,{value:"Runtime type API",label:"Runtime type API",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"// returns null when the resolution fails.\nobject job = container.ResolveOrDefault(typeof(IJob));\n\n// throws ResolutionFailedException when the resolution fails.\nobject job = container.Resolve(typeof(IJob));\n")))))),(0,a.kt)("h2",{id:"dependency-overrides"},"Dependency overrides"),(0,a.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"At resolution time, you can override a service's dependencies by passing an ",(0,a.kt)("inlineCode",{parentName:"p"},"object[]")," to the ",(0,a.kt)("inlineCode",{parentName:"p"},"Resolve()")," method."),(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"class DbBackup : IJob\n{\n public DbBackup(ILogger logger)\n { }\n}\n"))),(0,a.kt)("div",null,(0,a.kt)(i.Z,{groupId:"generic-runtime-apis",mdxType:"Tabs"},(0,a.kt)(l.Z,{value:"Generic API",label:"Generic API",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"DbBackup backup = container.Resolve( \n dependencyOverrides: new object[] \n { \n new ConsoleLogger() \n });\n"))),(0,a.kt)(l.Z,{value:"Runtime type API",label:"Runtime type API",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"object backup = container.Resolve(typeof(DbBackup),\n dependencyOverrides: new object[] \n { \n new ConsoleLogger() \n });\n")))))),(0,a.kt)("h2",{id:"activation"},"Activation"),(0,a.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"You can use the container's ",(0,a.kt)("inlineCode",{parentName:"p"},".Activate()")," method when you only want to build up an instance from a type on the fly without registration."),(0,a.kt)("p",null,"It allows dependency overriding with ",(0,a.kt)("inlineCode",{parentName:"p"},"object")," arguments and performs property/field/method injection (when configured)."),(0,a.kt)("p",null,"It works like ",(0,a.kt)("inlineCode",{parentName:"p"},"Activator.CreateInstance()")," except that Stashbox supplies the dependencies.")),(0,a.kt)("div",null,(0,a.kt)(i.Z,{groupId:"generic-runtime-apis",mdxType:"Tabs"},(0,a.kt)(l.Z,{value:"Generic API",label:"Generic API",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"// use dependency injected by container.\nDbBackup backup = container.Activate();\n\n// override the injected dependency.\nDbBackup backup = container.Activate(new ConsoleLogger());\n"))),(0,a.kt)(l.Z,{value:"Runtime type API",label:"Runtime type API",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"// use dependency injected by container.\nobject backup = container.Activate(typeof(DbBackup));\n\n// override the injected dependency.\nobject backup = container.Activate(typeof(DbBackup), new ConsoleLogger());\n")))))),(0,a.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("h3",{id:"build-up"},"Build-up"),(0,a.kt)("p",null,"With the ",(0,a.kt)("inlineCode",{parentName:"p"},".BuildUp()")," method, you can do the same ",(0,a.kt)("em",{parentName:"p"},"on the fly")," post-processing (property/field/method injection) on already constructed instances. "),(0,a.kt)("admonition",{type:"caution"},(0,a.kt)("p",{parentName:"admonition"},(0,a.kt)("inlineCode",{parentName:"p"},".BuildUp()")," won't register the given instance into the container."))),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"class DbBackup : IJob\n{\n public ILogger Logger { get; set; }\n}\n\nDbBackup backup = new DbBackup();\n// the container fills the Logger property.\ncontainer.BuildUp(backup); \n")))))}b.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/115c3c33.97f612ab.js b/assets/js/115c3c33.97f612ab.js new file mode 100644 index 00000000..b9b972e9 --- /dev/null +++ b/assets/js/115c3c33.97f612ab.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkwebsite=self.webpackChunkwebsite||[]).push([[196],{3905:(e,t,n)=>{n.d(t,{Zo:()=>p,kt:()=>h});var a=n(7294);function r(e,t,n){return t in e?Object.defineProperty(e,t,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[t]=n,e}function o(e,t){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);t&&(a=a.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),n.push.apply(n,a)}return n}function i(e){for(var t=1;t=0||(r[n]=e[n]);return r}(e,t);if(Object.getOwnPropertySymbols){var o=Object.getOwnPropertySymbols(e);for(a=0;a=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(r[n]=e[n])}return r}var l=a.createContext({}),c=function(e){var t=a.useContext(l),n=t;return e&&(n="function"==typeof e?e(t):i(i({},t),e)),n},p=function(e){var t=c(e.components);return a.createElement(l.Provider,{value:t},e.children)},u="mdxType",d={inlineCode:"code",wrapper:function(e){var t=e.children;return a.createElement(a.Fragment,{},t)}},m=a.forwardRef((function(e,t){var n=e.components,r=e.mdxType,o=e.originalType,l=e.parentName,p=s(e,["components","mdxType","originalType","parentName"]),u=c(n),m=r,h=u["".concat(l,".").concat(m)]||u[m]||d[m]||o;return n?a.createElement(h,i(i({ref:t},p),{},{components:n})):a.createElement(h,i({ref:t},p))}));function h(e,t){var n=arguments,r=t&&t.mdxType;if("string"==typeof e||r){var o=n.length,i=new Array(o);i[0]=m;var s={};for(var l in t)hasOwnProperty.call(t,l)&&(s[l]=t[l]);s.originalType=e,s[u]="string"==typeof e?e:r,i[1]=s;for(var c=2;c{n.d(t,{Z:()=>i});var a=n(7294),r=n(6010);const o="tabItem_Ymn6";function i(e){let{children:t,hidden:n,className:i}=e;return a.createElement("div",{role:"tabpanel",className:(0,r.Z)(o,i),hidden:n},t)}},4866:(e,t,n)=>{n.d(t,{Z:()=>w});var a=n(7462),r=n(7294),o=n(6010),i=n(2466),s=n(6550),l=n(1980),c=n(7392),p=n(12);function u(e){return function(e){return r.Children.map(e,(e=>{if(!e||(0,r.isValidElement)(e)&&function(e){const{props:t}=e;return!!t&&"object"==typeof t&&"value"in t}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}(e).map((e=>{let{props:{value:t,label:n,attributes:a,default:r}}=e;return{value:t,label:n,attributes:a,default:r}}))}function d(e){const{values:t,children:n}=e;return(0,r.useMemo)((()=>{const e=t??u(n);return function(e){const t=(0,c.l)(e,((e,t)=>e.value===t.value));if(t.length>0)throw new Error(`Docusaurus error: Duplicate values "${t.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[t,n])}function m(e){let{value:t,tabValues:n}=e;return n.some((e=>e.value===t))}function h(e){let{queryString:t=!1,groupId:n}=e;const a=(0,s.k6)(),o=function(e){let{queryString:t=!1,groupId:n}=e;if("string"==typeof t)return t;if(!1===t)return null;if(!0===t&&!n)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return n??null}({queryString:t,groupId:n});return[(0,l._X)(o),(0,r.useCallback)((e=>{if(!o)return;const t=new URLSearchParams(a.location.search);t.set(o,e),a.replace({...a.location,search:t.toString()})}),[o,a])]}function g(e){const{defaultValue:t,queryString:n=!1,groupId:a}=e,o=d(e),[i,s]=(0,r.useState)((()=>function(e){let{defaultValue:t,tabValues:n}=e;if(0===n.length)throw new Error("Docusaurus error: the component requires at least one children component");if(t){if(!m({value:t,tabValues:n}))throw new Error(`Docusaurus error: The has a defaultValue "${t}" but none of its children has the corresponding value. Available values are: ${n.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return t}const a=n.find((e=>e.default))??n[0];if(!a)throw new Error("Unexpected error: 0 tabValues");return a.value}({defaultValue:t,tabValues:o}))),[l,c]=h({queryString:n,groupId:a}),[u,g]=function(e){let{groupId:t}=e;const n=function(e){return e?`docusaurus.tab.${e}`:null}(t),[a,o]=(0,p.Nk)(n);return[a,(0,r.useCallback)((e=>{n&&o.set(e)}),[n,o])]}({groupId:a}),v=(()=>{const e=l??u;return m({value:e,tabValues:o})?e:null})();(0,r.useLayoutEffect)((()=>{v&&s(v)}),[v]);return{selectedValue:i,selectValue:(0,r.useCallback)((e=>{if(!m({value:e,tabValues:o}))throw new Error(`Can't select invalid tab value=${e}`);s(e),c(e),g(e)}),[c,g,o]),tabValues:o}}var v=n(2389);const f="tabList__CuJ",y="tabItem_LNqP";function k(e){let{className:t,block:n,selectedValue:s,selectValue:l,tabValues:c}=e;const p=[],{blockElementScrollPositionUntilNextRender:u}=(0,i.o5)(),d=e=>{const t=e.currentTarget,n=p.indexOf(t),a=c[n].value;a!==s&&(u(t),l(a))},m=e=>{let t=null;switch(e.key){case"Enter":d(e);break;case"ArrowRight":{const n=p.indexOf(e.currentTarget)+1;t=p[n]??p[0];break}case"ArrowLeft":{const n=p.indexOf(e.currentTarget)-1;t=p[n]??p[p.length-1];break}}t?.focus()};return r.createElement("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,o.Z)("tabs",{"tabs--block":n},t)},c.map((e=>{let{value:t,label:n,attributes:i}=e;return r.createElement("li",(0,a.Z)({role:"tab",tabIndex:s===t?0:-1,"aria-selected":s===t,key:t,ref:e=>p.push(e),onKeyDown:m,onClick:d},i,{className:(0,o.Z)("tabs__item",y,i?.className,{"tabs__item--active":s===t})}),n??t)})))}function N(e){let{lazy:t,children:n,selectedValue:a}=e;const o=(Array.isArray(n)?n:[n]).filter(Boolean);if(t){const e=o.find((e=>e.props.value===a));return e?(0,r.cloneElement)(e,{className:"margin-top--md"}):null}return r.createElement("div",{className:"margin-top--md"},o.map(((e,t)=>(0,r.cloneElement)(e,{key:t,hidden:e.props.value!==a}))))}function b(e){const t=g(e);return r.createElement("div",{className:(0,o.Z)("tabs-container",f)},r.createElement(k,(0,a.Z)({},e,t)),r.createElement(N,(0,a.Z)({},e,t)))}function w(e){const t=(0,v.Z)();return r.createElement(b,(0,a.Z)({key:String(t)},e))}},8243:(e,t,n)=>{n.r(t),n.d(t,{assets:()=>p,contentTitle:()=>l,default:()=>m,frontMatter:()=>s,metadata:()=>c,toc:()=>u});var a=n(7462),r=(n(7294),n(3905)),o=n(4866),i=n(5162);const s={},l="Validation",c={unversionedId:"diagnostics/validation",id:"diagnostics/validation",title:"Validation",description:"Stashbox validation routines help you detect and solve common misconfiguration issues. You can verify the container's actual state with its .Validate() method. It walks through the whole resolution tree and collects all issues into an AggregateException.",source:"@site/docs/diagnostics/validation.md",sourceDirName:"diagnostics",slug:"/diagnostics/validation",permalink:"/stashbox/docs/diagnostics/validation",draft:!1,editUrl:"https://github.com/z4kn4fein/stashbox/edit/master/docs/docs/diagnostics/validation.md",tags:[],version:"current",lastUpdatedBy:"dependabot[bot]",lastUpdatedAt:1699538162,formattedLastUpdatedAt:"Nov 9, 2023",frontMatter:{},sidebar:"docs",previous:{title:"Special resolution cases",permalink:"/stashbox/docs/advanced/special-resolution-cases"},next:{title:"Utilities",permalink:"/stashbox/docs/diagnostics/utilities"}},p={},u=[{value:"Registration validation",id:"registration-validation",level:2},{value:"InvalidRegistrationException",id:"invalidregistrationexception",level:3},{value:"ServiceAlreadyRegisteredException",id:"servicealreadyregisteredexception",level:3},{value:"Resolution validation",id:"resolution-validation",level:2},{value:"Lifetime validation",id:"lifetime-validation",level:2},{value:"Circular dependency",id:"circular-dependency",level:2},{value:"Other exceptions",id:"other-exceptions",level:2},{value:"CompositionRootNotFoundException",id:"compositionrootnotfoundexception",level:3},{value:"ConstructorNotFoundException",id:"constructornotfoundexception",level:3}],d={toc:u};function m(e){let{components:t,...n}=e;return(0,r.kt)("wrapper",(0,a.Z)({},d,n,{components:t,mdxType:"MDXLayout"}),(0,r.kt)("h1",{id:"validation"},"Validation"),(0,r.kt)("p",null,"Stashbox validation routines help you detect and solve common misconfiguration issues. You can verify the container's actual state with its ",(0,r.kt)("inlineCode",{parentName:"p"},".Validate()")," method. It walks through the whole ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#resolution-tree"},"resolution tree")," and collects all issues into an ",(0,r.kt)("inlineCode",{parentName:"p"},"AggregateException"),"."),(0,r.kt)("h2",{id:"registration-validation"},"Registration validation"),(0,r.kt)("p",null,"During registration, the container validates the passed types and throws the following exceptions when the validation fails."),(0,r.kt)("h3",{id:"invalidregistrationexception"},"InvalidRegistrationException"),(0,r.kt)("ol",null,(0,r.kt)("li",{parentName:"ol"},(0,r.kt)("strong",{parentName:"li"},"When the ",(0,r.kt)("a",{parentName:"strong",href:"/docs/getting-started/glossary#service-type--implementation-type"},"implementation type")," is not resolvable.")," (it's an interface or an abstract class registered like: ",(0,r.kt)("inlineCode",{parentName:"li"},"Register()"),"):",(0,r.kt)("pre",{parentName:"li"},(0,r.kt)("code",{parentName:"pre"},"The type Namespace.IService could not be resolved. It's probably an interface, abstract class, or primitive type.\n"))),(0,r.kt)("li",{parentName:"ol"},(0,r.kt)("strong",{parentName:"li"},"When the ",(0,r.kt)("a",{parentName:"strong",href:"/docs/getting-started/glossary#service-type--implementation-type"},"implementation type")," does not implement the ",(0,r.kt)("a",{parentName:"strong",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type")),".",(0,r.kt)("pre",{parentName:"li"},(0,r.kt)("code",{parentName:"pre"},"The type Namespace.MotorCycle does not implement the '[service type](/docs/getting-started/glossary#service-type--implementation-type)' Namespace.ICar.\n")))),(0,r.kt)("h3",{id:"servicealreadyregisteredexception"},"ServiceAlreadyRegisteredException"),(0,r.kt)("p",null,(0,r.kt)("strong",{parentName:"p"},"When the given ",(0,r.kt)("a",{parentName:"strong",href:"/docs/getting-started/glossary#service-type--implementation-type"},"implementation type")," is already registered")," and the ",(0,r.kt)("inlineCode",{parentName:"p"},"RegistrationBehavior")," ",(0,r.kt)("a",{parentName:"p",href:"/docs/configuration/container-configuration#registration-behavior"},"container configuration option")," is set to ",(0,r.kt)("inlineCode",{parentName:"p"},"ThrowException"),":"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre"},"The type Namespace.Service is already registered.\n")),(0,r.kt)("h2",{id:"resolution-validation"},"Resolution validation"),(0,r.kt)("p",null,"During the ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#resolution-tree"},"resolution tree's")," construction, the container continuously checks its actual state to ensure stability. When any of the following issues occur, the container throws a ",(0,r.kt)("inlineCode",{parentName:"p"},"ResolutionFailedException"),"."),(0,r.kt)("ol",null,(0,r.kt)("li",{parentName:"ol"},(0,r.kt)("p",{parentName:"li"},(0,r.kt)("strong",{parentName:"p"},"When a dependency is missing from the ",(0,r.kt)("a",{parentName:"strong",href:"/docs/getting-started/glossary#resolution-tree"},"resolution tree")),"."),(0,r.kt)(o.Z,{mdxType:"Tabs"},(0,r.kt)(i.Z,{value:"Parameter",label:"Parameter",mdxType:"TabItem"},(0,r.kt)("pre",{parentName:"li"},(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"class Service\n{\n public Service(Dependency dep) { }\n\n public Service(Dependency2 dep2) { }\n}\n\ncontainer.Register();\nvar service = container.Resolve();\n")),(0,r.kt)("p",{parentName:"li"},"This will result in the following exception message:"),(0,r.kt)("pre",{parentName:"li"},(0,r.kt)("code",{parentName:"pre"},"Could not resolve type Namespace.Service.\nConstructor Void .ctor(Dependency) found with unresolvable parameter: (Namespace.Dependency)dep.\nConstructor Void .ctor(Dependency2) found with unresolvable parameter: (Namespace.Dependency2)dep2.\n"))),(0,r.kt)(i.Z,{value:"Property / field",label:"Property / field",mdxType:"TabItem"},(0,r.kt)("pre",{parentName:"li"},(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"class Service\n{\n public Dependency Dep { get; set; }\n}\n\ncontainer.Register(options => options.WithDependencyBinding(s => s.Dep));\nvar service = container.Resolve();\n")),(0,r.kt)("p",{parentName:"li"},"This will show the following message:"),(0,r.kt)("pre",{parentName:"li"},(0,r.kt)("code",{parentName:"pre"},"Could not resolve type Namespace.Service.\nUnresolvable property: (Namespace.Dependency)Dep.\n"))))),(0,r.kt)("li",{parentName:"ol"},(0,r.kt)("p",{parentName:"li"},(0,r.kt)("strong",{parentName:"p"},"When the requested type is unresolvable.")," E.g., it doesn't have a public constructor."),(0,r.kt)("pre",{parentName:"li"},(0,r.kt)("code",{parentName:"pre"},"Could not resolve type Namespace.Service.\nService is not registered or unresolvable type requested.\n")))),(0,r.kt)("h2",{id:"lifetime-validation"},"Lifetime validation"),(0,r.kt)("p",null,"This validation enforces the following rules. When they are violated, the container throws a ",(0,r.kt)("inlineCode",{parentName:"p"},"LifetimeValidationFailedException"),"."),(0,r.kt)("ol",null,(0,r.kt)("li",{parentName:"ol"},(0,r.kt)("p",{parentName:"li"},(0,r.kt)("strong",{parentName:"p"},"When a scoped service is requested from the ",(0,r.kt)("a",{parentName:"strong",href:"/docs/getting-started/glossary#root-scope"},"root scope")),".",(0,r.kt)("br",{parentName:"p"}),"\n","As the ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#root-scope"},"root scope's")," lifetime is bound to the container's lifetime, this action unintentionally promotes the scoped service's lifetime to singleton:"),(0,r.kt)("pre",{parentName:"li"},(0,r.kt)("code",{parentName:"pre"},"Resolution of Namespace.Service (ScopedLifetime) from the '[root scope](/docs/getting-started/glossary#root-scope)' is not allowed, \nthat would promote the service's lifetime to a singleton.\n"))),(0,r.kt)("li",{parentName:"ol"},(0,r.kt)("p",{parentName:"li"},(0,r.kt)("strong",{parentName:"p"},"When the life-span of a dependency is shorter than its parent's"),".",(0,r.kt)("br",{parentName:"p"}),"\n","It's called ",(0,r.kt)("a",{parentName:"p",href:"https://blog.ploeh.dk/2014/06/02/captive-dependency/"},"captive dependency"),". Every lifetime has a ",(0,r.kt)("inlineCode",{parentName:"p"},"LifeSpan")," value, which determines how long the related service lives. The main rule is that services may not contain dependencies with shorter life spans. E.g., singletons should not depend on scoped services. The only exception is the life span value ",(0,r.kt)("inlineCode",{parentName:"p"},"0"),", which indicates that the related service is state-less and could be injected into any service. "),(0,r.kt)("p",{parentName:"li"}," These are the ",(0,r.kt)("inlineCode",{parentName:"p"},"LifeSpan")," values of the pre-defined lifetimes: "),(0,r.kt)("ul",{parentName:"li"},(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("strong",{parentName:"li"},"Singleton"),": 20"),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("strong",{parentName:"li"},"Scoped"),": 10"),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("strong",{parentName:"li"},"NamedScope"),": 10"),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("strong",{parentName:"li"},"PerRequest"),": 0"),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("strong",{parentName:"li"},"PerScopedRequest"),": 0"),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("strong",{parentName:"li"},"Transient"),": 0")),(0,r.kt)("p",{parentName:"li"},"In case of a failed validation the exception message would be:"),(0,r.kt)("pre",{parentName:"li"},(0,r.kt)("code",{parentName:"pre"},"The life-span of Namespace.Service (ScopedLifetime|10) is shorter than \nits direct or indirect parent's Namespace.Dependency (Singleton|20). \nThis could lead to incidental lifetime promotions with longer life-span, \nit's recommended to double-check your lifetime configurations.\n")))),(0,r.kt)("h2",{id:"circular-dependency"},"Circular dependency"),(0,r.kt)("p",null,"When the container encounters a circular dependency loop in the ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#resolution-tree"},"resolution tree"),", it throws a ",(0,r.kt)("inlineCode",{parentName:"p"},"CircularDependencyException"),"."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"class Service1\n{\n public Service1(Service2 service2) { }\n}\n\nclass Service2\n{\n public Service2(Service1 service1) { }\n}\n\ncontainer.Register();\ncontainer.Register();\nvar service = container.Resolve();\n")),(0,r.kt)("p",null,"The exception message is: "),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre"},"Circular dependency detected during the resolution of Namespace.Service1.\n")),(0,r.kt)("h2",{id:"other-exceptions"},"Other exceptions"),(0,r.kt)("h3",{id:"compositionrootnotfoundexception"},"CompositionRootNotFoundException"),(0,r.kt)("p",null,"This exception pops up when we try to compose an assembly, but it doesn't contain an ",(0,r.kt)("inlineCode",{parentName:"p"},"ICompositionRoot")," implementation."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.ComposeAssembly(typeof(Service).Assembly);\n")),(0,r.kt)("p",null,"The exception message is: "),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre"},"No ICompositionRoot found in the given assembly: {your-assembly-name}\n")),(0,r.kt)("h3",{id:"constructornotfoundexception"},"ConstructorNotFoundException"),(0,r.kt)("p",null,"During the registration phase, when you are using the ",(0,r.kt)("a",{parentName:"p",href:"/docs/configuration/registration-configuration#withconstructorbyargumenttypes"},(0,r.kt)("inlineCode",{parentName:"a"},"WithConstructorByArgumentTypes()"))," or ",(0,r.kt)("a",{parentName:"p",href:"/docs/configuration/registration-configuration#withconstructorbyarguments"},(0,r.kt)("inlineCode",{parentName:"a"},"WithConstructorByArguments()"))," options, you can accidentally point to a non-existing constructor. In that case, the container throws a ",(0,r.kt)("inlineCode",{parentName:"p"},"ConstructorNotFoundException"),"."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"class Service\n{\n public Service(Dependency dep) { }\n}\n\ncontainer.Register(options => options.WithConstructorByArgumentTypes(typeof(string), typeof(int)));\n")),(0,r.kt)("p",null,"The exception message is: "),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre"},"Constructor not found for Namespace.Service with the given argument types: System.String, System.Int32.\n")))}m.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/115c3c33.d8066e20.js b/assets/js/115c3c33.d8066e20.js deleted file mode 100644 index 67fd13fd..00000000 --- a/assets/js/115c3c33.d8066e20.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkwebsite=self.webpackChunkwebsite||[]).push([[196],{3905:(e,t,n)=>{n.d(t,{Zo:()=>p,kt:()=>h});var a=n(7294);function r(e,t,n){return t in e?Object.defineProperty(e,t,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[t]=n,e}function o(e,t){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);t&&(a=a.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),n.push.apply(n,a)}return n}function i(e){for(var t=1;t=0||(r[n]=e[n]);return r}(e,t);if(Object.getOwnPropertySymbols){var o=Object.getOwnPropertySymbols(e);for(a=0;a=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(r[n]=e[n])}return r}var l=a.createContext({}),c=function(e){var t=a.useContext(l),n=t;return e&&(n="function"==typeof e?e(t):i(i({},t),e)),n},p=function(e){var t=c(e.components);return a.createElement(l.Provider,{value:t},e.children)},u="mdxType",d={inlineCode:"code",wrapper:function(e){var t=e.children;return a.createElement(a.Fragment,{},t)}},m=a.forwardRef((function(e,t){var n=e.components,r=e.mdxType,o=e.originalType,l=e.parentName,p=s(e,["components","mdxType","originalType","parentName"]),u=c(n),m=r,h=u["".concat(l,".").concat(m)]||u[m]||d[m]||o;return n?a.createElement(h,i(i({ref:t},p),{},{components:n})):a.createElement(h,i({ref:t},p))}));function h(e,t){var n=arguments,r=t&&t.mdxType;if("string"==typeof e||r){var o=n.length,i=new Array(o);i[0]=m;var s={};for(var l in t)hasOwnProperty.call(t,l)&&(s[l]=t[l]);s.originalType=e,s[u]="string"==typeof e?e:r,i[1]=s;for(var c=2;c{n.d(t,{Z:()=>i});var a=n(7294),r=n(6010);const o="tabItem_Ymn6";function i(e){let{children:t,hidden:n,className:i}=e;return a.createElement("div",{role:"tabpanel",className:(0,r.Z)(o,i),hidden:n},t)}},4866:(e,t,n)=>{n.d(t,{Z:()=>w});var a=n(7462),r=n(7294),o=n(6010),i=n(2466),s=n(6550),l=n(1980),c=n(7392),p=n(12);function u(e){return function(e){return r.Children.map(e,(e=>{if(!e||(0,r.isValidElement)(e)&&function(e){const{props:t}=e;return!!t&&"object"==typeof t&&"value"in t}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}(e).map((e=>{let{props:{value:t,label:n,attributes:a,default:r}}=e;return{value:t,label:n,attributes:a,default:r}}))}function d(e){const{values:t,children:n}=e;return(0,r.useMemo)((()=>{const e=t??u(n);return function(e){const t=(0,c.l)(e,((e,t)=>e.value===t.value));if(t.length>0)throw new Error(`Docusaurus error: Duplicate values "${t.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[t,n])}function m(e){let{value:t,tabValues:n}=e;return n.some((e=>e.value===t))}function h(e){let{queryString:t=!1,groupId:n}=e;const a=(0,s.k6)(),o=function(e){let{queryString:t=!1,groupId:n}=e;if("string"==typeof t)return t;if(!1===t)return null;if(!0===t&&!n)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return n??null}({queryString:t,groupId:n});return[(0,l._X)(o),(0,r.useCallback)((e=>{if(!o)return;const t=new URLSearchParams(a.location.search);t.set(o,e),a.replace({...a.location,search:t.toString()})}),[o,a])]}function g(e){const{defaultValue:t,queryString:n=!1,groupId:a}=e,o=d(e),[i,s]=(0,r.useState)((()=>function(e){let{defaultValue:t,tabValues:n}=e;if(0===n.length)throw new Error("Docusaurus error: the component requires at least one children component");if(t){if(!m({value:t,tabValues:n}))throw new Error(`Docusaurus error: The has a defaultValue "${t}" but none of its children has the corresponding value. Available values are: ${n.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return t}const a=n.find((e=>e.default))??n[0];if(!a)throw new Error("Unexpected error: 0 tabValues");return a.value}({defaultValue:t,tabValues:o}))),[l,c]=h({queryString:n,groupId:a}),[u,g]=function(e){let{groupId:t}=e;const n=function(e){return e?`docusaurus.tab.${e}`:null}(t),[a,o]=(0,p.Nk)(n);return[a,(0,r.useCallback)((e=>{n&&o.set(e)}),[n,o])]}({groupId:a}),v=(()=>{const e=l??u;return m({value:e,tabValues:o})?e:null})();(0,r.useLayoutEffect)((()=>{v&&s(v)}),[v]);return{selectedValue:i,selectValue:(0,r.useCallback)((e=>{if(!m({value:e,tabValues:o}))throw new Error(`Can't select invalid tab value=${e}`);s(e),c(e),g(e)}),[c,g,o]),tabValues:o}}var v=n(2389);const f="tabList__CuJ",y="tabItem_LNqP";function k(e){let{className:t,block:n,selectedValue:s,selectValue:l,tabValues:c}=e;const p=[],{blockElementScrollPositionUntilNextRender:u}=(0,i.o5)(),d=e=>{const t=e.currentTarget,n=p.indexOf(t),a=c[n].value;a!==s&&(u(t),l(a))},m=e=>{let t=null;switch(e.key){case"Enter":d(e);break;case"ArrowRight":{const n=p.indexOf(e.currentTarget)+1;t=p[n]??p[0];break}case"ArrowLeft":{const n=p.indexOf(e.currentTarget)-1;t=p[n]??p[p.length-1];break}}t?.focus()};return r.createElement("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,o.Z)("tabs",{"tabs--block":n},t)},c.map((e=>{let{value:t,label:n,attributes:i}=e;return r.createElement("li",(0,a.Z)({role:"tab",tabIndex:s===t?0:-1,"aria-selected":s===t,key:t,ref:e=>p.push(e),onKeyDown:m,onClick:d},i,{className:(0,o.Z)("tabs__item",y,i?.className,{"tabs__item--active":s===t})}),n??t)})))}function b(e){let{lazy:t,children:n,selectedValue:a}=e;const o=(Array.isArray(n)?n:[n]).filter(Boolean);if(t){const e=o.find((e=>e.props.value===a));return e?(0,r.cloneElement)(e,{className:"margin-top--md"}):null}return r.createElement("div",{className:"margin-top--md"},o.map(((e,t)=>(0,r.cloneElement)(e,{key:t,hidden:e.props.value!==a}))))}function N(e){const t=g(e);return r.createElement("div",{className:(0,o.Z)("tabs-container",f)},r.createElement(k,(0,a.Z)({},e,t)),r.createElement(b,(0,a.Z)({},e,t)))}function w(e){const t=(0,v.Z)();return r.createElement(N,(0,a.Z)({key:String(t)},e))}},8243:(e,t,n)=>{n.r(t),n.d(t,{assets:()=>p,contentTitle:()=>l,default:()=>m,frontMatter:()=>s,metadata:()=>c,toc:()=>u});var a=n(7462),r=(n(7294),n(3905)),o=n(4866),i=n(5162);const s={},l="Validation",c={unversionedId:"diagnostics/validation",id:"diagnostics/validation",title:"Validation",description:"Stashbox validation routines help you detect and solve common misconfiguration issues. You can verify the container's actual state with its .Validate() method. It walks through the whole resolution tree and collects all issues into an AggregateException.",source:"@site/docs/diagnostics/validation.md",sourceDirName:"diagnostics",slug:"/diagnostics/validation",permalink:"/stashbox/docs/diagnostics/validation",draft:!1,editUrl:"https://github.com/z4kn4fein/stashbox/edit/master/docs/docs/diagnostics/validation.md",tags:[],version:"current",lastUpdatedBy:"dependabot[bot]",lastUpdatedAt:1697644420,formattedLastUpdatedAt:"Oct 18, 2023",frontMatter:{},sidebar:"docs",previous:{title:"Special resolution cases",permalink:"/stashbox/docs/advanced/special-resolution-cases"},next:{title:"Utilities",permalink:"/stashbox/docs/diagnostics/utilities"}},p={},u=[{value:"Registration validation",id:"registration-validation",level:2},{value:"InvalidRegistrationException",id:"invalidregistrationexception",level:3},{value:"ServiceAlreadyRegisteredException",id:"servicealreadyregisteredexception",level:3},{value:"Resolution validation",id:"resolution-validation",level:2},{value:"Lifetime validation",id:"lifetime-validation",level:2},{value:"Circular dependency",id:"circular-dependency",level:2},{value:"Other exceptions",id:"other-exceptions",level:2},{value:"CompositionRootNotFoundException",id:"compositionrootnotfoundexception",level:3},{value:"ConstructorNotFoundException",id:"constructornotfoundexception",level:3}],d={toc:u};function m(e){let{components:t,...n}=e;return(0,r.kt)("wrapper",(0,a.Z)({},d,n,{components:t,mdxType:"MDXLayout"}),(0,r.kt)("h1",{id:"validation"},"Validation"),(0,r.kt)("p",null,"Stashbox validation routines help you detect and solve common misconfiguration issues. You can verify the container's actual state with its ",(0,r.kt)("inlineCode",{parentName:"p"},".Validate()")," method. It walks through the whole ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#resolution-tree"},"resolution tree")," and collects all issues into an ",(0,r.kt)("inlineCode",{parentName:"p"},"AggregateException"),"."),(0,r.kt)("h2",{id:"registration-validation"},"Registration validation"),(0,r.kt)("p",null,"During registration, the container validates the passed types and throws the following exceptions when the validation fails."),(0,r.kt)("h3",{id:"invalidregistrationexception"},"InvalidRegistrationException"),(0,r.kt)("ol",null,(0,r.kt)("li",{parentName:"ol"},(0,r.kt)("strong",{parentName:"li"},"When the ",(0,r.kt)("a",{parentName:"strong",href:"/docs/getting-started/glossary#service-type--implementation-type"},"implementation type")," is not resolvable.")," (it's an interface or an abstract class registered like: ",(0,r.kt)("inlineCode",{parentName:"li"},"Register()"),"):",(0,r.kt)("pre",{parentName:"li"},(0,r.kt)("code",{parentName:"pre"},"The type Namespace.IService could not be resolved. It's probably an interface, abstract class, or primitive type.\n"))),(0,r.kt)("li",{parentName:"ol"},(0,r.kt)("strong",{parentName:"li"},"When the ",(0,r.kt)("a",{parentName:"strong",href:"/docs/getting-started/glossary#service-type--implementation-type"},"implementation type")," does not implement the ",(0,r.kt)("a",{parentName:"strong",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type")),".",(0,r.kt)("pre",{parentName:"li"},(0,r.kt)("code",{parentName:"pre"},"The type Namespace.MotorCycle does not implement the '[service type](/docs/getting-started/glossary#service-type--implementation-type)' Namespace.ICar.\n")))),(0,r.kt)("h3",{id:"servicealreadyregisteredexception"},"ServiceAlreadyRegisteredException"),(0,r.kt)("p",null,(0,r.kt)("strong",{parentName:"p"},"When the given ",(0,r.kt)("a",{parentName:"strong",href:"/docs/getting-started/glossary#service-type--implementation-type"},"implementation type")," is already registered")," and the ",(0,r.kt)("inlineCode",{parentName:"p"},"RegistrationBehavior")," ",(0,r.kt)("a",{parentName:"p",href:"/docs/configuration/container-configuration#registration-behavior"},"container configuration option")," is set to ",(0,r.kt)("inlineCode",{parentName:"p"},"ThrowException"),":"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre"},"The type Namespace.Service is already registered.\n")),(0,r.kt)("h2",{id:"resolution-validation"},"Resolution validation"),(0,r.kt)("p",null,"During the ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#resolution-tree"},"resolution tree's")," construction, the container continuously checks its actual state to ensure stability. When any of the following issues occur, the container throws a ",(0,r.kt)("inlineCode",{parentName:"p"},"ResolutionFailedException"),"."),(0,r.kt)("ol",null,(0,r.kt)("li",{parentName:"ol"},(0,r.kt)("p",{parentName:"li"},(0,r.kt)("strong",{parentName:"p"},"When a dependency is missing from the ",(0,r.kt)("a",{parentName:"strong",href:"/docs/getting-started/glossary#resolution-tree"},"resolution tree")),"."),(0,r.kt)(o.Z,{mdxType:"Tabs"},(0,r.kt)(i.Z,{value:"Parameter",label:"Parameter",mdxType:"TabItem"},(0,r.kt)("pre",{parentName:"li"},(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"class Service\n{\n public Service(Dependency dep) { }\n\n public Service(Dependency2 dep2) { }\n}\n\ncontainer.Register();\nvar service = container.Resolve();\n")),(0,r.kt)("p",{parentName:"li"},"This will result in the following exception message:"),(0,r.kt)("pre",{parentName:"li"},(0,r.kt)("code",{parentName:"pre"},"Could not resolve type Namespace.Service.\nConstructor Void .ctor(Dependency) found with unresolvable parameter: (Namespace.Dependency)dep.\nConstructor Void .ctor(Dependency2) found with unresolvable parameter: (Namespace.Dependency2)dep2.\n"))),(0,r.kt)(i.Z,{value:"Property / field",label:"Property / field",mdxType:"TabItem"},(0,r.kt)("pre",{parentName:"li"},(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"class Service\n{\n public Dependency Dep { get; set; }\n}\n\ncontainer.Register(options => options.WithDependencyBinding(s => s.Dep));\nvar service = container.Resolve();\n")),(0,r.kt)("p",{parentName:"li"},"This will show the following message:"),(0,r.kt)("pre",{parentName:"li"},(0,r.kt)("code",{parentName:"pre"},"Could not resolve type Namespace.Service.\nUnresolvable property: (Namespace.Dependency)Dep.\n"))))),(0,r.kt)("li",{parentName:"ol"},(0,r.kt)("p",{parentName:"li"},(0,r.kt)("strong",{parentName:"p"},"When the requested type is unresolvable.")," E.g., it doesn't have a public constructor."),(0,r.kt)("pre",{parentName:"li"},(0,r.kt)("code",{parentName:"pre"},"Could not resolve type Namespace.Service.\nService is not registered or unresolvable type requested.\n")))),(0,r.kt)("h2",{id:"lifetime-validation"},"Lifetime validation"),(0,r.kt)("p",null,"This validation enforces the following rules. When they are violated, the container throws a ",(0,r.kt)("inlineCode",{parentName:"p"},"LifetimeValidationFailedException"),"."),(0,r.kt)("ol",null,(0,r.kt)("li",{parentName:"ol"},(0,r.kt)("p",{parentName:"li"},(0,r.kt)("strong",{parentName:"p"},"When a scoped service is requested from the ",(0,r.kt)("a",{parentName:"strong",href:"/docs/getting-started/glossary#root-scope"},"root scope")),".",(0,r.kt)("br",{parentName:"p"}),"\n","As the ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#root-scope"},"root scope's")," lifetime is bound to the container's lifetime, this action unintentionally promotes the scoped service's lifetime to singleton:"),(0,r.kt)("pre",{parentName:"li"},(0,r.kt)("code",{parentName:"pre"},"Resolution of Namespace.Service (ScopedLifetime) from the '[root scope](/docs/getting-started/glossary#root-scope)' is not allowed, \nthat would promote the service's lifetime to a singleton.\n"))),(0,r.kt)("li",{parentName:"ol"},(0,r.kt)("p",{parentName:"li"},(0,r.kt)("strong",{parentName:"p"},"When the life-span of a dependency is shorter than its parent's"),".",(0,r.kt)("br",{parentName:"p"}),"\n","It's called ",(0,r.kt)("a",{parentName:"p",href:"https://blog.ploeh.dk/2014/06/02/captive-dependency/"},"captive dependency"),". Every lifetime has a ",(0,r.kt)("inlineCode",{parentName:"p"},"LifeSpan")," value, which determines how long the related service lives. The main rule is that services may not contain dependencies with shorter life spans. E.g., singletons should not depend on scoped services. The only exception is the life span value ",(0,r.kt)("inlineCode",{parentName:"p"},"0"),", which indicates that the related service is state-less and could be injected into any service. "),(0,r.kt)("p",{parentName:"li"}," These are the ",(0,r.kt)("inlineCode",{parentName:"p"},"LifeSpan")," values of the pre-defined lifetimes: "),(0,r.kt)("ul",{parentName:"li"},(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("strong",{parentName:"li"},"Singleton"),": 20"),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("strong",{parentName:"li"},"Scoped"),": 10"),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("strong",{parentName:"li"},"NamedScope"),": 10"),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("strong",{parentName:"li"},"PerRequest"),": 0"),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("strong",{parentName:"li"},"PerScopedRequest"),": 0"),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("strong",{parentName:"li"},"Transient"),": 0")),(0,r.kt)("p",{parentName:"li"},"In case of a failed validation the exception message would be:"),(0,r.kt)("pre",{parentName:"li"},(0,r.kt)("code",{parentName:"pre"},"The life-span of Namespace.Service (ScopedLifetime|10) is shorter than \nits direct or indirect parent's Namespace.Dependency (Singleton|20). \nThis could lead to incidental lifetime promotions with longer life-span, \nit's recommended to double-check your lifetime configurations.\n")))),(0,r.kt)("h2",{id:"circular-dependency"},"Circular dependency"),(0,r.kt)("p",null,"When the container encounters a circular dependency loop in the ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#resolution-tree"},"resolution tree"),", it throws a ",(0,r.kt)("inlineCode",{parentName:"p"},"CircularDependencyException"),"."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"class Service1\n{\n public Service1(Service2 service2) { }\n}\n\nclass Service2\n{\n public Service2(Service1 service1) { }\n}\n\ncontainer.Register();\ncontainer.Register();\nvar service = container.Resolve();\n")),(0,r.kt)("p",null,"The exception message is: "),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre"},"Circular dependency detected during the resolution of Namespace.Service1.\n")),(0,r.kt)("h2",{id:"other-exceptions"},"Other exceptions"),(0,r.kt)("h3",{id:"compositionrootnotfoundexception"},"CompositionRootNotFoundException"),(0,r.kt)("p",null,"This exception pops up when we try to compose an assembly, but it doesn't contain an ",(0,r.kt)("inlineCode",{parentName:"p"},"ICompositionRoot")," implementation."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.ComposeAssembly(typeof(Service).Assembly);\n")),(0,r.kt)("p",null,"The exception message is: "),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre"},"No ICompositionRoot found in the given assembly: {your-assembly-name}\n")),(0,r.kt)("h3",{id:"constructornotfoundexception"},"ConstructorNotFoundException"),(0,r.kt)("p",null,"During the registration phase, when you are using the ",(0,r.kt)("a",{parentName:"p",href:"/docs/configuration/registration-configuration#withconstructorbyargumenttypes"},(0,r.kt)("inlineCode",{parentName:"a"},"WithConstructorByArgumentTypes()"))," or ",(0,r.kt)("a",{parentName:"p",href:"/docs/configuration/registration-configuration#withconstructorbyarguments"},(0,r.kt)("inlineCode",{parentName:"a"},"WithConstructorByArguments()"))," options, you can accidentally point to a non-existing constructor. In that case, the container throws a ",(0,r.kt)("inlineCode",{parentName:"p"},"ConstructorNotFoundException"),"."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"class Service\n{\n public Service(Dependency dep) { }\n}\n\ncontainer.Register(options => options.WithConstructorByArgumentTypes(typeof(string), typeof(int)));\n")),(0,r.kt)("p",null,"The exception message is: "),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre"},"Constructor not found for Namespace.Service with the given argument types: System.String, System.Int32.\n")))}m.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/3a87badd.15110844.js b/assets/js/3a87badd.15110844.js deleted file mode 100644 index 94c1839f..00000000 --- a/assets/js/3a87badd.15110844.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkwebsite=self.webpackChunkwebsite||[]).push([[80],{3905:(e,t,n)=>{n.d(t,{Zo:()=>p,kt:()=>g});var a=n(7294);function o(e,t,n){return t in e?Object.defineProperty(e,t,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[t]=n,e}function r(e,t){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);t&&(a=a.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),n.push.apply(n,a)}return n}function i(e){for(var t=1;t=0||(o[n]=e[n]);return o}(e,t);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);for(a=0;a=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(o[n]=e[n])}return o}var l=a.createContext({}),c=function(e){var t=a.useContext(l),n=t;return e&&(n="function"==typeof e?e(t):i(i({},t),e)),n},p=function(e){var t=c(e.components);return a.createElement(l.Provider,{value:t},e.children)},u="mdxType",d={inlineCode:"code",wrapper:function(e){var t=e.children;return a.createElement(a.Fragment,{},t)}},m=a.forwardRef((function(e,t){var n=e.components,o=e.mdxType,r=e.originalType,l=e.parentName,p=s(e,["components","mdxType","originalType","parentName"]),u=c(n),m=o,g=u["".concat(l,".").concat(m)]||u[m]||d[m]||r;return n?a.createElement(g,i(i({ref:t},p),{},{components:n})):a.createElement(g,i({ref:t},p))}));function g(e,t){var n=arguments,o=t&&t.mdxType;if("string"==typeof e||o){var r=n.length,i=new Array(r);i[0]=m;var s={};for(var l in t)hasOwnProperty.call(t,l)&&(s[l]=t[l]);s.originalType=e,s[u]="string"==typeof e?e:o,i[1]=s;for(var c=2;c{n.d(t,{Z:()=>i});var a=n(7294),o=n(6010);const r="tabItem_Ymn6";function i(e){let{children:t,hidden:n,className:i}=e;return a.createElement("div",{role:"tabpanel",className:(0,o.Z)(r,i),hidden:n},t)}},4866:(e,t,n)=>{n.d(t,{Z:()=>N});var a=n(7462),o=n(7294),r=n(6010),i=n(2466),s=n(6550),l=n(1980),c=n(7392),p=n(12);function u(e){return function(e){return o.Children.map(e,(e=>{if(!e||(0,o.isValidElement)(e)&&function(e){const{props:t}=e;return!!t&&"object"==typeof t&&"value"in t}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}(e).map((e=>{let{props:{value:t,label:n,attributes:a,default:o}}=e;return{value:t,label:n,attributes:a,default:o}}))}function d(e){const{values:t,children:n}=e;return(0,o.useMemo)((()=>{const e=t??u(n);return function(e){const t=(0,c.l)(e,((e,t)=>e.value===t.value));if(t.length>0)throw new Error(`Docusaurus error: Duplicate values "${t.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[t,n])}function m(e){let{value:t,tabValues:n}=e;return n.some((e=>e.value===t))}function g(e){let{queryString:t=!1,groupId:n}=e;const a=(0,s.k6)(),r=function(e){let{queryString:t=!1,groupId:n}=e;if("string"==typeof t)return t;if(!1===t)return null;if(!0===t&&!n)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return n??null}({queryString:t,groupId:n});return[(0,l._X)(r),(0,o.useCallback)((e=>{if(!r)return;const t=new URLSearchParams(a.location.search);t.set(r,e),a.replace({...a.location,search:t.toString()})}),[r,a])]}function b(e){const{defaultValue:t,queryString:n=!1,groupId:a}=e,r=d(e),[i,s]=(0,o.useState)((()=>function(e){let{defaultValue:t,tabValues:n}=e;if(0===n.length)throw new Error("Docusaurus error: the component requires at least one children component");if(t){if(!m({value:t,tabValues:n}))throw new Error(`Docusaurus error: The has a defaultValue "${t}" but none of its children has the corresponding value. Available values are: ${n.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return t}const a=n.find((e=>e.default))??n[0];if(!a)throw new Error("Unexpected error: 0 tabValues");return a.value}({defaultValue:t,tabValues:r}))),[l,c]=g({queryString:n,groupId:a}),[u,b]=function(e){let{groupId:t}=e;const n=function(e){return e?`docusaurus.tab.${e}`:null}(t),[a,r]=(0,p.Nk)(n);return[a,(0,o.useCallback)((e=>{n&&r.set(e)}),[n,r])]}({groupId:a}),v=(()=>{const e=l??u;return m({value:e,tabValues:r})?e:null})();(0,o.useLayoutEffect)((()=>{v&&s(v)}),[v]);return{selectedValue:i,selectValue:(0,o.useCallback)((e=>{if(!m({value:e,tabValues:r}))throw new Error(`Can't select invalid tab value=${e}`);s(e),c(e),b(e)}),[c,b,r]),tabValues:r}}var v=n(2389);const h="tabList__CuJ",k="tabItem_LNqP";function y(e){let{className:t,block:n,selectedValue:s,selectValue:l,tabValues:c}=e;const p=[],{blockElementScrollPositionUntilNextRender:u}=(0,i.o5)(),d=e=>{const t=e.currentTarget,n=p.indexOf(t),a=c[n].value;a!==s&&(u(t),l(a))},m=e=>{let t=null;switch(e.key){case"Enter":d(e);break;case"ArrowRight":{const n=p.indexOf(e.currentTarget)+1;t=p[n]??p[0];break}case"ArrowLeft":{const n=p.indexOf(e.currentTarget)-1;t=p[n]??p[p.length-1];break}}t?.focus()};return o.createElement("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,r.Z)("tabs",{"tabs--block":n},t)},c.map((e=>{let{value:t,label:n,attributes:i}=e;return o.createElement("li",(0,a.Z)({role:"tab",tabIndex:s===t?0:-1,"aria-selected":s===t,key:t,ref:e=>p.push(e),onKeyDown:m,onClick:d},i,{className:(0,r.Z)("tabs__item",k,i?.className,{"tabs__item--active":s===t})}),n??t)})))}function f(e){let{lazy:t,children:n,selectedValue:a}=e;const r=(Array.isArray(n)?n:[n]).filter(Boolean);if(t){const e=r.find((e=>e.props.value===a));return e?(0,o.cloneElement)(e,{className:"margin-top--md"}):null}return o.createElement("div",{className:"margin-top--md"},r.map(((e,t)=>(0,o.cloneElement)(e,{key:t,hidden:e.props.value!==a}))))}function I(e){const t=b(e);return o.createElement("div",{className:(0,r.Z)("tabs-container",h)},o.createElement(y,(0,a.Z)({},e,t)),o.createElement(f,(0,a.Z)({},e,t)))}function N(e){const t=(0,v.Z)();return o.createElement(I,(0,a.Z)({key:String(t)},e))}},8846:(e,t,n)=>{n.d(t,{Z:()=>s});var a=n(7294);const o="codeDescContainer_ie8f",r="desc_jyqI",i="example_eYlF";function s(e){let{children:t}=e,n=a.Children.toArray(t).filter((e=>e));return a.createElement("div",{className:o},a.createElement("div",{className:r},n[0]),a.createElement("div",{className:i},n[1]))}},9470:(e,t,n)=>{n.r(t),n.d(t,{assets:()=>u,contentTitle:()=>c,default:()=>g,frontMatter:()=>l,metadata:()=>p,toc:()=>d});var a=n(7462),o=(n(7294),n(3905)),r=n(8846),i=n(4866),s=n(5162);const l={},c="Advanced registration",p={unversionedId:"guides/advanced-registration",id:"guides/advanced-registration",title:"Advanced registration",description:"This section is about Stashbox's further configuration options, including the registration configuration API, the registration of factory delegates, multiple implementations, batch registrations, the concept of the Composition Root, and many more.",source:"@site/docs/guides/advanced-registration.md",sourceDirName:"guides",slug:"/guides/advanced-registration",permalink:"/stashbox/docs/guides/advanced-registration",draft:!1,editUrl:"https://github.com/z4kn4fein/stashbox/edit/master/docs/docs/guides/advanced-registration.md",tags:[],version:"current",lastUpdatedBy:"dependabot[bot]",lastUpdatedAt:1697644420,formattedLastUpdatedAt:"Oct 18, 2023",frontMatter:{},sidebar:"docs",previous:{title:"Basic usage",permalink:"/stashbox/docs/guides/basics"},next:{title:"Service resolution",permalink:"/stashbox/docs/guides/service-resolution"}},u={},d=[{value:"Factory registration",id:"factory-registration",level:2},{value:"Factories with parameter overrides",id:"factories-with-parameter-overrides",level:3},{value:"Consider this before using the resolver parameter inside a factory",id:"consider-this-before-using-the-resolver-parameter-inside-a-factory",level:3},{value:"Delegates with dependencies passed as parameters",id:"delegates-with-dependencies-passed-as-parameters",level:4},{value:"Accessing the currently resolving type in factories",id:"accessing-the-currently-resolving-type-in-factories",level:3},{value:"Multiple implementations",id:"multiple-implementations",level:2},{value:"Binding to multiple services",id:"binding-to-multiple-services",level:2},{value:"Batch registration",id:"batch-registration",level:2},{value:"Assembly registration",id:"assembly-registration",level:2},{value:"Composition root",id:"composition-root",level:2},{value:"Injection parameters",id:"injection-parameters",level:2},{value:"Initializer / finalizer",id:"initializer--finalizer",level:2}],m={toc:d};function g(e){let{components:t,...n}=e;return(0,o.kt)("wrapper",(0,a.Z)({},m,n,{components:t,mdxType:"MDXLayout"}),(0,o.kt)("h1",{id:"advanced-registration"},"Advanced registration"),(0,o.kt)("p",null,"This section is about Stashbox's further configuration options, including the registration configuration API, the registration of factory delegates, multiple implementations, batch registrations, the concept of the ",(0,o.kt)("a",{parentName:"p",href:"https://blog.ploeh.dk/2011/07/28/CompositionRoot/"},"Composition Root"),", and many more."),(0,o.kt)("admonition",{type:"info"},(0,o.kt)("p",{parentName:"admonition"},"This section won't cover all the available options of the registrations API, but you can find them ",(0,o.kt)("a",{parentName:"p",href:"/docs/configuration/registration-configuration"},"here"),".")),(0,o.kt)("h2",{id:"factory-registration"},"Factory registration"),(0,o.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,o.kt)("div",null,(0,o.kt)("p",null,"You can bind a factory delegate to a registration that the container will invoke directly to instantiate your service. "),(0,o.kt)("p",null,"You can use parameter-less and custom parameterized delegates as a factory. ",(0,o.kt)("a",{parentName:"p",href:"/docs/configuration/registration-configuration#factory"},"Here")," is the list of all available options."),(0,o.kt)("p",null,"You can also get the current ",(0,o.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#dependency-resolver"},"dependency resolver")," as a delegate parameter to resolve any additional dependencies required for the service construction.")),(0,o.kt)("div",null,(0,o.kt)(i.Z,{mdxType:"Tabs"},(0,o.kt)(s.Z,{value:"Parameter-less",label:"Parameter-less",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithFactory(() => new ConsoleLogger());\n\n// the container uses the factory for instantiation.\nIJob job = container.Resolve();\n"))),(0,o.kt)(s.Z,{value:"Parameterized",label:"Parameterized",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithFactory(logger => new DbBackup(logger));\n\n// the container uses the factory for instantiation.\nIJob job = container.Resolve();\n"))),(0,o.kt)(s.Z,{value:"Resolver parameter",label:"Resolver parameter",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithFactory(resolver => new DbBackup(resolver.Resolve()));\n \n// the container uses the factory for instantiation.\nIJob job = container.Resolve();\n")))))),(0,o.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,o.kt)("div",null,(0,o.kt)("p",null,"Delegate factories are useful when your service's instantiation is not straight-forward for the container, like when it depends on something that is not available at resolution time. E.g., a connection string.")),(0,o.kt)("div",null,(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register(options => options\n .WithFactory(logger => \n new DbBackup(Configuration["DbConnectionString"], logger));\n')))),(0,o.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,o.kt)("div",null,(0,o.kt)("h3",{id:"factories-with-parameter-overrides"},"Factories with parameter overrides"),(0,o.kt)("p",null,"Stashbox can implicitly ",(0,o.kt)("a",{parentName:"p",href:"/docs/advanced/wrappers-resolvers#delegate"},"wrap")," your service in a ",(0,o.kt)("inlineCode",{parentName:"p"},"Delegate")," and lets you pass parameters that can override your service's dependencies. Moreover, you can register your own custom delegate that the container will resolve when you request your service wrapped in a ",(0,o.kt)("inlineCode",{parentName:"p"},"Delegate"),".")),(0,o.kt)("div",null,(0,o.kt)(i.Z,{groupId:"generic-runtime-apis",mdxType:"Tabs"},(0,o.kt)(s.Z,{value:"Generic API",label:"Generic API",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},'container.RegisterFunc((connectionString, resolver) => \n new DbBackup(connectionString, resolver.Resolve()));\n\nFunc backupFactory = container.Resolve>();\nIJob dbBackup = backupFactory(Configuration["ConnectionString"]);\n'))),(0,o.kt)(s.Z,{value:"Runtime type API",label:"Runtime type API",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},'container.RegisterFunc((connectionString, resolver) => \n new DbBackup(connectionString, resolver.Resolve()));\n\nDelegate backupFactory = container.ResolveFactory(typeof(IJob), \n parameterTypes: new[] { typeof(string) });\nIJob dbBackup = backupFactory.DynamicInvoke(Configuration["ConnectionString"]);\n')))))),(0,o.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,o.kt)("div",null,(0,o.kt)("p",null,"If a service has multiple constructors, the container visits those first, that has matching parameters passed to the factory, with respecting the additional ",(0,o.kt)("a",{parentName:"p",href:"/docs/configuration/registration-configuration#constructor-selection"},"constructor selection rules"),".")),(0,o.kt)("div",null,(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"class Service\n{\n public Service(int number) { }\n public Service(string text) { }\n}\n\ncontainer.Register();\n\n// create the factory with an int input parameter.\nvar func = constainer.Resolve>();\n\n// the constructor with the int param \n// is used for instantiation.\nvar service = func(2);\n")))),(0,o.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,o.kt)("div",null,(0,o.kt)("h3",{id:"consider-this-before-using-the-resolver-parameter-inside-a-factory"},"Consider this before using the resolver parameter inside a factory"),(0,o.kt)("p",null,"Delegate factories are a black-box for the container. It doesn't have control over what's happening inside a delegate, which means when you resolve additional dependencies with the ",(0,o.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#dependency-resolver"},"dependency resolver")," parameter, they could easily bypass the ",(0,o.kt)("a",{parentName:"p",href:"/docs/diagnostics/validation#lifetime-validation"},"lifetime")," and ",(0,o.kt)("a",{parentName:"p",href:"/docs/diagnostics/validation#circular-dependency"},"circular dependency")," validations. Fortunately, you have the option to keep them validated anyway with parameterized factory delegates."),(0,o.kt)("h4",{id:"delegates-with-dependencies-passed-as-parameters"},"Delegates with dependencies passed as parameters"),(0,o.kt)("p",null,"Rather than using the ",(0,o.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#dependency-resolver"},"dependency resolver")," parameter inside the factory, let the container inject the dependencies into the delegate as parameters. This way, the ",(0,o.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#resolution-tree"},"resolution tree's")," integrity remains stable because no service resolution happens inside the black-box, and each parameter is validated.")),(0,o.kt)("div",null,(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"interface IEventProcessor { }\n\nclass EventProcessor : IEventProcessor\n{\n public EventProcessor(ILogger logger, IEventValidator validator)\n { }\n}\n\ncontainer.Register();\ncontainer.Register();\n\ncontainer.Register(options => options\n // Ilogger and IEventValidator instances are injected\n // by the container at resolution time, so they will be\n // validated against circular and captive dependencies.\n .WithFactory((logger, validator) => \n new EventProcessor(logger, validator));\n\n// the container resolves ILogger and IEventValidator first, then\n// it passes them to the factory as delegate parameters.\nIEventProcessor processor = container.Resolve();\n")))),(0,o.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,o.kt)("div",null,(0,o.kt)("h3",{id:"accessing-the-currently-resolving-type-in-factories"},"Accessing the currently resolving type in factories"),(0,o.kt)("p",null,"To access the currently resolving type in factory delegates, you can set the ",(0,o.kt)("inlineCode",{parentName:"p"},"TypeInformation")," type as an input parameter of the factory.\nThe ",(0,o.kt)("inlineCode",{parentName:"p"},"TypeInformation")," holds every reflected context information about the currently resolving type. "),(0,o.kt)("p",null,"This can be useful when the resolution is, e.g., in an open generic context, and we want to know which closed generic variant is requested.")),(0,o.kt)("div",null,(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"interface IService { }\n\nclass Service : IService { }\n\ncontainer.Register(typeof(IService<>), typeof(Service<>), options => \n options.WithFactory(typeInfo => \n {\n // typeInfo.Type here holds the actual type like\n // IService based on the resolution request below.\n }));\n \ncontainer.Resolve>();\n")))),(0,o.kt)("h2",{id:"multiple-implementations"},"Multiple implementations"),(0,o.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,o.kt)("div",null,(0,o.kt)("p",null,"As we previously saw in the ",(0,o.kt)("a",{parentName:"p",href:"/docs/guides/basics#named-registration"},"Named registration")," topic, Stashbox allows you to have multiple implementations bound to a particular ",(0,o.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type"),". You can use names to distinguish them, but you can also access them by requesting a typed collection using the ",(0,o.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type"),"."),(0,o.kt)("admonition",{type:"note"},(0,o.kt)("p",{parentName:"admonition"},"The returned collection is in the same order as the services were registered.\nAlso, to request a collection, you can use any interface implemented by an array."))),(0,o.kt)("div",null,(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register();\ncontainer.Register();\ncontainer.Register();\n")),(0,o.kt)(i.Z,{mdxType:"Tabs"},(0,o.kt)(s.Z,{value:"ResolveAll",label:"ResolveAll",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"// jobs contain all three services in registration order.\nIEnumerable jobs = container.ResolveAll();\n"))),(0,o.kt)(s.Z,{value:"Array",label:"Array",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"// jobs contain all three services in registration order.\nIJob[] jobs = container.Resolve();\n"))),(0,o.kt)(s.Z,{value:"IEnumerable",label:"IEnumerable",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"// jobs contain all three services in registration order.\nIEnumerable jobs = container.Resolve>();\n"))),(0,o.kt)(s.Z,{value:"IList",label:"IList",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"// jobs contain all three services in registration order.\nIList jobs = container.Resolve>();\n"))),(0,o.kt)(s.Z,{value:"ICollection",label:"ICollection",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"// jobs contain all three services in registration order.\nICollection jobs = container.Resolve>();\n")))))),(0,o.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,o.kt)("div",null,(0,o.kt)("p",null,"When you have multiple implementations registered to a service, a request to the ",(0,o.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type")," without a name will return the ",(0,o.kt)("strong",{parentName:"p"},"last registered implementation"),"."),(0,o.kt)("admonition",{type:"info"},(0,o.kt)("p",{parentName:"admonition"},"Not only names can be used to distinguish registrations, ",(0,o.kt)("a",{parentName:"p",href:"/docs/guides/service-resolution#conditional-resolution"},"conditions"),", ",(0,o.kt)("a",{parentName:"p",href:"/docs/guides/scopes#named-scopes"},"named scopes"),", and ",(0,o.kt)("a",{parentName:"p",href:"/docs/advanced/wrappers-resolvers#metadata--tuple"},"metadata")," can also influence the results."))),(0,o.kt)("div",null,(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register();\ncontainer.Register();\ncontainer.Register();\n\n// job will be the ImageProcess.\nIJob job = container.Resolve();\n")))),(0,o.kt)("h2",{id:"binding-to-multiple-services"},"Binding to multiple services"),(0,o.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,o.kt)("div",null,(0,o.kt)("p",null,"When you have a service that implements multiple interfaces, you have the option to bind its registration to all or some of those additional interfaces or base types."),(0,o.kt)("p",null,"Suppose we have the following class declaration:"),(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"class DbBackup : IJob, IScheduledJob\n{ \n public DbBackup() { }\n}\n"))),(0,o.kt)("div",null,(0,o.kt)(i.Z,{mdxType:"Tabs"},(0,o.kt)(s.Z,{value:"To another type",label:"To another type",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .AsServiceAlso());\n\nIJob job = container.Resolve(); // DbBackup\nIScheduledJob job = container.Resolve(); // DbBackup\nDbBackup job = container.Resolve(); // error, not found\n"))),(0,o.kt)(s.Z,{value:"To all implemented types",label:"To all implemented types",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .AsImplementedTypes());\n\nIJob job = container.Resolve(); // DbBackup\nIScheduledJob job = container.Resolve(); // DbBackup\nDbBackup job = container.Resolve(); // DbBackup\n")))))),(0,o.kt)("h2",{id:"batch-registration"},"Batch registration"),(0,o.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,o.kt)("div",null,(0,o.kt)("p",null,"You have the option to register multiple services in a single registration operation. "),(0,o.kt)("p",null,(0,o.kt)("strong",{parentName:"p"},"Filters (optional):"),"\nFirst, the container will use the ",(0,o.kt)("em",{parentName:"p"},"implementation filter")," action to select only those types from the collection we want to register. When we have those, the container will execute the ",(0,o.kt)("em",{parentName:"p"},"service filter")," on their implemented interfaces and base classes to select which ",(0,o.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type")," they should be mapped to."),(0,o.kt)("admonition",{type:"note"},(0,o.kt)("p",{parentName:"admonition"},"Framework types like ",(0,o.kt)("inlineCode",{parentName:"p"},"IDisposable")," are excluded from being considered as a ",(0,o.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type")," by default.")),(0,o.kt)("admonition",{type:"tip"},(0,o.kt)("p",{parentName:"admonition"},"You can use the registration configuration API to configure individual registrations."))),(0,o.kt)("div",null,(0,o.kt)(i.Z,{mdxType:"Tabs"},(0,o.kt)(s.Z,{value:"Default",label:"Default",mdxType:"TabItem"},(0,o.kt)("p",null,"This example will register three types to all their implemented interfaces, extended base classes, and to themselves (",(0,o.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#self-registration"},"self registration"),") without any filter:"),(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterTypes(new[] \n { \n typeof(DbBackup), \n typeof(ConsoleLogger), \n typeof(StorageCleanup) \n });\n\nIEnumerable jobs = container.ResolveAll(); // 2 items\nILogger logger = container.Resolve(); // ConsoleLogger\nIJob job = container.Resolve(); // StorageCleanup\nDbBackup backup = container.Resolve(); // DbBackup\n"))),(0,o.kt)(s.Z,{value:"Filters",label:"Filters",mdxType:"TabItem"},(0,o.kt)("p",null,"In this example, we assume that ",(0,o.kt)("inlineCode",{parentName:"p"},"DbBackup")," and ",(0,o.kt)("inlineCode",{parentName:"p"},"StorageCleanup")," are implementing ",(0,o.kt)("inlineCode",{parentName:"p"},"IDisposable")," besides ",(0,o.kt)("inlineCode",{parentName:"p"},"IJob")," and also extending a ",(0,o.kt)("inlineCode",{parentName:"p"},"JobBase")," abstract class."),(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterTypes(new[] \n { typeof(DbBackup), typeof(ConsoleLogger), typeof(StorageCleanup) },\n // implementation filter, only those implementations that implements IDisposable\n impl => typeof(IDisposable).IsAssignableFrom(impl),\n // service filter, register them to base classes only\n (impl, service) => service.IsAbstract && !service.IsInterface);\n\nIEnumerable jobs = container.ResolveAll(); // 0 items\nIEnumerable jobs = container.ResolveAll(); // 2 items\nILogger logger = container.Resolve(); // error, not found\nDbBackup backup = container.Resolve(); // DbBackup\n"))),(0,o.kt)(s.Z,{value:"Without self",label:"Without self",mdxType:"TabItem"},(0,o.kt)("p",null,"This example ignores the ",(0,o.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#self-registration"},"self registrations")," completely:"),(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterTypes(new[] \n { \n typeof(DbBackup), \n typeof(ConsoleLogger), \n typeof(StorageCleanup)\n },\n registerSelf: false);\n\nIEnumerable jobs = container.ResolveAll(); // 2 items\nILogger logger = container.Resolve(); // ConsoleLogger\nDbBackup backup = container.Resolve(); // error, not found\nConsoleLogger logger = container.Resolve(); // error, not found\n"))),(0,o.kt)(s.Z,{value:"Registration options",label:"Registration options",mdxType:"TabItem"},(0,o.kt)("p",null,"This example will configure all registrations mapped to ",(0,o.kt)("inlineCode",{parentName:"p"},"ILogger")," as ",(0,o.kt)("inlineCode",{parentName:"p"},"Singleton"),":"),(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterTypes(new[] \n { \n typeof(DbBackup), \n typeof(ConsoleLogger), \n typeof(StorageCleanup)\n },\n configurator: options => \n {\n if (options.HasServiceType())\n options.WithSingletonLifetime();\n });\n\nILogger logger = container.Resolve(); // ConsoleLogger\nILogger newLogger = container.Resolve(); // the same ConsoleLogger\nIEnumerable jobs = container.ResolveAll(); // 2 items\n")))))),(0,o.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,o.kt)("div",null,(0,o.kt)("p",null,"Another type of service filter is the ",(0,o.kt)("inlineCode",{parentName:"p"},".RegisterTypesAs()")," method, which registers only those types that implements the ",(0,o.kt)("inlineCode",{parentName:"p"},"T")," ",(0,o.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type"),"."),(0,o.kt)("admonition",{type:"note"},(0,o.kt)("p",{parentName:"admonition"},"This method also accepts an implementation filter and a registration configurator action like ",(0,o.kt)("inlineCode",{parentName:"p"},".RegisterTypes()"),".")),(0,o.kt)("admonition",{type:"caution"},(0,o.kt)("p",{parentName:"admonition"},(0,o.kt)("inlineCode",{parentName:"p"},".RegisterTypesAs()")," doesn't create ",(0,o.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#self-registration"},"self registrations")," as it only maps the implementations to the given ",(0,o.kt)("inlineCode",{parentName:"p"},"T")," ",(0,o.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type"),"."))),(0,o.kt)("div",null,(0,o.kt)(i.Z,{groupId:"generic-runtime-apis",mdxType:"Tabs"},(0,o.kt)(s.Z,{value:"Generic API",label:"Generic API",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterTypesAs(new[] \n { \n typeof(DbBackup), \n typeof(ConsoleLogger), \n typeof(StorageCleanup) \n });\n\nIEnumerable jobs = container.ResolveAll(); // 2 items\nILogger logger = container.Resolve(); // error, not found\nIJob job = container.Resolve(); // StorageCleanup\nDbBackup backup = container.Resolve(); // error, not found\n"))),(0,o.kt)(s.Z,{value:"Runtime type API",label:"Runtime type API",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterTypesAs(typeof(IJob), new[] \n { \n typeof(DbBackup), \n typeof(ConsoleLogger), \n typeof(StorageCleanup) \n });\n\nIEnumerable jobs = container.ResolveAll(); // 2 items\nILogger logger = container.Resolve(); // error, not found\nIJob job = container.Resolve(); // StorageCleanup\nDbBackup backup = container.Resolve(); // error, not found\n")))))),(0,o.kt)("h2",{id:"assembly-registration"},"Assembly registration"),(0,o.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,o.kt)("div",null,(0,o.kt)("p",null,"The batch registration API ",(0,o.kt)("em",{parentName:"p"},"(filters, registration configuration action, self-registration)")," is also usable for registering services from given assemblies."),(0,o.kt)("p",null,"In this example, we assume that the same three services we used in the ",(0,o.kt)("a",{parentName:"p",href:"#batch-registration"},"batch registration")," section are in the same assembly."),(0,o.kt)("admonition",{type:"info"},(0,o.kt)("p",{parentName:"admonition"},"The container also detects and registers open-generic definitions (when applicable) from the supplied type collection. You can read about ",(0,o.kt)("a",{parentName:"p",href:"/docs/advanced/generics#open-generics"},"open-generics here"),"."))),(0,o.kt)("div",null,(0,o.kt)(i.Z,{mdxType:"Tabs"},(0,o.kt)(s.Z,{value:"Single assembly",label:"Single assembly",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterAssembly(typeof(DbBackup).Assembly,\n // service filter, register to interfaces only\n serviceTypeSelector: (impl, service) => service.IsInterface,\n registerSelf: false,\n configurator: options => options.WithoutDisposalTracking()\n);\n\nIEnumerable jobs = container.ResolveAll(); // 2 items\nIEnumerable jobs = container.ResolveAll(); // 0 items\nILogger logger = container.Resolve(); // ConsoleLogger\nDbBackup backup = container.Resolve(); // error, not found\n"))),(0,o.kt)(s.Z,{value:"Multiple assemblies",label:"Multiple assemblies",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterAssemblies(new[] \n { \n typeof(DbBackup).Assembly, \n typeof(JobFromAnotherAssembly).Assembly \n },\n // service filter, register to interfaces only\n serviceTypeSelector: (impl, service) => service.IsInterface,\n registerSelf: false,\n configurator: options => options.WithoutDisposalTracking()\n);\n\nIEnumerable jobs = container.ResolveAll(); // 2 items\nIEnumerable jobs = container.ResolveAll(); // 0 items\nILogger logger = container.Resolve(); // ConsoleLogger\nDbBackup backup = container.Resolve(); // error, not found\n"))),(0,o.kt)(s.Z,{value:"Containing type",label:"Containing type",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterAssemblyContaining(\n // service filter, register to interfaces only\n serviceTypeSelector: (impl, service) => service.IsInterface,\n registerSelf: false,\n configurator: options => options.WithoutDisposalTracking()\n);\n\nIEnumerable jobs = container.ResolveAll(); // 2 items\nIEnumerable jobs = container.ResolveAll(); // 0 items\nILogger logger = container.Resolve(); // ConsoleLogger\nDbBackup backup = container.Resolve(); // error, not found\n")))))),(0,o.kt)("h2",{id:"composition-root"},"Composition root"),(0,o.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,o.kt)("div",null,(0,o.kt)("p",null,"The ",(0,o.kt)("a",{parentName:"p",href:"https://blog.ploeh.dk/2011/07/28/CompositionRoot/"},"Composition Root")," is an entry point where all services required to make a component functional are wired together."),(0,o.kt)("p",null,"Stashbox provides an ",(0,o.kt)("inlineCode",{parentName:"p"},"ICompositionRoot")," interface that can be used to define an entry point for a given component or even for an entire assembly. "),(0,o.kt)("p",null,"You can wire up your ",(0,o.kt)("em",{parentName:"p"},"composition root")," implementation with ",(0,o.kt)("inlineCode",{parentName:"p"},"ComposeBy()"),", or you can let the container find and execute all available ",(0,o.kt)("em",{parentName:"p"},"composition root")," implementations within an assembly."),(0,o.kt)("admonition",{type:"note"},(0,o.kt)("p",{parentName:"admonition"},"Your ",(0,o.kt)("inlineCode",{parentName:"p"},"ICompositionRoot")," implementation also can have dependencies that the container will resolve."))),(0,o.kt)("div",null,(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"class ExampleRoot : ICompositionRoot\n{\n public ExampleRoot(IDependency rootDependency)\n { }\n\n public void Compose(IStashboxContainer container)\n {\n container.Register();\n container.Register();\n }\n}\n")),(0,o.kt)(i.Z,{mdxType:"Tabs"},(0,o.kt)(s.Z,{value:"Single",label:"Single",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"// compose a single root.\ncontainer.ComposeBy();\n"))),(0,o.kt)(s.Z,{value:"Assembly",label:"Assembly",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"// compose every root in the given assembly.\ncontainer.ComposeAssembly(typeof(IServiceA).Assembly);\n"))),(0,o.kt)(s.Z,{value:"Override",label:"Override",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"// compose a single root with dependency override.\ncontainer.ComposeBy(new CustomRootDependency());\n")))))),(0,o.kt)("h2",{id:"injection-parameters"},"Injection parameters"),(0,o.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,o.kt)("div",null,(0,o.kt)("p",null,"If you have pre-evaluated dependencies you'd like to inject at resolution time, you can set them as injection parameters during registration. "),(0,o.kt)("admonition",{type:"note"},(0,o.kt)("p",{parentName:"admonition"},"Injection parameter names are matched to constructor arguments or field/property names."))),(0,o.kt)("div",null,(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register(options => options\n .WithInjectionParameter("logger", new ConsoleLogger())\n .WithInjectionParameter("eventBroadcaster", new MessageBus());\n\n// the injection parameters will be passed to DbBackup\'s constructor.\nIJob backup = container.Resolve();\n')))),(0,o.kt)("h2",{id:"initializer--finalizer"},"Initializer / finalizer"),(0,o.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,o.kt)("div",null,(0,o.kt)("p",null,"The container provides specific extension points to let you react to lifetime events of an instantiated service. "),(0,o.kt)("p",null,"For this reason, you can specify ",(0,o.kt)("em",{parentName:"p"},"Initializer")," and ",(0,o.kt)("em",{parentName:"p"},"Finalizer")," delegates. The ",(0,o.kt)("em",{parentName:"p"},"finalizer")," is called upon the service's ",(0,o.kt)("a",{parentName:"p",href:"/docs/guides/scopes#disposal"},"disposal"),", and the ",(0,o.kt)("em",{parentName:"p"},"initializer")," is called upon the service's construction.")),(0,o.kt)("div",null,(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n // delegate that called right after instantiation.\n .WithInitializer((logger, resolver) => logger.OpenFile())\n // delegate that called right before the instance's disposal.\n .WithFinalizer(logger => logger.CloseFile()));\n")))))}g.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/3a87badd.faed2df0.js b/assets/js/3a87badd.faed2df0.js new file mode 100644 index 00000000..8282d11d --- /dev/null +++ b/assets/js/3a87badd.faed2df0.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkwebsite=self.webpackChunkwebsite||[]).push([[80],{3905:(e,t,n)=>{n.d(t,{Zo:()=>p,kt:()=>g});var a=n(7294);function o(e,t,n){return t in e?Object.defineProperty(e,t,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[t]=n,e}function r(e,t){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);t&&(a=a.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),n.push.apply(n,a)}return n}function i(e){for(var t=1;t=0||(o[n]=e[n]);return o}(e,t);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);for(a=0;a=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(o[n]=e[n])}return o}var l=a.createContext({}),c=function(e){var t=a.useContext(l),n=t;return e&&(n="function"==typeof e?e(t):i(i({},t),e)),n},p=function(e){var t=c(e.components);return a.createElement(l.Provider,{value:t},e.children)},u="mdxType",d={inlineCode:"code",wrapper:function(e){var t=e.children;return a.createElement(a.Fragment,{},t)}},m=a.forwardRef((function(e,t){var n=e.components,o=e.mdxType,r=e.originalType,l=e.parentName,p=s(e,["components","mdxType","originalType","parentName"]),u=c(n),m=o,g=u["".concat(l,".").concat(m)]||u[m]||d[m]||r;return n?a.createElement(g,i(i({ref:t},p),{},{components:n})):a.createElement(g,i({ref:t},p))}));function g(e,t){var n=arguments,o=t&&t.mdxType;if("string"==typeof e||o){var r=n.length,i=new Array(r);i[0]=m;var s={};for(var l in t)hasOwnProperty.call(t,l)&&(s[l]=t[l]);s.originalType=e,s[u]="string"==typeof e?e:o,i[1]=s;for(var c=2;c{n.d(t,{Z:()=>i});var a=n(7294),o=n(6010);const r="tabItem_Ymn6";function i(e){let{children:t,hidden:n,className:i}=e;return a.createElement("div",{role:"tabpanel",className:(0,o.Z)(r,i),hidden:n},t)}},4866:(e,t,n)=>{n.d(t,{Z:()=>N});var a=n(7462),o=n(7294),r=n(6010),i=n(2466),s=n(6550),l=n(1980),c=n(7392),p=n(12);function u(e){return function(e){return o.Children.map(e,(e=>{if(!e||(0,o.isValidElement)(e)&&function(e){const{props:t}=e;return!!t&&"object"==typeof t&&"value"in t}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}(e).map((e=>{let{props:{value:t,label:n,attributes:a,default:o}}=e;return{value:t,label:n,attributes:a,default:o}}))}function d(e){const{values:t,children:n}=e;return(0,o.useMemo)((()=>{const e=t??u(n);return function(e){const t=(0,c.l)(e,((e,t)=>e.value===t.value));if(t.length>0)throw new Error(`Docusaurus error: Duplicate values "${t.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[t,n])}function m(e){let{value:t,tabValues:n}=e;return n.some((e=>e.value===t))}function g(e){let{queryString:t=!1,groupId:n}=e;const a=(0,s.k6)(),r=function(e){let{queryString:t=!1,groupId:n}=e;if("string"==typeof t)return t;if(!1===t)return null;if(!0===t&&!n)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return n??null}({queryString:t,groupId:n});return[(0,l._X)(r),(0,o.useCallback)((e=>{if(!r)return;const t=new URLSearchParams(a.location.search);t.set(r,e),a.replace({...a.location,search:t.toString()})}),[r,a])]}function b(e){const{defaultValue:t,queryString:n=!1,groupId:a}=e,r=d(e),[i,s]=(0,o.useState)((()=>function(e){let{defaultValue:t,tabValues:n}=e;if(0===n.length)throw new Error("Docusaurus error: the component requires at least one children component");if(t){if(!m({value:t,tabValues:n}))throw new Error(`Docusaurus error: The has a defaultValue "${t}" but none of its children has the corresponding value. Available values are: ${n.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return t}const a=n.find((e=>e.default))??n[0];if(!a)throw new Error("Unexpected error: 0 tabValues");return a.value}({defaultValue:t,tabValues:r}))),[l,c]=g({queryString:n,groupId:a}),[u,b]=function(e){let{groupId:t}=e;const n=function(e){return e?`docusaurus.tab.${e}`:null}(t),[a,r]=(0,p.Nk)(n);return[a,(0,o.useCallback)((e=>{n&&r.set(e)}),[n,r])]}({groupId:a}),v=(()=>{const e=l??u;return m({value:e,tabValues:r})?e:null})();(0,o.useLayoutEffect)((()=>{v&&s(v)}),[v]);return{selectedValue:i,selectValue:(0,o.useCallback)((e=>{if(!m({value:e,tabValues:r}))throw new Error(`Can't select invalid tab value=${e}`);s(e),c(e),b(e)}),[c,b,r]),tabValues:r}}var v=n(2389);const h="tabList__CuJ",k="tabItem_LNqP";function y(e){let{className:t,block:n,selectedValue:s,selectValue:l,tabValues:c}=e;const p=[],{blockElementScrollPositionUntilNextRender:u}=(0,i.o5)(),d=e=>{const t=e.currentTarget,n=p.indexOf(t),a=c[n].value;a!==s&&(u(t),l(a))},m=e=>{let t=null;switch(e.key){case"Enter":d(e);break;case"ArrowRight":{const n=p.indexOf(e.currentTarget)+1;t=p[n]??p[0];break}case"ArrowLeft":{const n=p.indexOf(e.currentTarget)-1;t=p[n]??p[p.length-1];break}}t?.focus()};return o.createElement("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,r.Z)("tabs",{"tabs--block":n},t)},c.map((e=>{let{value:t,label:n,attributes:i}=e;return o.createElement("li",(0,a.Z)({role:"tab",tabIndex:s===t?0:-1,"aria-selected":s===t,key:t,ref:e=>p.push(e),onKeyDown:m,onClick:d},i,{className:(0,r.Z)("tabs__item",k,i?.className,{"tabs__item--active":s===t})}),n??t)})))}function f(e){let{lazy:t,children:n,selectedValue:a}=e;const r=(Array.isArray(n)?n:[n]).filter(Boolean);if(t){const e=r.find((e=>e.props.value===a));return e?(0,o.cloneElement)(e,{className:"margin-top--md"}):null}return o.createElement("div",{className:"margin-top--md"},r.map(((e,t)=>(0,o.cloneElement)(e,{key:t,hidden:e.props.value!==a}))))}function I(e){const t=b(e);return o.createElement("div",{className:(0,r.Z)("tabs-container",h)},o.createElement(y,(0,a.Z)({},e,t)),o.createElement(f,(0,a.Z)({},e,t)))}function N(e){const t=(0,v.Z)();return o.createElement(I,(0,a.Z)({key:String(t)},e))}},8846:(e,t,n)=>{n.d(t,{Z:()=>s});var a=n(7294);const o="codeDescContainer_ie8f",r="desc_jyqI",i="example_eYlF";function s(e){let{children:t}=e,n=a.Children.toArray(t).filter((e=>e));return a.createElement("div",{className:o},a.createElement("div",{className:r},n[0]),a.createElement("div",{className:i},n[1]))}},9470:(e,t,n)=>{n.r(t),n.d(t,{assets:()=>u,contentTitle:()=>c,default:()=>g,frontMatter:()=>l,metadata:()=>p,toc:()=>d});var a=n(7462),o=(n(7294),n(3905)),r=n(8846),i=n(4866),s=n(5162);const l={},c="Advanced registration",p={unversionedId:"guides/advanced-registration",id:"guides/advanced-registration",title:"Advanced registration",description:"This section is about Stashbox's further configuration options, including the registration configuration API, the registration of factory delegates, multiple implementations, batch registrations, the concept of the Composition Root, and many more.",source:"@site/docs/guides/advanced-registration.md",sourceDirName:"guides",slug:"/guides/advanced-registration",permalink:"/stashbox/docs/guides/advanced-registration",draft:!1,editUrl:"https://github.com/z4kn4fein/stashbox/edit/master/docs/docs/guides/advanced-registration.md",tags:[],version:"current",lastUpdatedBy:"dependabot[bot]",lastUpdatedAt:1699538162,formattedLastUpdatedAt:"Nov 9, 2023",frontMatter:{},sidebar:"docs",previous:{title:"Basic usage",permalink:"/stashbox/docs/guides/basics"},next:{title:"Service resolution",permalink:"/stashbox/docs/guides/service-resolution"}},u={},d=[{value:"Factory registration",id:"factory-registration",level:2},{value:"Factories with parameter overrides",id:"factories-with-parameter-overrides",level:3},{value:"Consider this before using the resolver parameter inside a factory",id:"consider-this-before-using-the-resolver-parameter-inside-a-factory",level:3},{value:"Delegates with dependencies passed as parameters",id:"delegates-with-dependencies-passed-as-parameters",level:4},{value:"Accessing the currently resolving type in factories",id:"accessing-the-currently-resolving-type-in-factories",level:3},{value:"Multiple implementations",id:"multiple-implementations",level:2},{value:"Binding to multiple services",id:"binding-to-multiple-services",level:2},{value:"Batch registration",id:"batch-registration",level:2},{value:"Assembly registration",id:"assembly-registration",level:2},{value:"Composition root",id:"composition-root",level:2},{value:"Injection parameters",id:"injection-parameters",level:2},{value:"Initializer / finalizer",id:"initializer--finalizer",level:2}],m={toc:d};function g(e){let{components:t,...n}=e;return(0,o.kt)("wrapper",(0,a.Z)({},m,n,{components:t,mdxType:"MDXLayout"}),(0,o.kt)("h1",{id:"advanced-registration"},"Advanced registration"),(0,o.kt)("p",null,"This section is about Stashbox's further configuration options, including the registration configuration API, the registration of factory delegates, multiple implementations, batch registrations, the concept of the ",(0,o.kt)("a",{parentName:"p",href:"https://blog.ploeh.dk/2011/07/28/CompositionRoot/"},"Composition Root"),", and many more."),(0,o.kt)("admonition",{type:"info"},(0,o.kt)("p",{parentName:"admonition"},"This section won't cover all the available options of the registrations API, but you can find them ",(0,o.kt)("a",{parentName:"p",href:"/docs/configuration/registration-configuration"},"here"),".")),(0,o.kt)("h2",{id:"factory-registration"},"Factory registration"),(0,o.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,o.kt)("div",null,(0,o.kt)("p",null,"You can bind a factory delegate to a registration that the container will invoke directly to instantiate your service. "),(0,o.kt)("p",null,"You can use parameter-less and custom parameterized delegates as a factory. ",(0,o.kt)("a",{parentName:"p",href:"/docs/configuration/registration-configuration#factory"},"Here")," is the list of all available options."),(0,o.kt)("p",null,"You can also get the current ",(0,o.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#dependency-resolver"},"dependency resolver")," as a delegate parameter to resolve any additional dependencies required for the service construction.")),(0,o.kt)("div",null,(0,o.kt)(i.Z,{mdxType:"Tabs"},(0,o.kt)(s.Z,{value:"Parameter-less",label:"Parameter-less",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithFactory(() => new ConsoleLogger());\n\n// the container uses the factory for instantiation.\nIJob job = container.Resolve();\n"))),(0,o.kt)(s.Z,{value:"Parameterized",label:"Parameterized",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithFactory(logger => new DbBackup(logger));\n\n// the container uses the factory for instantiation.\nIJob job = container.Resolve();\n"))),(0,o.kt)(s.Z,{value:"Resolver parameter",label:"Resolver parameter",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithFactory(resolver => new DbBackup(resolver.Resolve()));\n \n// the container uses the factory for instantiation.\nIJob job = container.Resolve();\n")))))),(0,o.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,o.kt)("div",null,(0,o.kt)("p",null,"Delegate factories are useful when your service's instantiation is not straight-forward for the container, like when it depends on something that is not available at resolution time. E.g., a connection string.")),(0,o.kt)("div",null,(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register(options => options\n .WithFactory(logger => \n new DbBackup(Configuration["DbConnectionString"], logger));\n')))),(0,o.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,o.kt)("div",null,(0,o.kt)("h3",{id:"factories-with-parameter-overrides"},"Factories with parameter overrides"),(0,o.kt)("p",null,"Stashbox can implicitly ",(0,o.kt)("a",{parentName:"p",href:"/docs/advanced/wrappers-resolvers#delegate"},"wrap")," your service in a ",(0,o.kt)("inlineCode",{parentName:"p"},"Delegate")," and lets you pass parameters that can override your service's dependencies. Moreover, you can register your own custom delegate that the container will resolve when you request your service wrapped in a ",(0,o.kt)("inlineCode",{parentName:"p"},"Delegate"),".")),(0,o.kt)("div",null,(0,o.kt)(i.Z,{groupId:"generic-runtime-apis",mdxType:"Tabs"},(0,o.kt)(s.Z,{value:"Generic API",label:"Generic API",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},'container.RegisterFunc((connectionString, resolver) => \n new DbBackup(connectionString, resolver.Resolve()));\n\nFunc backupFactory = container.Resolve>();\nIJob dbBackup = backupFactory(Configuration["ConnectionString"]);\n'))),(0,o.kt)(s.Z,{value:"Runtime type API",label:"Runtime type API",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},'container.RegisterFunc((connectionString, resolver) => \n new DbBackup(connectionString, resolver.Resolve()));\n\nDelegate backupFactory = container.ResolveFactory(typeof(IJob), \n parameterTypes: new[] { typeof(string) });\nIJob dbBackup = backupFactory.DynamicInvoke(Configuration["ConnectionString"]);\n')))))),(0,o.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,o.kt)("div",null,(0,o.kt)("p",null,"If a service has multiple constructors, the container visits those first, that has matching parameters passed to the factory, with respecting the additional ",(0,o.kt)("a",{parentName:"p",href:"/docs/configuration/registration-configuration#constructor-selection"},"constructor selection rules"),".")),(0,o.kt)("div",null,(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"class Service\n{\n public Service(int number) { }\n public Service(string text) { }\n}\n\ncontainer.Register();\n\n// create the factory with an int input parameter.\nvar func = constainer.Resolve>();\n\n// the constructor with the int param \n// is used for instantiation.\nvar service = func(2);\n")))),(0,o.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,o.kt)("div",null,(0,o.kt)("h3",{id:"consider-this-before-using-the-resolver-parameter-inside-a-factory"},"Consider this before using the resolver parameter inside a factory"),(0,o.kt)("p",null,"Delegate factories are a black-box for the container. It doesn't have control over what's happening inside a delegate, which means when you resolve additional dependencies with the ",(0,o.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#dependency-resolver"},"dependency resolver")," parameter, they could easily bypass the ",(0,o.kt)("a",{parentName:"p",href:"/docs/diagnostics/validation#lifetime-validation"},"lifetime")," and ",(0,o.kt)("a",{parentName:"p",href:"/docs/diagnostics/validation#circular-dependency"},"circular dependency")," validations. Fortunately, you have the option to keep them validated anyway with parameterized factory delegates."),(0,o.kt)("h4",{id:"delegates-with-dependencies-passed-as-parameters"},"Delegates with dependencies passed as parameters"),(0,o.kt)("p",null,"Rather than using the ",(0,o.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#dependency-resolver"},"dependency resolver")," parameter inside the factory, let the container inject the dependencies into the delegate as parameters. This way, the ",(0,o.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#resolution-tree"},"resolution tree's")," integrity remains stable because no service resolution happens inside the black-box, and each parameter is validated.")),(0,o.kt)("div",null,(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"interface IEventProcessor { }\n\nclass EventProcessor : IEventProcessor\n{\n public EventProcessor(ILogger logger, IEventValidator validator)\n { }\n}\n\ncontainer.Register();\ncontainer.Register();\n\ncontainer.Register(options => options\n // Ilogger and IEventValidator instances are injected\n // by the container at resolution time, so they will be\n // validated against circular and captive dependencies.\n .WithFactory((logger, validator) => \n new EventProcessor(logger, validator));\n\n// the container resolves ILogger and IEventValidator first, then\n// it passes them to the factory as delegate parameters.\nIEventProcessor processor = container.Resolve();\n")))),(0,o.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,o.kt)("div",null,(0,o.kt)("h3",{id:"accessing-the-currently-resolving-type-in-factories"},"Accessing the currently resolving type in factories"),(0,o.kt)("p",null,"To access the currently resolving type in factory delegates, you can set the ",(0,o.kt)("inlineCode",{parentName:"p"},"TypeInformation")," type as an input parameter of the factory.\nThe ",(0,o.kt)("inlineCode",{parentName:"p"},"TypeInformation")," holds every reflected context information about the currently resolving type. "),(0,o.kt)("p",null,"This can be useful when the resolution is, e.g., in an open generic context, and we want to know which closed generic variant is requested.")),(0,o.kt)("div",null,(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"interface IService { }\n\nclass Service : IService { }\n\ncontainer.Register(typeof(IService<>), typeof(Service<>), options => \n options.WithFactory(typeInfo => \n {\n // typeInfo.Type here holds the actual type like\n // IService based on the resolution request below.\n }));\n \ncontainer.Resolve>();\n")))),(0,o.kt)("h2",{id:"multiple-implementations"},"Multiple implementations"),(0,o.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,o.kt)("div",null,(0,o.kt)("p",null,"As we previously saw in the ",(0,o.kt)("a",{parentName:"p",href:"/docs/guides/basics#named-registration"},"Named registration")," topic, Stashbox allows you to have multiple implementations bound to a particular ",(0,o.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type"),". You can use names to distinguish them, but you can also access them by requesting a typed collection using the ",(0,o.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type"),"."),(0,o.kt)("admonition",{type:"note"},(0,o.kt)("p",{parentName:"admonition"},"The returned collection is in the same order as the services were registered.\nAlso, to request a collection, you can use any interface implemented by an array."))),(0,o.kt)("div",null,(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register();\ncontainer.Register();\ncontainer.Register();\n")),(0,o.kt)(i.Z,{mdxType:"Tabs"},(0,o.kt)(s.Z,{value:"ResolveAll",label:"ResolveAll",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"// jobs contain all three services in registration order.\nIEnumerable jobs = container.ResolveAll();\n"))),(0,o.kt)(s.Z,{value:"Array",label:"Array",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"// jobs contain all three services in registration order.\nIJob[] jobs = container.Resolve();\n"))),(0,o.kt)(s.Z,{value:"IEnumerable",label:"IEnumerable",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"// jobs contain all three services in registration order.\nIEnumerable jobs = container.Resolve>();\n"))),(0,o.kt)(s.Z,{value:"IList",label:"IList",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"// jobs contain all three services in registration order.\nIList jobs = container.Resolve>();\n"))),(0,o.kt)(s.Z,{value:"ICollection",label:"ICollection",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"// jobs contain all three services in registration order.\nICollection jobs = container.Resolve>();\n")))))),(0,o.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,o.kt)("div",null,(0,o.kt)("p",null,"When you have multiple implementations registered to a service, a request to the ",(0,o.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type")," without a name will return the ",(0,o.kt)("strong",{parentName:"p"},"last registered implementation"),"."),(0,o.kt)("admonition",{type:"info"},(0,o.kt)("p",{parentName:"admonition"},"Not only names can be used to distinguish registrations, ",(0,o.kt)("a",{parentName:"p",href:"/docs/guides/service-resolution#conditional-resolution"},"conditions"),", ",(0,o.kt)("a",{parentName:"p",href:"/docs/guides/scopes#named-scopes"},"named scopes"),", and ",(0,o.kt)("a",{parentName:"p",href:"/docs/advanced/wrappers-resolvers#metadata--tuple"},"metadata")," can also influence the results."))),(0,o.kt)("div",null,(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register();\ncontainer.Register();\ncontainer.Register();\n\n// job will be the ImageProcess.\nIJob job = container.Resolve();\n")))),(0,o.kt)("h2",{id:"binding-to-multiple-services"},"Binding to multiple services"),(0,o.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,o.kt)("div",null,(0,o.kt)("p",null,"When you have a service that implements multiple interfaces, you have the option to bind its registration to all or some of those additional interfaces or base types."),(0,o.kt)("p",null,"Suppose we have the following class declaration:"),(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"class DbBackup : IJob, IScheduledJob\n{ \n public DbBackup() { }\n}\n"))),(0,o.kt)("div",null,(0,o.kt)(i.Z,{mdxType:"Tabs"},(0,o.kt)(s.Z,{value:"To another type",label:"To another type",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .AsServiceAlso());\n\nIJob job = container.Resolve(); // DbBackup\nIScheduledJob job = container.Resolve(); // DbBackup\nDbBackup job = container.Resolve(); // error, not found\n"))),(0,o.kt)(s.Z,{value:"To all implemented types",label:"To all implemented types",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .AsImplementedTypes());\n\nIJob job = container.Resolve(); // DbBackup\nIScheduledJob job = container.Resolve(); // DbBackup\nDbBackup job = container.Resolve(); // DbBackup\n")))))),(0,o.kt)("h2",{id:"batch-registration"},"Batch registration"),(0,o.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,o.kt)("div",null,(0,o.kt)("p",null,"You have the option to register multiple services in a single registration operation. "),(0,o.kt)("p",null,(0,o.kt)("strong",{parentName:"p"},"Filters (optional):"),"\nFirst, the container will use the ",(0,o.kt)("em",{parentName:"p"},"implementation filter")," action to select only those types from the collection we want to register. When we have those, the container will execute the ",(0,o.kt)("em",{parentName:"p"},"service filter")," on their implemented interfaces and base classes to select which ",(0,o.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type")," they should be mapped to."),(0,o.kt)("admonition",{type:"note"},(0,o.kt)("p",{parentName:"admonition"},"Framework types like ",(0,o.kt)("inlineCode",{parentName:"p"},"IDisposable")," are excluded from being considered as a ",(0,o.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type")," by default.")),(0,o.kt)("admonition",{type:"tip"},(0,o.kt)("p",{parentName:"admonition"},"You can use the registration configuration API to configure individual registrations."))),(0,o.kt)("div",null,(0,o.kt)(i.Z,{mdxType:"Tabs"},(0,o.kt)(s.Z,{value:"Default",label:"Default",mdxType:"TabItem"},(0,o.kt)("p",null,"This example will register three types to all their implemented interfaces, extended base classes, and to themselves (",(0,o.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#self-registration"},"self registration"),") without any filter:"),(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterTypes(new[] \n { \n typeof(DbBackup), \n typeof(ConsoleLogger), \n typeof(StorageCleanup) \n });\n\nIEnumerable jobs = container.ResolveAll(); // 2 items\nILogger logger = container.Resolve(); // ConsoleLogger\nIJob job = container.Resolve(); // StorageCleanup\nDbBackup backup = container.Resolve(); // DbBackup\n"))),(0,o.kt)(s.Z,{value:"Filters",label:"Filters",mdxType:"TabItem"},(0,o.kt)("p",null,"In this example, we assume that ",(0,o.kt)("inlineCode",{parentName:"p"},"DbBackup")," and ",(0,o.kt)("inlineCode",{parentName:"p"},"StorageCleanup")," are implementing ",(0,o.kt)("inlineCode",{parentName:"p"},"IDisposable")," besides ",(0,o.kt)("inlineCode",{parentName:"p"},"IJob")," and also extending a ",(0,o.kt)("inlineCode",{parentName:"p"},"JobBase")," abstract class."),(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterTypes(new[] \n { typeof(DbBackup), typeof(ConsoleLogger), typeof(StorageCleanup) },\n // implementation filter, only those implementations that implements IDisposable\n impl => typeof(IDisposable).IsAssignableFrom(impl),\n // service filter, register them to base classes only\n (impl, service) => service.IsAbstract && !service.IsInterface);\n\nIEnumerable jobs = container.ResolveAll(); // 0 items\nIEnumerable jobs = container.ResolveAll(); // 2 items\nILogger logger = container.Resolve(); // error, not found\nDbBackup backup = container.Resolve(); // DbBackup\n"))),(0,o.kt)(s.Z,{value:"Without self",label:"Without self",mdxType:"TabItem"},(0,o.kt)("p",null,"This example ignores the ",(0,o.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#self-registration"},"self registrations")," completely:"),(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterTypes(new[] \n { \n typeof(DbBackup), \n typeof(ConsoleLogger), \n typeof(StorageCleanup)\n },\n registerSelf: false);\n\nIEnumerable jobs = container.ResolveAll(); // 2 items\nILogger logger = container.Resolve(); // ConsoleLogger\nDbBackup backup = container.Resolve(); // error, not found\nConsoleLogger logger = container.Resolve(); // error, not found\n"))),(0,o.kt)(s.Z,{value:"Registration options",label:"Registration options",mdxType:"TabItem"},(0,o.kt)("p",null,"This example will configure all registrations mapped to ",(0,o.kt)("inlineCode",{parentName:"p"},"ILogger")," as ",(0,o.kt)("inlineCode",{parentName:"p"},"Singleton"),":"),(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterTypes(new[] \n { \n typeof(DbBackup), \n typeof(ConsoleLogger), \n typeof(StorageCleanup)\n },\n configurator: options => \n {\n if (options.HasServiceType())\n options.WithSingletonLifetime();\n });\n\nILogger logger = container.Resolve(); // ConsoleLogger\nILogger newLogger = container.Resolve(); // the same ConsoleLogger\nIEnumerable jobs = container.ResolveAll(); // 2 items\n")))))),(0,o.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,o.kt)("div",null,(0,o.kt)("p",null,"Another type of service filter is the ",(0,o.kt)("inlineCode",{parentName:"p"},".RegisterTypesAs()")," method, which registers only those types that implements the ",(0,o.kt)("inlineCode",{parentName:"p"},"T")," ",(0,o.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type"),"."),(0,o.kt)("admonition",{type:"note"},(0,o.kt)("p",{parentName:"admonition"},"This method also accepts an implementation filter and a registration configurator action like ",(0,o.kt)("inlineCode",{parentName:"p"},".RegisterTypes()"),".")),(0,o.kt)("admonition",{type:"caution"},(0,o.kt)("p",{parentName:"admonition"},(0,o.kt)("inlineCode",{parentName:"p"},".RegisterTypesAs()")," doesn't create ",(0,o.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#self-registration"},"self registrations")," as it only maps the implementations to the given ",(0,o.kt)("inlineCode",{parentName:"p"},"T")," ",(0,o.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type"),"."))),(0,o.kt)("div",null,(0,o.kt)(i.Z,{groupId:"generic-runtime-apis",mdxType:"Tabs"},(0,o.kt)(s.Z,{value:"Generic API",label:"Generic API",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterTypesAs(new[] \n { \n typeof(DbBackup), \n typeof(ConsoleLogger), \n typeof(StorageCleanup) \n });\n\nIEnumerable jobs = container.ResolveAll(); // 2 items\nILogger logger = container.Resolve(); // error, not found\nIJob job = container.Resolve(); // StorageCleanup\nDbBackup backup = container.Resolve(); // error, not found\n"))),(0,o.kt)(s.Z,{value:"Runtime type API",label:"Runtime type API",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterTypesAs(typeof(IJob), new[] \n { \n typeof(DbBackup), \n typeof(ConsoleLogger), \n typeof(StorageCleanup) \n });\n\nIEnumerable jobs = container.ResolveAll(); // 2 items\nILogger logger = container.Resolve(); // error, not found\nIJob job = container.Resolve(); // StorageCleanup\nDbBackup backup = container.Resolve(); // error, not found\n")))))),(0,o.kt)("h2",{id:"assembly-registration"},"Assembly registration"),(0,o.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,o.kt)("div",null,(0,o.kt)("p",null,"The batch registration API ",(0,o.kt)("em",{parentName:"p"},"(filters, registration configuration action, self-registration)")," is also usable for registering services from given assemblies."),(0,o.kt)("p",null,"In this example, we assume that the same three services we used in the ",(0,o.kt)("a",{parentName:"p",href:"#batch-registration"},"batch registration")," section are in the same assembly."),(0,o.kt)("admonition",{type:"info"},(0,o.kt)("p",{parentName:"admonition"},"The container also detects and registers open-generic definitions (when applicable) from the supplied type collection. You can read about ",(0,o.kt)("a",{parentName:"p",href:"/docs/advanced/generics#open-generics"},"open-generics here"),"."))),(0,o.kt)("div",null,(0,o.kt)(i.Z,{mdxType:"Tabs"},(0,o.kt)(s.Z,{value:"Single assembly",label:"Single assembly",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterAssembly(typeof(DbBackup).Assembly,\n // service filter, register to interfaces only\n serviceTypeSelector: (impl, service) => service.IsInterface,\n registerSelf: false,\n configurator: options => options.WithoutDisposalTracking()\n);\n\nIEnumerable jobs = container.ResolveAll(); // 2 items\nIEnumerable jobs = container.ResolveAll(); // 0 items\nILogger logger = container.Resolve(); // ConsoleLogger\nDbBackup backup = container.Resolve(); // error, not found\n"))),(0,o.kt)(s.Z,{value:"Multiple assemblies",label:"Multiple assemblies",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterAssemblies(new[] \n { \n typeof(DbBackup).Assembly, \n typeof(JobFromAnotherAssembly).Assembly \n },\n // service filter, register to interfaces only\n serviceTypeSelector: (impl, service) => service.IsInterface,\n registerSelf: false,\n configurator: options => options.WithoutDisposalTracking()\n);\n\nIEnumerable jobs = container.ResolveAll(); // 2 items\nIEnumerable jobs = container.ResolveAll(); // 0 items\nILogger logger = container.Resolve(); // ConsoleLogger\nDbBackup backup = container.Resolve(); // error, not found\n"))),(0,o.kt)(s.Z,{value:"Containing type",label:"Containing type",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterAssemblyContaining(\n // service filter, register to interfaces only\n serviceTypeSelector: (impl, service) => service.IsInterface,\n registerSelf: false,\n configurator: options => options.WithoutDisposalTracking()\n);\n\nIEnumerable jobs = container.ResolveAll(); // 2 items\nIEnumerable jobs = container.ResolveAll(); // 0 items\nILogger logger = container.Resolve(); // ConsoleLogger\nDbBackup backup = container.Resolve(); // error, not found\n")))))),(0,o.kt)("h2",{id:"composition-root"},"Composition root"),(0,o.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,o.kt)("div",null,(0,o.kt)("p",null,"The ",(0,o.kt)("a",{parentName:"p",href:"https://blog.ploeh.dk/2011/07/28/CompositionRoot/"},"Composition Root")," is an entry point where all services required to make a component functional are wired together."),(0,o.kt)("p",null,"Stashbox provides an ",(0,o.kt)("inlineCode",{parentName:"p"},"ICompositionRoot")," interface that can be used to define an entry point for a given component or even for an entire assembly. "),(0,o.kt)("p",null,"You can wire up your ",(0,o.kt)("em",{parentName:"p"},"composition root")," implementation with ",(0,o.kt)("inlineCode",{parentName:"p"},"ComposeBy()"),", or you can let the container find and execute all available ",(0,o.kt)("em",{parentName:"p"},"composition root")," implementations within an assembly."),(0,o.kt)("admonition",{type:"note"},(0,o.kt)("p",{parentName:"admonition"},"Your ",(0,o.kt)("inlineCode",{parentName:"p"},"ICompositionRoot")," implementation also can have dependencies that the container will resolve."))),(0,o.kt)("div",null,(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"class ExampleRoot : ICompositionRoot\n{\n public ExampleRoot(IDependency rootDependency)\n { }\n\n public void Compose(IStashboxContainer container)\n {\n container.Register();\n container.Register();\n }\n}\n")),(0,o.kt)(i.Z,{mdxType:"Tabs"},(0,o.kt)(s.Z,{value:"Single",label:"Single",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"// compose a single root.\ncontainer.ComposeBy();\n"))),(0,o.kt)(s.Z,{value:"Assembly",label:"Assembly",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"// compose every root in the given assembly.\ncontainer.ComposeAssembly(typeof(IServiceA).Assembly);\n"))),(0,o.kt)(s.Z,{value:"Override",label:"Override",mdxType:"TabItem"},(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"// compose a single root with dependency override.\ncontainer.ComposeBy(new CustomRootDependency());\n")))))),(0,o.kt)("h2",{id:"injection-parameters"},"Injection parameters"),(0,o.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,o.kt)("div",null,(0,o.kt)("p",null,"If you have pre-evaluated dependencies you'd like to inject at resolution time, you can set them as injection parameters during registration. "),(0,o.kt)("admonition",{type:"note"},(0,o.kt)("p",{parentName:"admonition"},"Injection parameter names are matched to constructor arguments or field/property names."))),(0,o.kt)("div",null,(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register(options => options\n .WithInjectionParameter("logger", new ConsoleLogger())\n .WithInjectionParameter("eventBroadcaster", new MessageBus());\n\n// the injection parameters will be passed to DbBackup\'s constructor.\nIJob backup = container.Resolve();\n')))),(0,o.kt)("h2",{id:"initializer--finalizer"},"Initializer / finalizer"),(0,o.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,o.kt)("div",null,(0,o.kt)("p",null,"The container provides specific extension points to let you react to lifetime events of an instantiated service. "),(0,o.kt)("p",null,"For this reason, you can specify ",(0,o.kt)("em",{parentName:"p"},"Initializer")," and ",(0,o.kt)("em",{parentName:"p"},"Finalizer")," delegates. The ",(0,o.kt)("em",{parentName:"p"},"finalizer")," is called upon the service's ",(0,o.kt)("a",{parentName:"p",href:"/docs/guides/scopes#disposal"},"disposal"),", and the ",(0,o.kt)("em",{parentName:"p"},"initializer")," is called upon the service's construction.")),(0,o.kt)("div",null,(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n // delegate that called right after instantiation.\n .WithInitializer((logger, resolver) => logger.OpenFile())\n // delegate that called right before the instance's disposal.\n .WithFinalizer(logger => logger.CloseFile()));\n")))))}g.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/4047545b.25600c2c.js b/assets/js/4047545b.25600c2c.js deleted file mode 100644 index b6cb861d..00000000 --- a/assets/js/4047545b.25600c2c.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkwebsite=self.webpackChunkwebsite||[]).push([[371],{3905:(e,t,n)=>{n.d(t,{Zo:()=>p,kt:()=>g});var o=n(7294);function i(e,t,n){return t in e?Object.defineProperty(e,t,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[t]=n,e}function r(e,t){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var o=Object.getOwnPropertySymbols(e);t&&(o=o.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),n.push.apply(n,o)}return n}function a(e){for(var t=1;t=0||(i[n]=e[n]);return i}(e,t);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);for(o=0;o=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(i[n]=e[n])}return i}var l=o.createContext({}),c=function(e){var t=o.useContext(l),n=t;return e&&(n="function"==typeof e?e(t):a(a({},t),e)),n},p=function(e){var t=c(e.components);return o.createElement(l.Provider,{value:t},e.children)},d="mdxType",u={inlineCode:"code",wrapper:function(e){var t=e.children;return o.createElement(o.Fragment,{},t)}},m=o.forwardRef((function(e,t){var n=e.components,i=e.mdxType,r=e.originalType,l=e.parentName,p=s(e,["components","mdxType","originalType","parentName"]),d=c(n),m=i,g=d["".concat(l,".").concat(m)]||d[m]||u[m]||r;return n?o.createElement(g,a(a({ref:t},p),{},{components:n})):o.createElement(g,a({ref:t},p))}));function g(e,t){var n=arguments,i=t&&t.mdxType;if("string"==typeof e||i){var r=n.length,a=new Array(r);a[0]=m;var s={};for(var l in t)hasOwnProperty.call(t,l)&&(s[l]=t[l]);s.originalType=e,s[d]="string"==typeof e?e:i,a[1]=s;for(var c=2;c{n.d(t,{Z:()=>a});var o=n(7294),i=n(6010);const r="tabItem_Ymn6";function a(e){let{children:t,hidden:n,className:a}=e;return o.createElement("div",{role:"tabpanel",className:(0,i.Z)(r,a),hidden:n},t)}},4866:(e,t,n)=>{n.d(t,{Z:()=>I});var o=n(7462),i=n(7294),r=n(6010),a=n(2466),s=n(6550),l=n(1980),c=n(7392),p=n(12);function d(e){return function(e){return i.Children.map(e,(e=>{if(!e||(0,i.isValidElement)(e)&&function(e){const{props:t}=e;return!!t&&"object"==typeof t&&"value"in t}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}(e).map((e=>{let{props:{value:t,label:n,attributes:o,default:i}}=e;return{value:t,label:n,attributes:o,default:i}}))}function u(e){const{values:t,children:n}=e;return(0,i.useMemo)((()=>{const e=t??d(n);return function(e){const t=(0,c.l)(e,((e,t)=>e.value===t.value));if(t.length>0)throw new Error(`Docusaurus error: Duplicate values "${t.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[t,n])}function m(e){let{value:t,tabValues:n}=e;return n.some((e=>e.value===t))}function g(e){let{queryString:t=!1,groupId:n}=e;const o=(0,s.k6)(),r=function(e){let{queryString:t=!1,groupId:n}=e;if("string"==typeof t)return t;if(!1===t)return null;if(!0===t&&!n)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return n??null}({queryString:t,groupId:n});return[(0,l._X)(r),(0,i.useCallback)((e=>{if(!r)return;const t=new URLSearchParams(o.location.search);t.set(r,e),o.replace({...o.location,search:t.toString()})}),[r,o])]}function h(e){const{defaultValue:t,queryString:n=!1,groupId:o}=e,r=u(e),[a,s]=(0,i.useState)((()=>function(e){let{defaultValue:t,tabValues:n}=e;if(0===n.length)throw new Error("Docusaurus error: the component requires at least one children component");if(t){if(!m({value:t,tabValues:n}))throw new Error(`Docusaurus error: The has a defaultValue "${t}" but none of its children has the corresponding value. Available values are: ${n.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return t}const o=n.find((e=>e.default))??n[0];if(!o)throw new Error("Unexpected error: 0 tabValues");return o.value}({defaultValue:t,tabValues:r}))),[l,c]=g({queryString:n,groupId:o}),[d,h]=function(e){let{groupId:t}=e;const n=function(e){return e?`docusaurus.tab.${e}`:null}(t),[o,r]=(0,p.Nk)(n);return[o,(0,i.useCallback)((e=>{n&&r.set(e)}),[n,r])]}({groupId:o}),k=(()=>{const e=l??d;return m({value:e,tabValues:r})?e:null})();(0,i.useLayoutEffect)((()=>{k&&s(k)}),[k]);return{selectedValue:a,selectValue:(0,i.useCallback)((e=>{if(!m({value:e,tabValues:r}))throw new Error(`Can't select invalid tab value=${e}`);s(e),c(e),h(e)}),[c,h,r]),tabValues:r}}var k=n(2389);const v="tabList__CuJ",f="tabItem_LNqP";function y(e){let{className:t,block:n,selectedValue:s,selectValue:l,tabValues:c}=e;const p=[],{blockElementScrollPositionUntilNextRender:d}=(0,a.o5)(),u=e=>{const t=e.currentTarget,n=p.indexOf(t),o=c[n].value;o!==s&&(d(t),l(o))},m=e=>{let t=null;switch(e.key){case"Enter":u(e);break;case"ArrowRight":{const n=p.indexOf(e.currentTarget)+1;t=p[n]??p[0];break}case"ArrowLeft":{const n=p.indexOf(e.currentTarget)-1;t=p[n]??p[p.length-1];break}}t?.focus()};return i.createElement("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,r.Z)("tabs",{"tabs--block":n},t)},c.map((e=>{let{value:t,label:n,attributes:a}=e;return i.createElement("li",(0,o.Z)({role:"tab",tabIndex:s===t?0:-1,"aria-selected":s===t,key:t,ref:e=>p.push(e),onKeyDown:m,onClick:u},a,{className:(0,r.Z)("tabs__item",f,a?.className,{"tabs__item--active":s===t})}),n??t)})))}function b(e){let{lazy:t,children:n,selectedValue:o}=e;const r=(Array.isArray(n)?n:[n]).filter(Boolean);if(t){const e=r.find((e=>e.props.value===o));return e?(0,i.cloneElement)(e,{className:"margin-top--md"}):null}return i.createElement("div",{className:"margin-top--md"},r.map(((e,t)=>(0,i.cloneElement)(e,{key:t,hidden:e.props.value!==o}))))}function N(e){const t=h(e);return i.createElement("div",{className:(0,r.Z)("tabs-container",v)},i.createElement(y,(0,o.Z)({},e,t)),i.createElement(b,(0,o.Z)({},e,t)))}function I(e){const t=(0,k.Z)();return i.createElement(N,(0,o.Z)({key:String(t)},e))}},8846:(e,t,n)=>{n.d(t,{Z:()=>s});var o=n(7294);const i="codeDescContainer_ie8f",r="desc_jyqI",a="example_eYlF";function s(e){let{children:t}=e,n=o.Children.toArray(t).filter((e=>e));return o.createElement("div",{className:i},o.createElement("div",{className:r},n[0]),o.createElement("div",{className:a},n[1]))}},3419:(e,t,n)=>{n.r(t),n.d(t,{assets:()=>d,contentTitle:()=>c,default:()=>g,frontMatter:()=>l,metadata:()=>p,toc:()=>u});var o=n(7462),i=(n(7294),n(3905)),r=n(8846),a=n(4866),s=n(5162);const l={},c="Registration configuration",p={unversionedId:"configuration/registration-configuration",id:"configuration/registration-configuration",title:"Registration configuration",description:"Most of the registration methods have an Action parameter, enabling several customization options on the given registration.",source:"@site/docs/configuration/registration-configuration.md",sourceDirName:"configuration",slug:"/configuration/registration-configuration",permalink:"/stashbox/docs/configuration/registration-configuration",draft:!1,editUrl:"https://github.com/z4kn4fein/stashbox/edit/master/docs/docs/configuration/registration-configuration.md",tags:[],version:"current",lastUpdatedBy:"dependabot[bot]",lastUpdatedAt:1697644420,formattedLastUpdatedAt:"Oct 18, 2023",frontMatter:{},sidebar:"docs",previous:{title:"Scopes",permalink:"/stashbox/docs/guides/scopes"},next:{title:"Container configuration",permalink:"/stashbox/docs/configuration/container-configuration"}},d={},u=[{value:"General options",id:"general-options",level:2},{value:"WithName",id:"withname",level:3},{value:"WithInstance",id:"withinstance",level:3},{value:"WithoutDisposalTracking",id:"withoutdisposaltracking",level:3},{value:"WithMetadata",id:"withmetadata",level:3},{value:"WithDynamicResolution",id:"withdynamicresolution",level:3},{value:"HasServiceType",id:"hasservicetype",level:3},{value:"Initializer / finalizer",id:"initializer--finalizer",level:2},{value:"WithFinalizer",id:"withfinalizer",level:3},{value:"WithInitializer",id:"withinitializer",level:3},{value:"Replace",id:"replace",level:2},{value:"Multiple services",id:"multiple-services",level:2},{value:"AsImplementedTypes",id:"asimplementedtypes",level:3},{value:"AsServiceAlso",id:"asservicealso",level:3},{value:"Dependency configuration",id:"dependency-configuration",level:2},{value:"Lifetime",id:"lifetime",level:2},{value:"WithSingletonLifetime",id:"withsingletonlifetime",level:3},{value:"WithScopedLifetime",id:"withscopedlifetime",level:3},{value:"WithPerScopedRequestLifetime",id:"withperscopedrequestlifetime",level:3},{value:"WithLifetime",id:"withlifetime",level:3},{value:"Conditions",id:"conditions",level:2},{value:"WhenHas",id:"whenhas",level:3},{value:"WhenResolutionPathHas",id:"whenresolutionpathhas",level:3},{value:"WhenDependantIs",id:"whendependantis",level:3},{value:"WhenInResolutionPathOf",id:"wheninresolutionpathof",level:3},{value:"When",id:"when",level:3},{value:"Constructor selection",id:"constructor-selection",level:2},{value:"WithConstructorSelectionRule",id:"withconstructorselectionrule",level:3},{value:"PreferMostParameters",id:"prefermostparameters",level:4},{value:"PreferLeastParameters",id:"preferleastparameters",level:4},{value:"Custom",id:"custom",level:4},{value:"WithConstructorByArgumentTypes",id:"withconstructorbyargumenttypes",level:3},{value:"WithConstructorByArguments",id:"withconstructorbyarguments",level:3},{value:"Property / field Injection",id:"property--field-injection",level:2},{value:"WithAutoMemberInjection",id:"withautomemberinjection",level:3},{value:"PropertiesWithPublicSetter",id:"propertieswithpublicsetter",level:4},{value:"PropertiesWithLimitedAccess",id:"propertieswithlimitedaccess",level:4},{value:"PrivateFields",id:"privatefields",level:4},{value:"Combined rules",id:"combined-rules",level:4},{value:"Member selection filter",id:"member-selection-filter",level:4},{value:"Injection parameters",id:"injection-parameters",level:2},{value:"WithInjectionParameters",id:"withinjectionparameters",level:3},{value:"WithInjectionParameter",id:"withinjectionparameter",level:3},{value:"Factory",id:"factory",level:2},{value:"Scope definition",id:"scope-definition",level:2},{value:"InNamedScope",id:"innamedscope",level:3},{value:"InScopeDefinedBy",id:"inscopedefinedby",level:3},{value:"DefinesScope",id:"definesscope",level:3},{value:"Decorator specific",id:"decorator-specific",level:2},{value:"WhenDecoratedServiceIs",id:"whendecoratedserviceis",level:3},{value:"WhenDecoratedServiceHas",id:"whendecoratedservicehas",level:3},{value:"Unknown registration specific",id:"unknown-registration-specific",level:2},{value:"SetImplementationType",id:"setimplementationtype",level:3},{value:"Skip",id:"skip",level:3}],m={toc:u};function g(e){let{components:t,...n}=e;return(0,i.kt)("wrapper",(0,o.Z)({},m,n,{components:t,mdxType:"MDXLayout"}),(0,i.kt)("h1",{id:"registration-configuration"},"Registration configuration"),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("p",null,"Most of the registration methods have an ",(0,i.kt)("inlineCode",{parentName:"p"},"Action")," parameter, enabling several customization options on the given registration."),(0,i.kt)("p",null,"Here are three examples that show how the API's usage looks like.\nThey cover the exact functionalities you've read about in the ",(0,i.kt)("a",{parentName:"p",href:"/docs/guides/basics"},"basics")," section but are achieved with the options API.")),(0,i.kt)("div",null,(0,i.kt)(a.Z,{mdxType:"Tabs"},(0,i.kt)(s.Z,{value:"Named",label:"Named",mdxType:"TabItem"},(0,i.kt)("p",null,"This is how you can use the options API to set a registration's name:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register(options => options\n .WithName("DbBackup"));\n'))),(0,i.kt)(s.Z,{value:"Lifetime",label:"Lifetime",mdxType:"TabItem"},(0,i.kt)("p",null,"It was mentioned in the ",(0,i.kt)("a",{parentName:"p",href:"/docs/guides/basics#lifetime-shortcuts"},"Lifetime shortcuts")," section, that those methods are only sugars; under the curtain, they are also using this API:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithLifetime(Lifetimes.Singleton));\n"))),(0,i.kt)(s.Z,{value:"Instance",label:"Instance",mdxType:"TabItem"},(0,i.kt)("p",null,"An example of how you can register an instance with the options API:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithInstance(new DbBackup()));\n")))))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("p",null,"The registration configuration API is fluent, which means all option methods can be chained after each other.\nThis provides an easier way to configure complicated registrations.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register(options => options\n .WithName("DbBackup")\n .WithLifetime(Lifetimes.Singleton)\n .WithoutDisposalTracking());\n')))),(0,i.kt)("h2",{id:"general-options"},"General options"),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"withname"},(0,i.kt)("inlineCode",{parentName:"h3"},"WithName")),(0,i.kt)("p",null,"Sets the name identifier of the registration.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register(config => config\n .WithName("Console"));\n')))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"withinstance"},(0,i.kt)("inlineCode",{parentName:"h3"},"WithInstance")),(0,i.kt)("p",null,"Sets an existing instance for the registration. "),(0,i.kt)("p",null,"Passing true for the ",(0,i.kt)("inlineCode",{parentName:"p"},"wireUp")," parameter means that the container performs member / method injection on the registered instance.")),(0,i.kt)("div",null,(0,i.kt)(a.Z,{mdxType:"Tabs"},(0,i.kt)(s.Z,{value:"Instance",label:"Instance",mdxType:"TabItem"},(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithInstance(new ConsoleLogger()));\n"))),(0,i.kt)(s.Z,{value:"WireUp",label:"WireUp",mdxType:"TabItem"},(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithInstance(new ConsoleLogger(), wireUp: true));\n")))))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"withoutdisposaltracking"},(0,i.kt)("inlineCode",{parentName:"h3"},"WithoutDisposalTracking")),(0,i.kt)("p",null,"Force disables the disposal tracking on the registration.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithoutDisposalTracking());\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"withmetadata"},(0,i.kt)("inlineCode",{parentName:"h3"},"WithMetadata")),(0,i.kt)("p",null,"Sets additional metadata for the registration. It's attached to the service upon its resolution through ",(0,i.kt)("inlineCode",{parentName:"p"},"ValueTuple<,>"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"Tuple<,>"),", or ",(0,i.kt)("inlineCode",{parentName:"p"},"Metadata<,>")," wrappers.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithMetadata(connectionString));\n\nvar jobWithConnectionString = container.Resolve>();\nConsole.WriteLine(jobWithConnectionString.Item2); // prints the connection string.\n\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"withdynamicresolution"},(0,i.kt)("inlineCode",{parentName:"h3"},"WithDynamicResolution")),(0,i.kt)("p",null,"Indicates that the service's resolution should be handled by a dynamic ",(0,i.kt)("inlineCode",{parentName:"p"},"Resolve()")," call on the current ",(0,i.kt)("inlineCode",{parentName:"p"},"IDependencyResolver")," instead of a pre-built instantiation expression.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register();\ncontainer.Register(options => options\n .WithDynamicResolution());\n\n// new DbBackup(currentScope.Resolve());\nvar job = container.Resolve();\n\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"hasservicetype"},(0,i.kt)("inlineCode",{parentName:"h3"},"HasServiceType")),(0,i.kt)("p",null,"Used to build conditions based on service type in batch/assembly registrations.\nIt determines whether the registration is mapped to the given service type.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterAssemblyContaining(configurator: options =>\n {\n if (options.HasServiceType())\n options.WithScopedLifetime();\n });\n\n")))),(0,i.kt)("h2",{id:"initializer--finalizer"},"Initializer / finalizer"),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"withfinalizer"},(0,i.kt)("inlineCode",{parentName:"h3"},"WithFinalizer")),(0,i.kt)("p",null,"Sets a custom cleanup delegate that will be invoked when the scope / container holding the instance is being disposed.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithFinalizer(logger => logger\n .CloseFile()));\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"withinitializer"},(0,i.kt)("inlineCode",{parentName:"h3"},"WithInitializer")),(0,i.kt)("p",null,"Sets a custom initializer delegate that will be invoked when the given service is being instantiated.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithInitializer((logger, resolver) => logger\n .OpenFile()));\n")))),(0,i.kt)("h2",{id:"replace"},"Replace"),(0,i.kt)(a.Z,{mdxType:"Tabs"},(0,i.kt)(s.Z,{value:"ReplaceExisting",label:"ReplaceExisting",mdxType:"TabItem"},(0,i.kt)("p",null,"Indicates whether the container should replace an existing registration with the current one (based on ",(0,i.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"implementation type")," and name). If there's no existing registration in place, the actual one will be added to the registration list."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .ReplaceExisting());\n"))),(0,i.kt)(s.Z,{value:"ReplaceOnlyIfExists",label:"ReplaceOnlyIfExists",mdxType:"TabItem"},(0,i.kt)("p",null,"The same as ",(0,i.kt)("inlineCode",{parentName:"p"},"ReplaceExisting()")," except that the container will do the replace only when there's an already ",(0,i.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-registration--registered-service"},"registered service")," with the same type or name."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .ReplaceOnlyIfExists());\n")))),(0,i.kt)("h2",{id:"multiple-services"},"Multiple services"),(0,i.kt)("p",null,"You can read more about binding a registration to multiple services ",(0,i.kt)("a",{parentName:"p",href:"/docs/guides/advanced-registration#binding-to-multiple-services"},"here"),"."),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"asimplementedtypes"},(0,i.kt)("inlineCode",{parentName:"h3"},"AsImplementedTypes")),(0,i.kt)("p",null,"The service will be mapped to all of its implemented interfaces and base types.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .AsImplementedTypes());\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"asservicealso"},(0,i.kt)("inlineCode",{parentName:"h3"},"AsServiceAlso")),(0,i.kt)("p",null,"Binds the currently configured registration to an additional ",(0,i.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type"),". The registered type must implement or extend the additional ",(0,i.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type"),".")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .AsServiceAlso()\n // or\n .AsServiceAlso(typeof(IRepository)));\n")))),(0,i.kt)("h2",{id:"dependency-configuration"},"Dependency configuration"),(0,i.kt)("p",null,"These options allows the same configuration functionality as the ",(0,i.kt)("a",{parentName:"p",href:"/docs/guides/service-resolution#attributes"},"dependency attribute"),". "),(0,i.kt)(a.Z,{mdxType:"Tabs"},(0,i.kt)(s.Z,{value:"By parameter type",label:"By parameter type",mdxType:"TabItem"},(0,i.kt)("p",null,"Binds a constructor / method parameter or a property / field to a named registration by the parameter's type. The container will perform a ",(0,i.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#named-resolution"},"named resolution")," on the bound dependency. The second parameter used to set the name of the dependency."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register(options => options\n .WithDependencyBinding(typeof(ILogger), "FileLogger"));\n'))),(0,i.kt)(s.Z,{value:"By parameter name",label:"By parameter name",mdxType:"TabItem"},(0,i.kt)("p",null,"Binds a constructor / method parameter or a property / field to a named registration by the parameter's name. The container will perform a ",(0,i.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#named-resolution"},"named resolution")," on the bound dependency. The second parameter used to set the name of the dependency."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register(options => options\n .WithDependencyBinding("logger", "FileLogger"));\n'))),(0,i.kt)(s.Z,{value:"By expression",label:"By expression",mdxType:"TabItem"},(0,i.kt)("p",null,"Marks a member (property / field) as a dependency that should be filled by the container. The second parameter used to set the name of the dependency."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register(options => options\n .WithDependencyBinding(logger => logger.Logger, "ConsoleLogger"));\n')))),(0,i.kt)("h2",{id:"lifetime"},"Lifetime"),(0,i.kt)("p",null,"You can read more about lifetimes ",(0,i.kt)("a",{parentName:"p",href:"/docs/guides/lifetimes"},"here"),"."),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"withsingletonlifetime"},(0,i.kt)("inlineCode",{parentName:"h3"},"WithSingletonLifetime")),(0,i.kt)("p",null,"Sets a singleton lifetime for the registration.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(config => config\n .WithSingletonLifetime());\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"withscopedlifetime"},(0,i.kt)("inlineCode",{parentName:"h3"},"WithScopedLifetime")),(0,i.kt)("p",null,"Sets a scoped lifetime for the registration.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(config => config\n .WithScopedLifetime());\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"withperscopedrequestlifetime"},(0,i.kt)("inlineCode",{parentName:"h3"},"WithPerScopedRequestLifetime")),(0,i.kt)("p",null,"Sets the lifetime to ",(0,i.kt)("inlineCode",{parentName:"p"},"PerScopedRequestLifetime"),". That means this registration will behave like a singleton within every scoped resolution request.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithPerScopedRequestLifetime());\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"withlifetime"},(0,i.kt)("inlineCode",{parentName:"h3"},"WithLifetime")),(0,i.kt)("p",null,"Sets a custom lifetime for the registration.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(config => config\n .WithLifetime(new CustomLifetime()));\n")))),(0,i.kt)("h2",{id:"conditions"},"Conditions"),(0,i.kt)("p",null,"You can read more about the concept of conditional resolution ",(0,i.kt)("a",{parentName:"p",href:"/docs/guides/service-resolution#conditional-resolution"},"here"),"."),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"whenhas"},(0,i.kt)("inlineCode",{parentName:"h3"},"WhenHas")),(0,i.kt)("p",null,"Sets an attribute condition for the registration.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(config => config\n .WhenHas());\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"whenresolutionpathhas"},(0,i.kt)("inlineCode",{parentName:"h3"},"WhenResolutionPathHas")),(0,i.kt)("p",null,"Sets a resolution path condition for the registration. The service will be selected only in the resolution path of the target that has the given attribute.\nThis means that only the direct and sub-dependencies of the target type that has the given attribute will get the configured service.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(config => config\n // Each direct and sub-dependency of any service that has\n // a ConsoleAttribute will get FileLogger wherever they \n // depend on ILogger. \n .WhenResolutionPathHas());\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"whendependantis"},(0,i.kt)("inlineCode",{parentName:"h3"},"WhenDependantIs")),(0,i.kt)("p",null,"Sets a parent target condition for the registration.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(config => config\n .WhenDependantIs());\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"wheninresolutionpathof"},(0,i.kt)("inlineCode",{parentName:"h3"},"WhenInResolutionPathOf")),(0,i.kt)("p",null,"Sets a resolution path condition for the registration. The service will be selected only in the resolution path of the given target.\nThis means that only the direct and sub-dependencies of the target type will get the configured service.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(config => config\n // Each direct and sub-dependency of UserRepository\n // will get FileLogger wherever they depend on ILogger. \n .WhenInResolutionPathOf());\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"when"},(0,i.kt)("inlineCode",{parentName:"h3"},"When")),(0,i.kt)("p",null,"Sets a custom user-defined condition for the registration.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(config => config\n .When(typeInfo => typeInfo.ParentType == typeof(UserRepository)));\n")))),(0,i.kt)("h2",{id:"constructor-selection"},"Constructor selection"),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"withconstructorselectionrule"},(0,i.kt)("inlineCode",{parentName:"h3"},"WithConstructorSelectionRule")),(0,i.kt)("p",null,"Sets the constructor selection rule for the registration.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithConstructorSelectionRule(...));\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h4",{id:"prefermostparameters"},"PreferMostParameters"),(0,i.kt)("p",null,"Selects the constructor which has the longest parameter list.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"options.WithConstructorSelectionRule(\n Rules.ConstructorSelection.PreferMostParameters)\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h4",{id:"preferleastparameters"},"PreferLeastParameters"),(0,i.kt)("p",null,"Selects the constructor which has the shortest parameter list.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"options.WithConstructorSelectionRule(\n Rules.ConstructorSelection.PreferLeastParameters)\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h4",{id:"custom"},"Custom"),(0,i.kt)("p",null,"You can set your own custom constructor ordering logic.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"options.WithConstructorSelectionRule(\n constructors => { /* custom constructor sorting logic */ })\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"withconstructorbyargumenttypes"},(0,i.kt)("inlineCode",{parentName:"h3"},"WithConstructorByArgumentTypes")),(0,i.kt)("p",null,"Selects a constructor by its argument types.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithConstructorByArgumentTypes(typeof(ILogger)));\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"withconstructorbyarguments"},(0,i.kt)("inlineCode",{parentName:"h3"},"WithConstructorByArguments")),(0,i.kt)("p",null,"Selects a constructor by its arguments to use during resolution. These arguments are used to invoke the selected constructor.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithConstructorByArguments(new ConsoleLogger()));\n")))),(0,i.kt)("h2",{id:"property--field-injection"},"Property / field Injection"),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"withautomemberinjection"},(0,i.kt)("inlineCode",{parentName:"h3"},"WithAutoMemberInjection")),(0,i.kt)("p",null,"Enables the auto member injection and sets the rule for it.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithAutoMemberInjection(...));\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h4",{id:"propertieswithpublicsetter"},"PropertiesWithPublicSetter"),(0,i.kt)("p",null,"With this flag, the container will perform auto-injection on properties with a public setter.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"options.WithAutoMemberInjection(\n Rules.AutoMemberInjectionRules.PropertiesWithPublicSetter)\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h4",{id:"propertieswithlimitedaccess"},"PropertiesWithLimitedAccess"),(0,i.kt)("p",null,"With this flag, the container will perform auto-injection on properties which has a non-public setter as well.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"options.WithAutoMemberInjection(\n Rules.AutoMemberInjectionRules.PropertiesWithLimitedAccess)\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h4",{id:"privatefields"},"PrivateFields"),(0,i.kt)("p",null,"With this flag, the container will perform auto-injection on private fields too.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"options.WithAutoMemberInjection(\n Rules.AutoMemberInjectionRules.PrivateFields)\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h4",{id:"combined-rules"},"Combined rules"),(0,i.kt)("p",null,"As these rules are ",(0,i.kt)("a",{parentName:"p",href:"https://docs.microsoft.com/en-us/dotnet/csharp/language-reference/builtin-types/enum#enumeration-types-as-bit-flags"},"bit flags"),", you can use them combined together with bitwise logical operators.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"options.WithAutoMemberInjection(Rules.AutoMemberInjectionRules.PrivateFields | \n Rules.AutoMemberInjectionRules.PropertiesWithPublicSetter)\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h4",{id:"member-selection-filter"},"Member selection filter"),(0,i.kt)("p",null,"You can pass your own member selection logic to control which members should be auto injected.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithAutoMemberInjection(filter: member => member.Type != typeof(ILogger)));\n")))),(0,i.kt)("h2",{id:"injection-parameters"},"Injection parameters"),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"withinjectionparameters"},(0,i.kt)("inlineCode",{parentName:"h3"},"WithInjectionParameters")),(0,i.kt)("p",null,"Sets multiple injection parameters for the registration.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register(options => options\n .WithInjectionParameters(new KeyValuePair("logger", new ConsoleLogger()));\n')))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"withinjectionparameter"},(0,i.kt)("inlineCode",{parentName:"h3"},"WithInjectionParameter")),(0,i.kt)("p",null,"Sets a single injection parameter for the registration.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register(options => options\n .WithInjectionParameter("logger", new ConsoleLogger());\n')))),(0,i.kt)("h2",{id:"factory"},"Factory"),(0,i.kt)("p",null,"You can read more about the concept of factory registration ",(0,i.kt)("a",{parentName:"p",href:"/docs/guides/advanced-registration?id=factory-registration"},"here"),"."),(0,i.kt)(a.Z,{mdxType:"Tabs"},(0,i.kt)(s.Z,{value:"Parameterized",label:"Parameterized",mdxType:"TabItem"},(0,i.kt)("p",null,(0,i.kt)("strong",{parentName:"p"},"WithFactory")," - Sets a factory delegate that could take various number of pre-resolved dependencies as parameters and returns the service instance."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"// 1 parameter factory\ncontainer.Register(options => options\n .WithFactory(logger => new UserRepository(logger));\n\n// 2 parameters factory\ncontainer.Register(options => options\n .WithFactory((logger, context) => new UserRepository(logger, context));\n\n// 3 parameters factory\ncontainer.Register(options => options\n .WithFactory((logger, context, options) => \n new UserRepository(logger, context, options));\n\n// 4 parameters factory\ncontainer.Register(options => options\n .WithFactory((logger, connection, options, validator) => \n new UserRepository(logger, connection, options, validator));\n\n// 5 parameters factory\ncontainer.Register(options => options\n .WithFactory(\n (logger, connection, options, validator, permissionManager) => \n new UserRepository(logger, connection, options, validator, permissionManager));\n")),(0,i.kt)("p",null,"You can also get the current ",(0,i.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#dependency-resolver"},"dependency resolver")," as a pre-resolved parameter:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithFactory((logger, resolver) => \n new UserRepository(logger, resolver.Resolve())));\n"))),(0,i.kt)(s.Z,{value:"Parameter-less",label:"Parameter-less",mdxType:"TabItem"},(0,i.kt)("p",null,(0,i.kt)("strong",{parentName:"p"},"WithFactory")," - Sets a parameter-less factory delegate that returns the service instance."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithFactory(()) => new UserRepository(new ConsoleLogger()));\n"))),(0,i.kt)(s.Z,{value:"Resolver parameter",label:"Resolver parameter",mdxType:"TabItem"},(0,i.kt)("p",null,(0,i.kt)("strong",{parentName:"p"},"WithFactory")," - Sets a factory delegate that takes an ",(0,i.kt)("inlineCode",{parentName:"p"},"IDependencyResolver")," as parameter and returns the service instance."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithFactory(resolver => new UserRepository(resolver.Resolve()));\n")))),(0,i.kt)("admonition",{type:"info"},(0,i.kt)("p",{parentName:"admonition"},"All factory configuration method has an ",(0,i.kt)("inlineCode",{parentName:"p"},"isCompiledLambda")," parameter which should be set to ",(0,i.kt)("inlineCode",{parentName:"p"},"true")," if the passed delegate is compiled from an ",(0,i.kt)("inlineCode",{parentName:"p"},"Expression")," tree.")),(0,i.kt)("h2",{id:"scope-definition"},"Scope definition"),(0,i.kt)("p",null,"You can read more about the concept of defined scopes ",(0,i.kt)("a",{parentName:"p",href:"/docs/guides/scopes?id=service-as-scope"},"here"),"."),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"innamedscope"},(0,i.kt)("inlineCode",{parentName:"h3"},"InNamedScope")),(0,i.kt)("p",null,"Sets a scope name condition for the registration; it will be used only when a scope with the same name requests it.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register(options => options\n .InNamedScope("UserRepo"));\n')))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"inscopedefinedby"},(0,i.kt)("inlineCode",{parentName:"h3"},"InScopeDefinedBy")),(0,i.kt)("p",null,"Sets a condition for the registration; it will be used only within the scope defined by the given type.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .InScopeDefinedBy());\ncontainer.Register(options => options\n .DefinesScope());\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"definesscope"},(0,i.kt)("inlineCode",{parentName:"h3"},"DefinesScope")),(0,i.kt)("p",null,"This registration is used as a logical scope for it's dependencies. Dependencies registered with ",(0,i.kt)("inlineCode",{parentName:"p"},"InNamedScope()")," with the same name are preferred during resolution.\nWhen the ",(0,i.kt)("inlineCode",{parentName:"p"},"name")," is not set, the ",(0,i.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type")," is used as the name. Dependencies registered with ",(0,i.kt)("inlineCode",{parentName:"p"},"InScopeDefinedBy()")," are selected.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register(options => options\n .DefinesScope("UserRepo"));\n\n// or\ncontainer.Register(options => options\n .DefinesScope());\n')))),(0,i.kt)("h2",{id:"decorator-specific"},"Decorator specific"),(0,i.kt)("p",null,"You can read more about decorators ",(0,i.kt)("a",{parentName:"p",href:"/docs/advanced/decorators"},"here"),"."),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"whendecoratedserviceis"},(0,i.kt)("inlineCode",{parentName:"h3"},"WhenDecoratedServiceIs")),(0,i.kt)("p",null,"Sets a decorated target condition for the registration.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterDecorator(options => options\n .WhenDecoratedServiceIs());\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"whendecoratedservicehas"},(0,i.kt)("inlineCode",{parentName:"h3"},"WhenDecoratedServiceHas")),(0,i.kt)("p",null,"Sets an attribute condition that the decorated target has to satisfy.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterDecorator(options => options\n .WhenDecoratedServiceHas());\n")))),(0,i.kt)("h2",{id:"unknown-registration-specific"},"Unknown registration specific"),(0,i.kt)("p",null,"You can read more about unknown type resolution ",(0,i.kt)("a",{parentName:"p",href:"/docs/advanced/special-resolution-cases#unknown-type-resolution"},"here"),"."),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"setimplementationtype"},(0,i.kt)("inlineCode",{parentName:"h3"},"SetImplementationType")),(0,i.kt)("p",null,"Sets the current registration's ",(0,i.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"implementation type"),".")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"var container = new StashboxContainer(c => c.WithUnknownTypeResolution(config =>\n{\n if (config.ServiceType == typeof(IService))\n config.SetImplementationType(typeof(Service));\n}));\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"skip"},(0,i.kt)("inlineCode",{parentName:"h3"},"Skip")),(0,i.kt)("p",null,"Marks the current unknown type registration as skipped.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"var container = new StashboxContainer(c => c.WithUnknownTypeResolution(config =>\n{\n if (config.ServiceType == typeof(IService))\n config.Skip();\n}));\n")))))}g.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/4047545b.4c3bd074.js b/assets/js/4047545b.4c3bd074.js new file mode 100644 index 00000000..68c9d9b1 --- /dev/null +++ b/assets/js/4047545b.4c3bd074.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkwebsite=self.webpackChunkwebsite||[]).push([[371],{3905:(e,t,n)=>{n.d(t,{Zo:()=>p,kt:()=>g});var o=n(7294);function i(e,t,n){return t in e?Object.defineProperty(e,t,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[t]=n,e}function r(e,t){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var o=Object.getOwnPropertySymbols(e);t&&(o=o.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),n.push.apply(n,o)}return n}function a(e){for(var t=1;t=0||(i[n]=e[n]);return i}(e,t);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);for(o=0;o=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(i[n]=e[n])}return i}var l=o.createContext({}),c=function(e){var t=o.useContext(l),n=t;return e&&(n="function"==typeof e?e(t):a(a({},t),e)),n},p=function(e){var t=c(e.components);return o.createElement(l.Provider,{value:t},e.children)},d="mdxType",u={inlineCode:"code",wrapper:function(e){var t=e.children;return o.createElement(o.Fragment,{},t)}},m=o.forwardRef((function(e,t){var n=e.components,i=e.mdxType,r=e.originalType,l=e.parentName,p=s(e,["components","mdxType","originalType","parentName"]),d=c(n),m=i,g=d["".concat(l,".").concat(m)]||d[m]||u[m]||r;return n?o.createElement(g,a(a({ref:t},p),{},{components:n})):o.createElement(g,a({ref:t},p))}));function g(e,t){var n=arguments,i=t&&t.mdxType;if("string"==typeof e||i){var r=n.length,a=new Array(r);a[0]=m;var s={};for(var l in t)hasOwnProperty.call(t,l)&&(s[l]=t[l]);s.originalType=e,s[d]="string"==typeof e?e:i,a[1]=s;for(var c=2;c{n.d(t,{Z:()=>a});var o=n(7294),i=n(6010);const r="tabItem_Ymn6";function a(e){let{children:t,hidden:n,className:a}=e;return o.createElement("div",{role:"tabpanel",className:(0,i.Z)(r,a),hidden:n},t)}},4866:(e,t,n)=>{n.d(t,{Z:()=>I});var o=n(7462),i=n(7294),r=n(6010),a=n(2466),s=n(6550),l=n(1980),c=n(7392),p=n(12);function d(e){return function(e){return i.Children.map(e,(e=>{if(!e||(0,i.isValidElement)(e)&&function(e){const{props:t}=e;return!!t&&"object"==typeof t&&"value"in t}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}(e).map((e=>{let{props:{value:t,label:n,attributes:o,default:i}}=e;return{value:t,label:n,attributes:o,default:i}}))}function u(e){const{values:t,children:n}=e;return(0,i.useMemo)((()=>{const e=t??d(n);return function(e){const t=(0,c.l)(e,((e,t)=>e.value===t.value));if(t.length>0)throw new Error(`Docusaurus error: Duplicate values "${t.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[t,n])}function m(e){let{value:t,tabValues:n}=e;return n.some((e=>e.value===t))}function g(e){let{queryString:t=!1,groupId:n}=e;const o=(0,s.k6)(),r=function(e){let{queryString:t=!1,groupId:n}=e;if("string"==typeof t)return t;if(!1===t)return null;if(!0===t&&!n)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return n??null}({queryString:t,groupId:n});return[(0,l._X)(r),(0,i.useCallback)((e=>{if(!r)return;const t=new URLSearchParams(o.location.search);t.set(r,e),o.replace({...o.location,search:t.toString()})}),[r,o])]}function h(e){const{defaultValue:t,queryString:n=!1,groupId:o}=e,r=u(e),[a,s]=(0,i.useState)((()=>function(e){let{defaultValue:t,tabValues:n}=e;if(0===n.length)throw new Error("Docusaurus error: the component requires at least one children component");if(t){if(!m({value:t,tabValues:n}))throw new Error(`Docusaurus error: The has a defaultValue "${t}" but none of its children has the corresponding value. Available values are: ${n.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return t}const o=n.find((e=>e.default))??n[0];if(!o)throw new Error("Unexpected error: 0 tabValues");return o.value}({defaultValue:t,tabValues:r}))),[l,c]=g({queryString:n,groupId:o}),[d,h]=function(e){let{groupId:t}=e;const n=function(e){return e?`docusaurus.tab.${e}`:null}(t),[o,r]=(0,p.Nk)(n);return[o,(0,i.useCallback)((e=>{n&&r.set(e)}),[n,r])]}({groupId:o}),k=(()=>{const e=l??d;return m({value:e,tabValues:r})?e:null})();(0,i.useLayoutEffect)((()=>{k&&s(k)}),[k]);return{selectedValue:a,selectValue:(0,i.useCallback)((e=>{if(!m({value:e,tabValues:r}))throw new Error(`Can't select invalid tab value=${e}`);s(e),c(e),h(e)}),[c,h,r]),tabValues:r}}var k=n(2389);const v="tabList__CuJ",f="tabItem_LNqP";function y(e){let{className:t,block:n,selectedValue:s,selectValue:l,tabValues:c}=e;const p=[],{blockElementScrollPositionUntilNextRender:d}=(0,a.o5)(),u=e=>{const t=e.currentTarget,n=p.indexOf(t),o=c[n].value;o!==s&&(d(t),l(o))},m=e=>{let t=null;switch(e.key){case"Enter":u(e);break;case"ArrowRight":{const n=p.indexOf(e.currentTarget)+1;t=p[n]??p[0];break}case"ArrowLeft":{const n=p.indexOf(e.currentTarget)-1;t=p[n]??p[p.length-1];break}}t?.focus()};return i.createElement("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,r.Z)("tabs",{"tabs--block":n},t)},c.map((e=>{let{value:t,label:n,attributes:a}=e;return i.createElement("li",(0,o.Z)({role:"tab",tabIndex:s===t?0:-1,"aria-selected":s===t,key:t,ref:e=>p.push(e),onKeyDown:m,onClick:u},a,{className:(0,r.Z)("tabs__item",f,a?.className,{"tabs__item--active":s===t})}),n??t)})))}function b(e){let{lazy:t,children:n,selectedValue:o}=e;const r=(Array.isArray(n)?n:[n]).filter(Boolean);if(t){const e=r.find((e=>e.props.value===o));return e?(0,i.cloneElement)(e,{className:"margin-top--md"}):null}return i.createElement("div",{className:"margin-top--md"},r.map(((e,t)=>(0,i.cloneElement)(e,{key:t,hidden:e.props.value!==o}))))}function N(e){const t=h(e);return i.createElement("div",{className:(0,r.Z)("tabs-container",v)},i.createElement(y,(0,o.Z)({},e,t)),i.createElement(b,(0,o.Z)({},e,t)))}function I(e){const t=(0,k.Z)();return i.createElement(N,(0,o.Z)({key:String(t)},e))}},8846:(e,t,n)=>{n.d(t,{Z:()=>s});var o=n(7294);const i="codeDescContainer_ie8f",r="desc_jyqI",a="example_eYlF";function s(e){let{children:t}=e,n=o.Children.toArray(t).filter((e=>e));return o.createElement("div",{className:i},o.createElement("div",{className:r},n[0]),o.createElement("div",{className:a},n[1]))}},3419:(e,t,n)=>{n.r(t),n.d(t,{assets:()=>d,contentTitle:()=>c,default:()=>g,frontMatter:()=>l,metadata:()=>p,toc:()=>u});var o=n(7462),i=(n(7294),n(3905)),r=n(8846),a=n(4866),s=n(5162);const l={},c="Registration configuration",p={unversionedId:"configuration/registration-configuration",id:"configuration/registration-configuration",title:"Registration configuration",description:"Most of the registration methods have an Action parameter, enabling several customization options on the given registration.",source:"@site/docs/configuration/registration-configuration.md",sourceDirName:"configuration",slug:"/configuration/registration-configuration",permalink:"/stashbox/docs/configuration/registration-configuration",draft:!1,editUrl:"https://github.com/z4kn4fein/stashbox/edit/master/docs/docs/configuration/registration-configuration.md",tags:[],version:"current",lastUpdatedBy:"dependabot[bot]",lastUpdatedAt:1699538162,formattedLastUpdatedAt:"Nov 9, 2023",frontMatter:{},sidebar:"docs",previous:{title:"Scopes",permalink:"/stashbox/docs/guides/scopes"},next:{title:"Container configuration",permalink:"/stashbox/docs/configuration/container-configuration"}},d={},u=[{value:"General options",id:"general-options",level:2},{value:"WithName",id:"withname",level:3},{value:"WithInstance",id:"withinstance",level:3},{value:"WithoutDisposalTracking",id:"withoutdisposaltracking",level:3},{value:"WithMetadata",id:"withmetadata",level:3},{value:"WithDynamicResolution",id:"withdynamicresolution",level:3},{value:"HasServiceType",id:"hasservicetype",level:3},{value:"Initializer / finalizer",id:"initializer--finalizer",level:2},{value:"WithFinalizer",id:"withfinalizer",level:3},{value:"WithInitializer",id:"withinitializer",level:3},{value:"Replace",id:"replace",level:2},{value:"Multiple services",id:"multiple-services",level:2},{value:"AsImplementedTypes",id:"asimplementedtypes",level:3},{value:"AsServiceAlso",id:"asservicealso",level:3},{value:"Dependency configuration",id:"dependency-configuration",level:2},{value:"Lifetime",id:"lifetime",level:2},{value:"WithSingletonLifetime",id:"withsingletonlifetime",level:3},{value:"WithScopedLifetime",id:"withscopedlifetime",level:3},{value:"WithPerScopedRequestLifetime",id:"withperscopedrequestlifetime",level:3},{value:"WithLifetime",id:"withlifetime",level:3},{value:"Conditions",id:"conditions",level:2},{value:"WhenHas",id:"whenhas",level:3},{value:"WhenResolutionPathHas",id:"whenresolutionpathhas",level:3},{value:"WhenDependantIs",id:"whendependantis",level:3},{value:"WhenInResolutionPathOf",id:"wheninresolutionpathof",level:3},{value:"When",id:"when",level:3},{value:"Constructor selection",id:"constructor-selection",level:2},{value:"WithConstructorSelectionRule",id:"withconstructorselectionrule",level:3},{value:"PreferMostParameters",id:"prefermostparameters",level:4},{value:"PreferLeastParameters",id:"preferleastparameters",level:4},{value:"Custom",id:"custom",level:4},{value:"WithConstructorByArgumentTypes",id:"withconstructorbyargumenttypes",level:3},{value:"WithConstructorByArguments",id:"withconstructorbyarguments",level:3},{value:"Property / field Injection",id:"property--field-injection",level:2},{value:"WithAutoMemberInjection",id:"withautomemberinjection",level:3},{value:"PropertiesWithPublicSetter",id:"propertieswithpublicsetter",level:4},{value:"PropertiesWithLimitedAccess",id:"propertieswithlimitedaccess",level:4},{value:"PrivateFields",id:"privatefields",level:4},{value:"Combined rules",id:"combined-rules",level:4},{value:"Member selection filter",id:"member-selection-filter",level:4},{value:"Injection parameters",id:"injection-parameters",level:2},{value:"WithInjectionParameters",id:"withinjectionparameters",level:3},{value:"WithInjectionParameter",id:"withinjectionparameter",level:3},{value:"Factory",id:"factory",level:2},{value:"Scope definition",id:"scope-definition",level:2},{value:"InNamedScope",id:"innamedscope",level:3},{value:"InScopeDefinedBy",id:"inscopedefinedby",level:3},{value:"DefinesScope",id:"definesscope",level:3},{value:"Decorator specific",id:"decorator-specific",level:2},{value:"WhenDecoratedServiceIs",id:"whendecoratedserviceis",level:3},{value:"WhenDecoratedServiceHas",id:"whendecoratedservicehas",level:3},{value:"Unknown registration specific",id:"unknown-registration-specific",level:2},{value:"SetImplementationType",id:"setimplementationtype",level:3},{value:"Skip",id:"skip",level:3}],m={toc:u};function g(e){let{components:t,...n}=e;return(0,i.kt)("wrapper",(0,o.Z)({},m,n,{components:t,mdxType:"MDXLayout"}),(0,i.kt)("h1",{id:"registration-configuration"},"Registration configuration"),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("p",null,"Most of the registration methods have an ",(0,i.kt)("inlineCode",{parentName:"p"},"Action")," parameter, enabling several customization options on the given registration."),(0,i.kt)("p",null,"Here are three examples that show how the API's usage looks like.\nThey cover the exact functionalities you've read about in the ",(0,i.kt)("a",{parentName:"p",href:"/docs/guides/basics"},"basics")," section but are achieved with the options API.")),(0,i.kt)("div",null,(0,i.kt)(a.Z,{mdxType:"Tabs"},(0,i.kt)(s.Z,{value:"Named",label:"Named",mdxType:"TabItem"},(0,i.kt)("p",null,"This is how you can use the options API to set a registration's name:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register(options => options\n .WithName("DbBackup"));\n'))),(0,i.kt)(s.Z,{value:"Lifetime",label:"Lifetime",mdxType:"TabItem"},(0,i.kt)("p",null,"It was mentioned in the ",(0,i.kt)("a",{parentName:"p",href:"/docs/guides/basics#lifetime-shortcuts"},"Lifetime shortcuts")," section, that those methods are only sugars; under the curtain, they are also using this API:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithLifetime(Lifetimes.Singleton));\n"))),(0,i.kt)(s.Z,{value:"Instance",label:"Instance",mdxType:"TabItem"},(0,i.kt)("p",null,"An example of how you can register an instance with the options API:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithInstance(new DbBackup()));\n")))))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("p",null,"The registration configuration API is fluent, which means all option methods can be chained after each other.\nThis provides an easier way to configure complicated registrations.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register(options => options\n .WithName("DbBackup")\n .WithLifetime(Lifetimes.Singleton)\n .WithoutDisposalTracking());\n')))),(0,i.kt)("h2",{id:"general-options"},"General options"),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"withname"},(0,i.kt)("inlineCode",{parentName:"h3"},"WithName")),(0,i.kt)("p",null,"Sets the name identifier of the registration.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register(config => config\n .WithName("Console"));\n')))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"withinstance"},(0,i.kt)("inlineCode",{parentName:"h3"},"WithInstance")),(0,i.kt)("p",null,"Sets an existing instance for the registration. "),(0,i.kt)("p",null,"Passing true for the ",(0,i.kt)("inlineCode",{parentName:"p"},"wireUp")," parameter means that the container performs member / method injection on the registered instance.")),(0,i.kt)("div",null,(0,i.kt)(a.Z,{mdxType:"Tabs"},(0,i.kt)(s.Z,{value:"Instance",label:"Instance",mdxType:"TabItem"},(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithInstance(new ConsoleLogger()));\n"))),(0,i.kt)(s.Z,{value:"WireUp",label:"WireUp",mdxType:"TabItem"},(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithInstance(new ConsoleLogger(), wireUp: true));\n")))))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"withoutdisposaltracking"},(0,i.kt)("inlineCode",{parentName:"h3"},"WithoutDisposalTracking")),(0,i.kt)("p",null,"Force disables the disposal tracking on the registration.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithoutDisposalTracking());\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"withmetadata"},(0,i.kt)("inlineCode",{parentName:"h3"},"WithMetadata")),(0,i.kt)("p",null,"Sets additional metadata for the registration. It's attached to the service upon its resolution through ",(0,i.kt)("inlineCode",{parentName:"p"},"ValueTuple<,>"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"Tuple<,>"),", or ",(0,i.kt)("inlineCode",{parentName:"p"},"Metadata<,>")," wrappers.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithMetadata(connectionString));\n\nvar jobWithConnectionString = container.Resolve>();\nConsole.WriteLine(jobWithConnectionString.Item2); // prints the connection string.\n\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"withdynamicresolution"},(0,i.kt)("inlineCode",{parentName:"h3"},"WithDynamicResolution")),(0,i.kt)("p",null,"Indicates that the service's resolution should be handled by a dynamic ",(0,i.kt)("inlineCode",{parentName:"p"},"Resolve()")," call on the current ",(0,i.kt)("inlineCode",{parentName:"p"},"IDependencyResolver")," instead of a pre-built instantiation expression.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register();\ncontainer.Register(options => options\n .WithDynamicResolution());\n\n// new DbBackup(currentScope.Resolve());\nvar job = container.Resolve();\n\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"hasservicetype"},(0,i.kt)("inlineCode",{parentName:"h3"},"HasServiceType")),(0,i.kt)("p",null,"Used to build conditions based on service type in batch/assembly registrations.\nIt determines whether the registration is mapped to the given service type.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterAssemblyContaining(configurator: options =>\n {\n if (options.HasServiceType())\n options.WithScopedLifetime();\n });\n\n")))),(0,i.kt)("h2",{id:"initializer--finalizer"},"Initializer / finalizer"),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"withfinalizer"},(0,i.kt)("inlineCode",{parentName:"h3"},"WithFinalizer")),(0,i.kt)("p",null,"Sets a custom cleanup delegate that will be invoked when the scope / container holding the instance is being disposed.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithFinalizer(logger => logger\n .CloseFile()));\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"withinitializer"},(0,i.kt)("inlineCode",{parentName:"h3"},"WithInitializer")),(0,i.kt)("p",null,"Sets a custom initializer delegate that will be invoked when the given service is being instantiated.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithInitializer((logger, resolver) => logger\n .OpenFile()));\n")))),(0,i.kt)("h2",{id:"replace"},"Replace"),(0,i.kt)(a.Z,{mdxType:"Tabs"},(0,i.kt)(s.Z,{value:"ReplaceExisting",label:"ReplaceExisting",mdxType:"TabItem"},(0,i.kt)("p",null,"Indicates whether the container should replace an existing registration with the current one (based on ",(0,i.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"implementation type")," and name). If there's no existing registration in place, the actual one will be added to the registration list."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .ReplaceExisting());\n"))),(0,i.kt)(s.Z,{value:"ReplaceOnlyIfExists",label:"ReplaceOnlyIfExists",mdxType:"TabItem"},(0,i.kt)("p",null,"The same as ",(0,i.kt)("inlineCode",{parentName:"p"},"ReplaceExisting()")," except that the container will do the replace only when there's an already ",(0,i.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-registration--registered-service"},"registered service")," with the same type or name."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .ReplaceOnlyIfExists());\n")))),(0,i.kt)("h2",{id:"multiple-services"},"Multiple services"),(0,i.kt)("p",null,"You can read more about binding a registration to multiple services ",(0,i.kt)("a",{parentName:"p",href:"/docs/guides/advanced-registration#binding-to-multiple-services"},"here"),"."),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"asimplementedtypes"},(0,i.kt)("inlineCode",{parentName:"h3"},"AsImplementedTypes")),(0,i.kt)("p",null,"The service will be mapped to all of its implemented interfaces and base types.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .AsImplementedTypes());\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"asservicealso"},(0,i.kt)("inlineCode",{parentName:"h3"},"AsServiceAlso")),(0,i.kt)("p",null,"Binds the currently configured registration to an additional ",(0,i.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type"),". The registered type must implement or extend the additional ",(0,i.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type"),".")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .AsServiceAlso()\n // or\n .AsServiceAlso(typeof(IRepository)));\n")))),(0,i.kt)("h2",{id:"dependency-configuration"},"Dependency configuration"),(0,i.kt)("p",null,"These options allows the same configuration functionality as the ",(0,i.kt)("a",{parentName:"p",href:"/docs/guides/service-resolution#attributes"},"dependency attribute"),". "),(0,i.kt)(a.Z,{mdxType:"Tabs"},(0,i.kt)(s.Z,{value:"By parameter type",label:"By parameter type",mdxType:"TabItem"},(0,i.kt)("p",null,"Binds a constructor / method parameter or a property / field to a named registration by the parameter's type. The container will perform a ",(0,i.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#named-resolution"},"named resolution")," on the bound dependency. The second parameter used to set the name of the dependency."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register(options => options\n .WithDependencyBinding(typeof(ILogger), "FileLogger"));\n'))),(0,i.kt)(s.Z,{value:"By parameter name",label:"By parameter name",mdxType:"TabItem"},(0,i.kt)("p",null,"Binds a constructor / method parameter or a property / field to a named registration by the parameter's name. The container will perform a ",(0,i.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#named-resolution"},"named resolution")," on the bound dependency. The second parameter used to set the name of the dependency."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register(options => options\n .WithDependencyBinding("logger", "FileLogger"));\n'))),(0,i.kt)(s.Z,{value:"By expression",label:"By expression",mdxType:"TabItem"},(0,i.kt)("p",null,"Marks a member (property / field) as a dependency that should be filled by the container. The second parameter used to set the name of the dependency."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register(options => options\n .WithDependencyBinding(logger => logger.Logger, "ConsoleLogger"));\n')))),(0,i.kt)("h2",{id:"lifetime"},"Lifetime"),(0,i.kt)("p",null,"You can read more about lifetimes ",(0,i.kt)("a",{parentName:"p",href:"/docs/guides/lifetimes"},"here"),"."),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"withsingletonlifetime"},(0,i.kt)("inlineCode",{parentName:"h3"},"WithSingletonLifetime")),(0,i.kt)("p",null,"Sets a singleton lifetime for the registration.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(config => config\n .WithSingletonLifetime());\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"withscopedlifetime"},(0,i.kt)("inlineCode",{parentName:"h3"},"WithScopedLifetime")),(0,i.kt)("p",null,"Sets a scoped lifetime for the registration.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(config => config\n .WithScopedLifetime());\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"withperscopedrequestlifetime"},(0,i.kt)("inlineCode",{parentName:"h3"},"WithPerScopedRequestLifetime")),(0,i.kt)("p",null,"Sets the lifetime to ",(0,i.kt)("inlineCode",{parentName:"p"},"PerScopedRequestLifetime"),". That means this registration will behave like a singleton within every scoped resolution request.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithPerScopedRequestLifetime());\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"withlifetime"},(0,i.kt)("inlineCode",{parentName:"h3"},"WithLifetime")),(0,i.kt)("p",null,"Sets a custom lifetime for the registration.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(config => config\n .WithLifetime(new CustomLifetime()));\n")))),(0,i.kt)("h2",{id:"conditions"},"Conditions"),(0,i.kt)("p",null,"You can read more about the concept of conditional resolution ",(0,i.kt)("a",{parentName:"p",href:"/docs/guides/service-resolution#conditional-resolution"},"here"),"."),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"whenhas"},(0,i.kt)("inlineCode",{parentName:"h3"},"WhenHas")),(0,i.kt)("p",null,"Sets an attribute condition for the registration.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(config => config\n .WhenHas());\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"whenresolutionpathhas"},(0,i.kt)("inlineCode",{parentName:"h3"},"WhenResolutionPathHas")),(0,i.kt)("p",null,"Sets a resolution path condition for the registration. The service will be selected only in the resolution path of the target that has the given attribute.\nThis means that only the direct and sub-dependencies of the target type that has the given attribute will get the configured service.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(config => config\n // Each direct and sub-dependency of any service that has\n // a ConsoleAttribute will get FileLogger wherever they \n // depend on ILogger. \n .WhenResolutionPathHas());\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"whendependantis"},(0,i.kt)("inlineCode",{parentName:"h3"},"WhenDependantIs")),(0,i.kt)("p",null,"Sets a parent target condition for the registration.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(config => config\n .WhenDependantIs());\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"wheninresolutionpathof"},(0,i.kt)("inlineCode",{parentName:"h3"},"WhenInResolutionPathOf")),(0,i.kt)("p",null,"Sets a resolution path condition for the registration. The service will be selected only in the resolution path of the given target.\nThis means that only the direct and sub-dependencies of the target type will get the configured service.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(config => config\n // Each direct and sub-dependency of UserRepository\n // will get FileLogger wherever they depend on ILogger. \n .WhenInResolutionPathOf());\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"when"},(0,i.kt)("inlineCode",{parentName:"h3"},"When")),(0,i.kt)("p",null,"Sets a custom user-defined condition for the registration.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(config => config\n .When(typeInfo => typeInfo.ParentType == typeof(UserRepository)));\n")))),(0,i.kt)("h2",{id:"constructor-selection"},"Constructor selection"),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"withconstructorselectionrule"},(0,i.kt)("inlineCode",{parentName:"h3"},"WithConstructorSelectionRule")),(0,i.kt)("p",null,"Sets the constructor selection rule for the registration.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithConstructorSelectionRule(...));\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h4",{id:"prefermostparameters"},"PreferMostParameters"),(0,i.kt)("p",null,"Selects the constructor which has the longest parameter list.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"options.WithConstructorSelectionRule(\n Rules.ConstructorSelection.PreferMostParameters)\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h4",{id:"preferleastparameters"},"PreferLeastParameters"),(0,i.kt)("p",null,"Selects the constructor which has the shortest parameter list.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"options.WithConstructorSelectionRule(\n Rules.ConstructorSelection.PreferLeastParameters)\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h4",{id:"custom"},"Custom"),(0,i.kt)("p",null,"You can set your own custom constructor ordering logic.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"options.WithConstructorSelectionRule(\n constructors => { /* custom constructor sorting logic */ })\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"withconstructorbyargumenttypes"},(0,i.kt)("inlineCode",{parentName:"h3"},"WithConstructorByArgumentTypes")),(0,i.kt)("p",null,"Selects a constructor by its argument types.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithConstructorByArgumentTypes(typeof(ILogger)));\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"withconstructorbyarguments"},(0,i.kt)("inlineCode",{parentName:"h3"},"WithConstructorByArguments")),(0,i.kt)("p",null,"Selects a constructor by its arguments to use during resolution. These arguments are used to invoke the selected constructor.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithConstructorByArguments(new ConsoleLogger()));\n")))),(0,i.kt)("h2",{id:"property--field-injection"},"Property / field Injection"),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"withautomemberinjection"},(0,i.kt)("inlineCode",{parentName:"h3"},"WithAutoMemberInjection")),(0,i.kt)("p",null,"Enables the auto member injection and sets the rule for it.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithAutoMemberInjection(...));\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h4",{id:"propertieswithpublicsetter"},"PropertiesWithPublicSetter"),(0,i.kt)("p",null,"With this flag, the container will perform auto-injection on properties with a public setter.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"options.WithAutoMemberInjection(\n Rules.AutoMemberInjectionRules.PropertiesWithPublicSetter)\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h4",{id:"propertieswithlimitedaccess"},"PropertiesWithLimitedAccess"),(0,i.kt)("p",null,"With this flag, the container will perform auto-injection on properties which has a non-public setter as well.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"options.WithAutoMemberInjection(\n Rules.AutoMemberInjectionRules.PropertiesWithLimitedAccess)\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h4",{id:"privatefields"},"PrivateFields"),(0,i.kt)("p",null,"With this flag, the container will perform auto-injection on private fields too.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"options.WithAutoMemberInjection(\n Rules.AutoMemberInjectionRules.PrivateFields)\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h4",{id:"combined-rules"},"Combined rules"),(0,i.kt)("p",null,"As these rules are ",(0,i.kt)("a",{parentName:"p",href:"https://docs.microsoft.com/en-us/dotnet/csharp/language-reference/builtin-types/enum#enumeration-types-as-bit-flags"},"bit flags"),", you can use them combined together with bitwise logical operators.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"options.WithAutoMemberInjection(Rules.AutoMemberInjectionRules.PrivateFields | \n Rules.AutoMemberInjectionRules.PropertiesWithPublicSetter)\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h4",{id:"member-selection-filter"},"Member selection filter"),(0,i.kt)("p",null,"You can pass your own member selection logic to control which members should be auto injected.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithAutoMemberInjection(filter: member => member.Type != typeof(ILogger)));\n")))),(0,i.kt)("h2",{id:"injection-parameters"},"Injection parameters"),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"withinjectionparameters"},(0,i.kt)("inlineCode",{parentName:"h3"},"WithInjectionParameters")),(0,i.kt)("p",null,"Sets multiple injection parameters for the registration.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register(options => options\n .WithInjectionParameters(new KeyValuePair("logger", new ConsoleLogger()));\n')))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"withinjectionparameter"},(0,i.kt)("inlineCode",{parentName:"h3"},"WithInjectionParameter")),(0,i.kt)("p",null,"Sets a single injection parameter for the registration.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register(options => options\n .WithInjectionParameter("logger", new ConsoleLogger());\n')))),(0,i.kt)("h2",{id:"factory"},"Factory"),(0,i.kt)("p",null,"You can read more about the concept of factory registration ",(0,i.kt)("a",{parentName:"p",href:"/docs/guides/advanced-registration?id=factory-registration"},"here"),"."),(0,i.kt)(a.Z,{mdxType:"Tabs"},(0,i.kt)(s.Z,{value:"Parameterized",label:"Parameterized",mdxType:"TabItem"},(0,i.kt)("p",null,(0,i.kt)("strong",{parentName:"p"},"WithFactory")," - Sets a factory delegate that could take various number of pre-resolved dependencies as parameters and returns the service instance."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"// 1 parameter factory\ncontainer.Register(options => options\n .WithFactory(logger => new UserRepository(logger));\n\n// 2 parameters factory\ncontainer.Register(options => options\n .WithFactory((logger, context) => new UserRepository(logger, context));\n\n// 3 parameters factory\ncontainer.Register(options => options\n .WithFactory((logger, context, options) => \n new UserRepository(logger, context, options));\n\n// 4 parameters factory\ncontainer.Register(options => options\n .WithFactory((logger, connection, options, validator) => \n new UserRepository(logger, connection, options, validator));\n\n// 5 parameters factory\ncontainer.Register(options => options\n .WithFactory(\n (logger, connection, options, validator, permissionManager) => \n new UserRepository(logger, connection, options, validator, permissionManager));\n")),(0,i.kt)("p",null,"You can also get the current ",(0,i.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#dependency-resolver"},"dependency resolver")," as a pre-resolved parameter:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithFactory((logger, resolver) => \n new UserRepository(logger, resolver.Resolve())));\n"))),(0,i.kt)(s.Z,{value:"Parameter-less",label:"Parameter-less",mdxType:"TabItem"},(0,i.kt)("p",null,(0,i.kt)("strong",{parentName:"p"},"WithFactory")," - Sets a parameter-less factory delegate that returns the service instance."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithFactory(()) => new UserRepository(new ConsoleLogger()));\n"))),(0,i.kt)(s.Z,{value:"Resolver parameter",label:"Resolver parameter",mdxType:"TabItem"},(0,i.kt)("p",null,(0,i.kt)("strong",{parentName:"p"},"WithFactory")," - Sets a factory delegate that takes an ",(0,i.kt)("inlineCode",{parentName:"p"},"IDependencyResolver")," as parameter and returns the service instance."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithFactory(resolver => new UserRepository(resolver.Resolve()));\n")))),(0,i.kt)("admonition",{type:"info"},(0,i.kt)("p",{parentName:"admonition"},"All factory configuration method has an ",(0,i.kt)("inlineCode",{parentName:"p"},"isCompiledLambda")," parameter which should be set to ",(0,i.kt)("inlineCode",{parentName:"p"},"true")," if the passed delegate is compiled from an ",(0,i.kt)("inlineCode",{parentName:"p"},"Expression")," tree.")),(0,i.kt)("h2",{id:"scope-definition"},"Scope definition"),(0,i.kt)("p",null,"You can read more about the concept of defined scopes ",(0,i.kt)("a",{parentName:"p",href:"/docs/guides/scopes?id=service-as-scope"},"here"),"."),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"innamedscope"},(0,i.kt)("inlineCode",{parentName:"h3"},"InNamedScope")),(0,i.kt)("p",null,"Sets a scope name condition for the registration; it will be used only when a scope with the same name requests it.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register(options => options\n .InNamedScope("UserRepo"));\n')))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"inscopedefinedby"},(0,i.kt)("inlineCode",{parentName:"h3"},"InScopeDefinedBy")),(0,i.kt)("p",null,"Sets a condition for the registration; it will be used only within the scope defined by the given type.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .InScopeDefinedBy());\ncontainer.Register(options => options\n .DefinesScope());\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"definesscope"},(0,i.kt)("inlineCode",{parentName:"h3"},"DefinesScope")),(0,i.kt)("p",null,"This registration is used as a logical scope for it's dependencies. Dependencies registered with ",(0,i.kt)("inlineCode",{parentName:"p"},"InNamedScope()")," with the same name are preferred during resolution.\nWhen the ",(0,i.kt)("inlineCode",{parentName:"p"},"name")," is not set, the ",(0,i.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type")," is used as the name. Dependencies registered with ",(0,i.kt)("inlineCode",{parentName:"p"},"InScopeDefinedBy()")," are selected.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register(options => options\n .DefinesScope("UserRepo"));\n\n// or\ncontainer.Register(options => options\n .DefinesScope());\n')))),(0,i.kt)("h2",{id:"decorator-specific"},"Decorator specific"),(0,i.kt)("p",null,"You can read more about decorators ",(0,i.kt)("a",{parentName:"p",href:"/docs/advanced/decorators"},"here"),"."),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"whendecoratedserviceis"},(0,i.kt)("inlineCode",{parentName:"h3"},"WhenDecoratedServiceIs")),(0,i.kt)("p",null,"Sets a decorated target condition for the registration.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterDecorator(options => options\n .WhenDecoratedServiceIs());\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"whendecoratedservicehas"},(0,i.kt)("inlineCode",{parentName:"h3"},"WhenDecoratedServiceHas")),(0,i.kt)("p",null,"Sets an attribute condition that the decorated target has to satisfy.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterDecorator(options => options\n .WhenDecoratedServiceHas());\n")))),(0,i.kt)("h2",{id:"unknown-registration-specific"},"Unknown registration specific"),(0,i.kt)("p",null,"You can read more about unknown type resolution ",(0,i.kt)("a",{parentName:"p",href:"/docs/advanced/special-resolution-cases#unknown-type-resolution"},"here"),"."),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"setimplementationtype"},(0,i.kt)("inlineCode",{parentName:"h3"},"SetImplementationType")),(0,i.kt)("p",null,"Sets the current registration's ",(0,i.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"implementation type"),".")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"var container = new StashboxContainer(c => c.WithUnknownTypeResolution(config =>\n{\n if (config.ServiceType == typeof(IService))\n config.SetImplementationType(typeof(Service));\n}));\n")))),(0,i.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,i.kt)("div",null,(0,i.kt)("h3",{id:"skip"},(0,i.kt)("inlineCode",{parentName:"h3"},"Skip")),(0,i.kt)("p",null,"Marks the current unknown type registration as skipped.")),(0,i.kt)("div",null,(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-cs"},"var container = new StashboxContainer(c => c.WithUnknownTypeResolution(config =>\n{\n if (config.ServiceType == typeof(IService))\n config.Skip();\n}));\n")))))}g.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/5b5f0b93.cfbbcee3.js b/assets/js/5b5f0b93.cfbbcee3.js new file mode 100644 index 00000000..7ced5ef5 --- /dev/null +++ b/assets/js/5b5f0b93.cfbbcee3.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkwebsite=self.webpackChunkwebsite||[]).push([[946],{3905:(e,n,t)=>{t.d(n,{Zo:()=>d,kt:()=>m});var i=t(7294);function a(e,n,t){return n in e?Object.defineProperty(e,n,{value:t,enumerable:!0,configurable:!0,writable:!0}):e[n]=t,e}function r(e,n){var t=Object.keys(e);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(e);n&&(i=i.filter((function(n){return Object.getOwnPropertyDescriptor(e,n).enumerable}))),t.push.apply(t,i)}return t}function o(e){for(var n=1;n=0||(a[t]=e[t]);return a}(e,n);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);for(i=0;i=0||Object.prototype.propertyIsEnumerable.call(e,t)&&(a[t]=e[t])}return a}var s=i.createContext({}),c=function(e){var n=i.useContext(s),t=n;return e&&(t="function"==typeof e?e(n):o(o({},n),e)),t},d=function(e){var n=c(e.components);return i.createElement(s.Provider,{value:n},e.children)},h="mdxType",p={inlineCode:"code",wrapper:function(e){var n=e.children;return i.createElement(i.Fragment,{},n)}},u=i.forwardRef((function(e,n){var t=e.components,a=e.mdxType,r=e.originalType,s=e.parentName,d=l(e,["components","mdxType","originalType","parentName"]),h=c(t),u=a,m=h["".concat(s,".").concat(u)]||h[u]||p[u]||r;return t?i.createElement(m,o(o({ref:n},d),{},{components:t})):i.createElement(m,o({ref:n},d))}));function m(e,n){var t=arguments,a=n&&n.mdxType;if("string"==typeof e||a){var r=t.length,o=new Array(r);o[0]=u;var l={};for(var s in n)hasOwnProperty.call(n,s)&&(l[s]=n[s]);l.originalType=e,l[h]="string"==typeof e?e:a,o[1]=l;for(var c=2;c{t.d(n,{Z:()=>l});var i=t(7294);const a="codeDescContainer_ie8f",r="desc_jyqI",o="example_eYlF";function l(e){let{children:n}=e,t=i.Children.toArray(n).filter((e=>e));return i.createElement("div",{className:a},i.createElement("div",{className:r},t[0]),i.createElement("div",{className:o},t[1]))}},9674:(e,n,t)=>{t.r(n),t.d(n,{assets:()=>c,contentTitle:()=>l,default:()=>p,frontMatter:()=>o,metadata:()=>s,toc:()=>d});var i=t(7462),a=(t(7294),t(3905)),r=t(8846);const o={},l="Child containers",s={unversionedId:"advanced/child-containers",id:"advanced/child-containers",title:"Child containers",description:"With child containers, you can build up parent-child relationships between containers. This means you can have a different subset of services present in a child than in the parent container.",source:"@site/docs/advanced/child-containers.md",sourceDirName:"advanced",slug:"/advanced/child-containers",permalink:"/stashbox/docs/advanced/child-containers",draft:!1,editUrl:"https://github.com/z4kn4fein/stashbox/edit/master/docs/docs/advanced/child-containers.md",tags:[],version:"current",lastUpdatedBy:"dependabot[bot]",lastUpdatedAt:1699538162,formattedLastUpdatedAt:"Nov 9, 2023",frontMatter:{},sidebar:"docs",previous:{title:"Wrappers & resolvers",permalink:"/stashbox/docs/advanced/wrappers-resolvers"},next:{title:"Special resolution cases",permalink:"/stashbox/docs/advanced/special-resolution-cases"}},c={},d=[{value:"Example",id:"example",level:2},{value:"Accessing child containers",id:"accessing-child-containers",level:2},{value:"Resolution behavior",id:"resolution-behavior",level:2},{value:"Re-building singletons",id:"re-building-singletons",level:2},{value:"Nested child containers",id:"nested-child-containers",level:2},{value:"Dispose",id:"dispose",level:2}],h={toc:d};function p(e){let{components:n,...t}=e;return(0,a.kt)("wrapper",(0,i.Z)({},h,t,{components:n,mdxType:"MDXLayout"}),(0,a.kt)("h1",{id:"child-containers"},"Child containers"),(0,a.kt)("p",null,"With child containers, you can build up parent-child relationships between containers. This means you can have a different subset of services present in a child than in the parent container. "),(0,a.kt)("p",null,"When a dependency is missing from the child container during a resolution request, the parent will be asked to resolve the missing service. If it's found there, the parent will return only the service's registration, and the resolution request will jump back to the child. Also, child registrations with the same ",(0,a.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type")," will override the parent's services."),(0,a.kt)("p",null,"Resolving ",(0,a.kt)("inlineCode",{parentName:"p"},"IEnumerable")," and ",(0,a.kt)("a",{parentName:"p",href:"/docs/advanced/decorators"},"decorators")," also considers parent containers by default. However, this behavior can be controlled with the ",(0,a.kt)("a",{parentName:"p",href:"#resolution-behavior"},(0,a.kt)("inlineCode",{parentName:"a"},"ResolutionBehavior"))," parameter. "),(0,a.kt)("admonition",{type:"info"},(0,a.kt)("p",{parentName:"admonition"},"Child containers are the foundation of the ",(0,a.kt)("a",{parentName:"p",href:"https://github.com/z4kn4fein/stashbox-extensions-dependencyinjection#multitenant"},"ASP.NET Core multi-tenant extension"),".")),(0,a.kt)("h2",{id:"example"},"Example"),(0,a.kt)("p",null,"Here is an example case:"),(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"interface IDependency {}\n\nclass B : IDependency {}\nclass C : IDependency {}\n\nclass A \n{\n public A(IDependency dependency)\n { }\n}\n\nusing (var container = new StashboxContainer())\n{\n // register 'A' into the parent container.\n container.Register();\n\n // register 'B' as a dependency into the parent container.\n container.Register();\n\n var child = container.CreateChildContainer()\n \n // register 'C' as a dependency into the child container.\n child.Register();\n \n // 'A' is resolved from the parent and gets\n // 'C' as IDependency because the resolution\n // request was initiated on the child.\n A fromChild = child.Resolve();\n\n // 'A' gets 'B' as IDependency because the \n // resolution request was initiated on the parent.\n A fromParent = container.Resolve();\n} // using will dispose the parent along with the child.\n")),(0,a.kt)("p",null,"Let's see what's happening when we request ",(0,a.kt)("inlineCode",{parentName:"p"},"A")," from the ",(0,a.kt)("em",{parentName:"p"},"child"),":"),(0,a.kt)("ol",null,(0,a.kt)("li",{parentName:"ol"},(0,a.kt)("inlineCode",{parentName:"li"},"A")," not found in the ",(0,a.kt)("em",{parentName:"li"},"child"),", go up to the ",(0,a.kt)("em",{parentName:"li"},"parent")," and check there."),(0,a.kt)("li",{parentName:"ol"},(0,a.kt)("inlineCode",{parentName:"li"},"A")," found in the ",(0,a.kt)("em",{parentName:"li"},"parent"),", resolve."),(0,a.kt)("li",{parentName:"ol"},(0,a.kt)("inlineCode",{parentName:"li"},"A")," depends on ",(0,a.kt)("inlineCode",{parentName:"li"},"IDependency"),", go back to the ",(0,a.kt)("em",{parentName:"li"},"child")," and search ",(0,a.kt)("inlineCode",{parentName:"li"},"IDependency")," implementations."),(0,a.kt)("li",{parentName:"ol"},(0,a.kt)("inlineCode",{parentName:"li"},"C")," found in the ",(0,a.kt)("em",{parentName:"li"},"child"),", it does not have any dependencies, instantiate."),(0,a.kt)("li",{parentName:"ol"},"Inject the new ",(0,a.kt)("inlineCode",{parentName:"li"},"C")," instance into ",(0,a.kt)("inlineCode",{parentName:"li"},"A"),"."),(0,a.kt)("li",{parentName:"ol"},"All dependencies are resolved; return ",(0,a.kt)("inlineCode",{parentName:"li"},"A"),".")),(0,a.kt)("p",null,"When we make the same request on the parent, everything will go as usual because we have all dependencies in place. ",(0,a.kt)("inlineCode",{parentName:"p"},"B")," will be injected into ",(0,a.kt)("inlineCode",{parentName:"p"},"A"),"."),(0,a.kt)("admonition",{type:"info"},(0,a.kt)("p",{parentName:"admonition"},"You can ",(0,a.kt)("a",{parentName:"p",href:"/docs/configuration/container-configuration"},"re-configure")," child containers with the ",(0,a.kt)("inlineCode",{parentName:"p"},".Configure()")," method. It doesn't affect the parent container's configuration.")),(0,a.kt)("h2",{id:"accessing-child-containers"},"Accessing child containers"),(0,a.kt)("p",null,"You can identify child containers with the ",(0,a.kt)("inlineCode",{parentName:"p"},"identifier")," parameter of ",(0,a.kt)("inlineCode",{parentName:"p"},"CreateChildContainer()"),". Later, you can retrieve the given child container by passing its ID to ",(0,a.kt)("inlineCode",{parentName:"p"},"GetChildContainer()"),"."),(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},'using var container = new StashboxContainer();\ncontainer.CreateChildContainer("child");\n// ...\n\nvar child = container.GetChildContainer("child");\n')),(0,a.kt)("p",null,"Also, each child container created by a container is available through the ",(0,a.kt)("inlineCode",{parentName:"p"},"IStashboxContainer.ChildContainers")," propert."),(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},'using var container = new StashboxContainer();\ncontainer.CreateChildContainer("child1");\ncontainer.CreateChildContainer("child2");\n// ...\n\nforeach (var child in container.ChildContainers)\n{\n var id = child.Key;\n var childContainer = child.Value;\n}\n')),(0,a.kt)("h2",{id:"resolution-behavior"},"Resolution behavior"),(0,a.kt)("p",null,"You can control which level of the container hierarchy can participate in the service resolution with the ",(0,a.kt)("inlineCode",{parentName:"p"},"ResolutionBehavior")," parameter. "),(0,a.kt)("p",null,"Possible values:"),(0,a.kt)("ul",null,(0,a.kt)("li",{parentName:"ul"},(0,a.kt)("inlineCode",{parentName:"li"},"Default"),": The default behavior, it's used when the parameter is not specified. Its value is ",(0,a.kt)("inlineCode",{parentName:"li"},"Parent | Current"),", so the parents and the current container (which initiated the resolution request) can participate in the resolution request's service selection."),(0,a.kt)("li",{parentName:"ul"},(0,a.kt)("inlineCode",{parentName:"li"},"Parent"),": Indicates that parent containers (including indirect all ancestors) can participate in the resolution request's service selection."),(0,a.kt)("li",{parentName:"ul"},(0,a.kt)("inlineCode",{parentName:"li"},"Current"),": Indicates that the current container (which initiated the resolution request) can participate in the service selection."),(0,a.kt)("li",{parentName:"ul"},(0,a.kt)("inlineCode",{parentName:"li"},"ParentDependency"),": Indicates that parent containers (including indirect all ancestors) can only provide dependencies for services that are already selected for resolution.")),(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-csharp"},"interface IService {}\n\nclass A : IService {}\nclass B : IService {}\n\nusing (var container = new StashboxContainer())\n{\n // register 'A' into the parent container.\n container.Register();\n\n var child = container.CreateChildContainer()\n \n // register 'B' into the child container.\n child.Register();\n \n // 'A' is resolved because only parent\n // can participate in the resolution request.\n IService withParent = child.Resolve(ResolutionBehavior.Parent);\n\n // Only 'B' is in the collection because\n // only the caller container can take part\n // in the resolution request.\n IEnumerable allWithCurrent = child.Resolve>(ResolutionBehavior.Current);\n \n // Both 'A' and 'B' is in the collection\n // because both the parent and the caller container\n // participates in the resolution request.\n IEnumerable all = child.Resolve>(ResolutionBehavior.Current | ResolutionBehavior.Parent);\n} // using will dispose the parent along with the child.\n")),(0,a.kt)("h2",{id:"re-building-singletons"},"Re-building singletons"),(0,a.kt)("p",null,"By default, singletons are instantiated and stored only in those containers that registered them. However, you can enable the re-instantiation of singletons in child containers with the ",(0,a.kt)("inlineCode",{parentName:"p"},".WithReBuildSingletonsInChildContainer()")," ",(0,a.kt)("a",{parentName:"p",href:"/docs/configuration/container-configuration#re-build-singletons-in-child-containers"},"container configuration option"),". "),(0,a.kt)("p",null,"If it's enabled, all singletons will be re-created in those containers that initiated the resolution request. By this, re-built singletons can use overridden dependencies from child containers. "),(0,a.kt)("p",null,"Re-building in child containers does not affect the singletons instantiated in the parent container."),(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"interface IDependency {}\n\nclass B : IDependency {}\nclass C : IDependency {}\n\nclass A \n{\n public A(IDependency dependency)\n { }\n}\n\nusing (var container = new StashboxContainer(options => options.WithReBuildSingletonsInChildContainer()))\n{\n // register 'A' as a singleton into the parent container.\n container.RegisterSingleton();\n\n // register 'B' as a dependency into the parent container.\n container.Register();\n\n // 'A' gets 'B' as IDependency and will be stored\n // in the parent container as a singleton.\n A fromParent = container.Resolve();\n\n var child = container.CreateChildContainer();\n \n // register 'C' as a dependency into the child container.\n child.Register();\n\n // a new 'A' singleton will be created in\n // the child container with 'C' as IDependency.\n A fromChild = child.Resolve();\n} // using will dispose the parent along with the child.\n")),(0,a.kt)("h2",{id:"nested-child-containers"},"Nested child containers"),(0,a.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"You can build up a hierarchical tree structure from containers by creating more child containers with the ",(0,a.kt)("inlineCode",{parentName:"p"},".CreateChildContainer()")," method.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"using var container = new StashboxContainer();\n\nvar child1 = container.CreateChildContainer();\nvar child2 = child1.CreateChildContainer();\n")))),(0,a.kt)("h2",{id:"dispose"},"Dispose"),(0,a.kt)("p",null,"By default, the parent container's disposal also disposes its child containers. You can control this behavior with the ",(0,a.kt)("inlineCode",{parentName:"p"},"CreateChildContainer()")," method's ",(0,a.kt)("inlineCode",{parentName:"p"},"attachToParent")," boolean parameter."),(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"using (var container = new StashboxContainer())\n{\n using (var child1 = container.CreateChildContainer(attachToParent: false))\n {\n } // child1 will be disposed only once here.\n \n var child2 = container.CreateChildContainer();\n var child3 = container.CreateChildContainer();\n} // using will dispose the parent along with child2 and child3.\n")),(0,a.kt)("p",null,"You can safely dispose a child even if it's attached to its parent, in this case the parent's disposal will not dispose the already disposed child."),(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"using (var container = new StashboxContainer())\n{\n using (var child1 = container.CreateChildContainer())\n {\n } // child1 will be disposed only once here.\n \n var child2 = container.CreateChildContainer();\n} // using will dispose only the parent and child2.\n")))}p.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/5b5f0b93.f8a5ec34.js b/assets/js/5b5f0b93.f8a5ec34.js deleted file mode 100644 index 5513c578..00000000 --- a/assets/js/5b5f0b93.f8a5ec34.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkwebsite=self.webpackChunkwebsite||[]).push([[946],{3905:(e,n,t)=>{t.d(n,{Zo:()=>d,kt:()=>m});var i=t(7294);function a(e,n,t){return n in e?Object.defineProperty(e,n,{value:t,enumerable:!0,configurable:!0,writable:!0}):e[n]=t,e}function r(e,n){var t=Object.keys(e);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(e);n&&(i=i.filter((function(n){return Object.getOwnPropertyDescriptor(e,n).enumerable}))),t.push.apply(t,i)}return t}function o(e){for(var n=1;n=0||(a[t]=e[t]);return a}(e,n);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);for(i=0;i=0||Object.prototype.propertyIsEnumerable.call(e,t)&&(a[t]=e[t])}return a}var s=i.createContext({}),c=function(e){var n=i.useContext(s),t=n;return e&&(t="function"==typeof e?e(n):o(o({},n),e)),t},d=function(e){var n=c(e.components);return i.createElement(s.Provider,{value:n},e.children)},h="mdxType",p={inlineCode:"code",wrapper:function(e){var n=e.children;return i.createElement(i.Fragment,{},n)}},u=i.forwardRef((function(e,n){var t=e.components,a=e.mdxType,r=e.originalType,s=e.parentName,d=l(e,["components","mdxType","originalType","parentName"]),h=c(t),u=a,m=h["".concat(s,".").concat(u)]||h[u]||p[u]||r;return t?i.createElement(m,o(o({ref:n},d),{},{components:t})):i.createElement(m,o({ref:n},d))}));function m(e,n){var t=arguments,a=n&&n.mdxType;if("string"==typeof e||a){var r=t.length,o=new Array(r);o[0]=u;var l={};for(var s in n)hasOwnProperty.call(n,s)&&(l[s]=n[s]);l.originalType=e,l[h]="string"==typeof e?e:a,o[1]=l;for(var c=2;c{t.d(n,{Z:()=>l});var i=t(7294);const a="codeDescContainer_ie8f",r="desc_jyqI",o="example_eYlF";function l(e){let{children:n}=e,t=i.Children.toArray(n).filter((e=>e));return i.createElement("div",{className:a},i.createElement("div",{className:r},t[0]),i.createElement("div",{className:o},t[1]))}},9674:(e,n,t)=>{t.r(n),t.d(n,{assets:()=>c,contentTitle:()=>l,default:()=>p,frontMatter:()=>o,metadata:()=>s,toc:()=>d});var i=t(7462),a=(t(7294),t(3905)),r=t(8846);const o={},l="Child containers",s={unversionedId:"advanced/child-containers",id:"advanced/child-containers",title:"Child containers",description:"With child containers, you can build up parent-child relationships between containers. This means you can have a different subset of services present in a child than in the parent container.",source:"@site/docs/advanced/child-containers.md",sourceDirName:"advanced",slug:"/advanced/child-containers",permalink:"/stashbox/docs/advanced/child-containers",draft:!1,editUrl:"https://github.com/z4kn4fein/stashbox/edit/master/docs/docs/advanced/child-containers.md",tags:[],version:"current",lastUpdatedBy:"dependabot[bot]",lastUpdatedAt:1697644420,formattedLastUpdatedAt:"Oct 18, 2023",frontMatter:{},sidebar:"docs",previous:{title:"Wrappers & resolvers",permalink:"/stashbox/docs/advanced/wrappers-resolvers"},next:{title:"Special resolution cases",permalink:"/stashbox/docs/advanced/special-resolution-cases"}},c={},d=[{value:"Example",id:"example",level:2},{value:"Accessing child containers",id:"accessing-child-containers",level:2},{value:"Resolution behavior",id:"resolution-behavior",level:2},{value:"Re-building singletons",id:"re-building-singletons",level:2},{value:"Nested child containers",id:"nested-child-containers",level:2},{value:"Dispose",id:"dispose",level:2}],h={toc:d};function p(e){let{components:n,...t}=e;return(0,a.kt)("wrapper",(0,i.Z)({},h,t,{components:n,mdxType:"MDXLayout"}),(0,a.kt)("h1",{id:"child-containers"},"Child containers"),(0,a.kt)("p",null,"With child containers, you can build up parent-child relationships between containers. This means you can have a different subset of services present in a child than in the parent container. "),(0,a.kt)("p",null,"When a dependency is missing from the child container during a resolution request, the parent will be asked to resolve the missing service. If it's found there, the parent will return only the service's registration, and the resolution request will jump back to the child. Also, child registrations with the same ",(0,a.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type")," will override the parent's services."),(0,a.kt)("p",null,"Resolving ",(0,a.kt)("inlineCode",{parentName:"p"},"IEnumerable")," and ",(0,a.kt)("a",{parentName:"p",href:"/docs/advanced/decorators"},"decorators")," also considers parent containers by default. However, this behavior can be controlled with the ",(0,a.kt)("a",{parentName:"p",href:"#resolution-behavior"},(0,a.kt)("inlineCode",{parentName:"a"},"ResolutionBehavior"))," parameter. "),(0,a.kt)("admonition",{type:"info"},(0,a.kt)("p",{parentName:"admonition"},"Child containers are the foundation of the ",(0,a.kt)("a",{parentName:"p",href:"https://github.com/z4kn4fein/stashbox-extensions-dependencyinjection#multitenant"},"ASP.NET Core multi-tenant extension"),".")),(0,a.kt)("h2",{id:"example"},"Example"),(0,a.kt)("p",null,"Here is an example case:"),(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"interface IDependency {}\n\nclass B : IDependency {}\nclass C : IDependency {}\n\nclass A \n{\n public A(IDependency dependency)\n { }\n}\n\nusing (var container = new StashboxContainer())\n{\n // register 'A' into the parent container.\n container.Register();\n\n // register 'B' as a dependency into the parent container.\n container.Register();\n\n var child = container.CreateChildContainer()\n \n // register 'C' as a dependency into the child container.\n child.Register();\n \n // 'A' is resolved from the parent and gets\n // 'C' as IDependency because the resolution\n // request was initiated on the child.\n A fromChild = child.Resolve();\n\n // 'A' gets 'B' as IDependency because the \n // resolution request was initiated on the parent.\n A fromParent = container.Resolve();\n} // using will dispose the parent along with the child.\n")),(0,a.kt)("p",null,"Let's see what's happening when we request ",(0,a.kt)("inlineCode",{parentName:"p"},"A")," from the ",(0,a.kt)("em",{parentName:"p"},"child"),":"),(0,a.kt)("ol",null,(0,a.kt)("li",{parentName:"ol"},(0,a.kt)("inlineCode",{parentName:"li"},"A")," not found in the ",(0,a.kt)("em",{parentName:"li"},"child"),", go up to the ",(0,a.kt)("em",{parentName:"li"},"parent")," and check there."),(0,a.kt)("li",{parentName:"ol"},(0,a.kt)("inlineCode",{parentName:"li"},"A")," found in the ",(0,a.kt)("em",{parentName:"li"},"parent"),", resolve."),(0,a.kt)("li",{parentName:"ol"},(0,a.kt)("inlineCode",{parentName:"li"},"A")," depends on ",(0,a.kt)("inlineCode",{parentName:"li"},"IDependency"),", go back to the ",(0,a.kt)("em",{parentName:"li"},"child")," and search ",(0,a.kt)("inlineCode",{parentName:"li"},"IDependency")," implementations."),(0,a.kt)("li",{parentName:"ol"},(0,a.kt)("inlineCode",{parentName:"li"},"C")," found in the ",(0,a.kt)("em",{parentName:"li"},"child"),", it does not have any dependencies, instantiate."),(0,a.kt)("li",{parentName:"ol"},"Inject the new ",(0,a.kt)("inlineCode",{parentName:"li"},"C")," instance into ",(0,a.kt)("inlineCode",{parentName:"li"},"A"),"."),(0,a.kt)("li",{parentName:"ol"},"All dependencies are resolved; return ",(0,a.kt)("inlineCode",{parentName:"li"},"A"),".")),(0,a.kt)("p",null,"When we make the same request on the parent, everything will go as usual because we have all dependencies in place. ",(0,a.kt)("inlineCode",{parentName:"p"},"B")," will be injected into ",(0,a.kt)("inlineCode",{parentName:"p"},"A"),"."),(0,a.kt)("admonition",{type:"info"},(0,a.kt)("p",{parentName:"admonition"},"You can ",(0,a.kt)("a",{parentName:"p",href:"/docs/configuration/container-configuration"},"re-configure")," child containers with the ",(0,a.kt)("inlineCode",{parentName:"p"},".Configure()")," method. It doesn't affect the parent container's configuration.")),(0,a.kt)("h2",{id:"accessing-child-containers"},"Accessing child containers"),(0,a.kt)("p",null,"You can identify child containers with the ",(0,a.kt)("inlineCode",{parentName:"p"},"identifier")," parameter of ",(0,a.kt)("inlineCode",{parentName:"p"},"CreateChildContainer()"),". Later, you can retrieve the given child container by passing its ID to ",(0,a.kt)("inlineCode",{parentName:"p"},"GetChildContainer()"),"."),(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},'using var container = new StashboxContainer();\ncontainer.CreateChildContainer("child");\n// ...\n\nvar child = container.GetChildContainer("child");\n')),(0,a.kt)("p",null,"Also, each child container created by a container is available through the ",(0,a.kt)("inlineCode",{parentName:"p"},"IStashboxContainer.ChildContainers")," propert."),(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},'using var container = new StashboxContainer();\ncontainer.CreateChildContainer("child1");\ncontainer.CreateChildContainer("child2");\n// ...\n\nforeach (var child in container.ChildContainers)\n{\n var id = child.Key;\n var childContainer = child.Value;\n}\n')),(0,a.kt)("h2",{id:"resolution-behavior"},"Resolution behavior"),(0,a.kt)("p",null,"You can control which level of the container hierarchy can participate in the service resolution with the ",(0,a.kt)("inlineCode",{parentName:"p"},"ResolutionBehavior")," parameter. "),(0,a.kt)("p",null,"Possible values:"),(0,a.kt)("ul",null,(0,a.kt)("li",{parentName:"ul"},(0,a.kt)("inlineCode",{parentName:"li"},"Default"),": The default behavior, it's used when the parameter is not specified. Its value is ",(0,a.kt)("inlineCode",{parentName:"li"},"Parent | Current"),", so the parents and the current container (which initiated the resolution request) can participate in the resolution request's service selection."),(0,a.kt)("li",{parentName:"ul"},(0,a.kt)("inlineCode",{parentName:"li"},"Parent"),": Indicates that parent containers (including indirect all ancestors) can participate in the resolution request's service selection."),(0,a.kt)("li",{parentName:"ul"},(0,a.kt)("inlineCode",{parentName:"li"},"Current"),": Indicates that the current container (which initiated the resolution request) can participate in the service selection."),(0,a.kt)("li",{parentName:"ul"},(0,a.kt)("inlineCode",{parentName:"li"},"ParentDependency"),": Indicates that parent containers (including indirect all ancestors) can only provide dependencies for services that are already selected for resolution.")),(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-csharp"},"interface IService {}\n\nclass A : IService {}\nclass B : IService {}\n\nusing (var container = new StashboxContainer())\n{\n // register 'A' into the parent container.\n container.Register();\n\n var child = container.CreateChildContainer()\n \n // register 'B' into the child container.\n child.Register();\n \n // 'A' is resolved because only parent\n // can participate in the resolution request.\n IService withParent = child.Resolve(ResolutionBehavior.Parent);\n\n // Only 'B' is in the collection because\n // only the caller container can take part\n // in the resolution request.\n IEnumerable allWithCurrent = child.Resolve>(ResolutionBehavior.Current);\n \n // Both 'A' and 'B' is in the collection\n // because both the parent and the caller container\n // participates in the resolution request.\n IEnumerable all = child.Resolve>(ResolutionBehavior.Current | ResolutionBehavior.Parent);\n} // using will dispose the parent along with the child.\n")),(0,a.kt)("h2",{id:"re-building-singletons"},"Re-building singletons"),(0,a.kt)("p",null,"By default, singletons are instantiated and stored only in those containers that registered them. However, you can enable the re-instantiation of singletons in child containers with the ",(0,a.kt)("inlineCode",{parentName:"p"},".WithReBuildSingletonsInChildContainer()")," ",(0,a.kt)("a",{parentName:"p",href:"/docs/configuration/container-configuration#re-build-singletons-in-child-containers"},"container configuration option"),". "),(0,a.kt)("p",null,"If it's enabled, all singletons will be re-created in those containers that initiated the resolution request. By this, re-built singletons can use overridden dependencies from child containers. "),(0,a.kt)("p",null,"Re-building in child containers does not affect the singletons instantiated in the parent container."),(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"interface IDependency {}\n\nclass B : IDependency {}\nclass C : IDependency {}\n\nclass A \n{\n public A(IDependency dependency)\n { }\n}\n\nusing (var container = new StashboxContainer(options => options.WithReBuildSingletonsInChildContainer()))\n{\n // register 'A' as a singleton into the parent container.\n container.RegisterSingleton();\n\n // register 'B' as a dependency into the parent container.\n container.Register();\n\n // 'A' gets 'B' as IDependency and will be stored\n // in the parent container as a singleton.\n A fromParent = container.Resolve();\n\n var child = container.CreateChildContainer();\n \n // register 'C' as a dependency into the child container.\n child.Register();\n\n // a new 'A' singleton will be created in\n // the child container with 'C' as IDependency.\n A fromChild = child.Resolve();\n} // using will dispose the parent along with the child.\n")),(0,a.kt)("h2",{id:"nested-child-containers"},"Nested child containers"),(0,a.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"You can build up a hierarchical tree structure from containers by creating more child containers with the ",(0,a.kt)("inlineCode",{parentName:"p"},".CreateChildContainer()")," method.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"using var container = new StashboxContainer();\n\nvar child1 = container.CreateChildContainer();\nvar child2 = child1.CreateChildContainer();\n")))),(0,a.kt)("h2",{id:"dispose"},"Dispose"),(0,a.kt)("p",null,"By default, the parent container's disposal also disposes its child containers. You can control this behavior with the ",(0,a.kt)("inlineCode",{parentName:"p"},"CreateChildContainer()")," method's ",(0,a.kt)("inlineCode",{parentName:"p"},"attachToParent")," boolean parameter."),(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"using (var container = new StashboxContainer())\n{\n using (var child1 = container.CreateChildContainer(attachToParent: false))\n {\n } // child1 will be disposed only once here.\n \n var child2 = container.CreateChildContainer();\n var child3 = container.CreateChildContainer();\n} // using will dispose the parent along with child2 and child3.\n")),(0,a.kt)("p",null,"You can safely dispose a child even if it's attached to its parent, in this case the parent's disposal will not dispose the already disposed child."),(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"using (var container = new StashboxContainer())\n{\n using (var child1 = container.CreateChildContainer())\n {\n } // child1 will be disposed only once here.\n \n var child2 = container.CreateChildContainer();\n} // using will dispose only the parent and child2.\n")))}p.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/61ac6d0c.69a685cf.js b/assets/js/61ac6d0c.69a685cf.js new file mode 100644 index 00000000..ba4b676e --- /dev/null +++ b/assets/js/61ac6d0c.69a685cf.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkwebsite=self.webpackChunkwebsite||[]).push([[834],{3905:(e,t,n)=>{n.d(t,{Zo:()=>c,kt:()=>v});var a=n(7294);function r(e,t,n){return t in e?Object.defineProperty(e,t,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[t]=n,e}function i(e,t){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);t&&(a=a.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),n.push.apply(n,a)}return n}function o(e){for(var t=1;t=0||(r[n]=e[n]);return r}(e,t);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(e);for(a=0;a=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(r[n]=e[n])}return r}var s=a.createContext({}),p=function(e){var t=a.useContext(s),n=t;return e&&(n="function"==typeof e?e(t):o(o({},t),e)),n},c=function(e){var t=p(e.components);return a.createElement(s.Provider,{value:t},e.children)},u="mdxType",d={inlineCode:"code",wrapper:function(e){var t=e.children;return a.createElement(a.Fragment,{},t)}},m=a.forwardRef((function(e,t){var n=e.components,r=e.mdxType,i=e.originalType,s=e.parentName,c=l(e,["components","mdxType","originalType","parentName"]),u=p(n),m=r,v=u["".concat(s,".").concat(m)]||u[m]||d[m]||i;return n?a.createElement(v,o(o({ref:t},c),{},{components:n})):a.createElement(v,o({ref:t},c))}));function v(e,t){var n=arguments,r=t&&t.mdxType;if("string"==typeof e||r){var i=n.length,o=new Array(i);o[0]=m;var l={};for(var s in t)hasOwnProperty.call(t,s)&&(l[s]=t[s]);l.originalType=e,l[u]="string"==typeof e?e:r,o[1]=l;for(var p=2;p{n.d(t,{Z:()=>o});var a=n(7294),r=n(6010);const i="tabItem_Ymn6";function o(e){let{children:t,hidden:n,className:o}=e;return a.createElement("div",{role:"tabpanel",className:(0,r.Z)(i,o),hidden:n},t)}},4866:(e,t,n)=>{n.d(t,{Z:()=>C});var a=n(7462),r=n(7294),i=n(6010),o=n(2466),l=n(6550),s=n(1980),p=n(7392),c=n(12);function u(e){return function(e){return r.Children.map(e,(e=>{if(!e||(0,r.isValidElement)(e)&&function(e){const{props:t}=e;return!!t&&"object"==typeof t&&"value"in t}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}(e).map((e=>{let{props:{value:t,label:n,attributes:a,default:r}}=e;return{value:t,label:n,attributes:a,default:r}}))}function d(e){const{values:t,children:n}=e;return(0,r.useMemo)((()=>{const e=t??u(n);return function(e){const t=(0,p.l)(e,((e,t)=>e.value===t.value));if(t.length>0)throw new Error(`Docusaurus error: Duplicate values "${t.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[t,n])}function m(e){let{value:t,tabValues:n}=e;return n.some((e=>e.value===t))}function v(e){let{queryString:t=!1,groupId:n}=e;const a=(0,l.k6)(),i=function(e){let{queryString:t=!1,groupId:n}=e;if("string"==typeof t)return t;if(!1===t)return null;if(!0===t&&!n)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return n??null}({queryString:t,groupId:n});return[(0,s._X)(i),(0,r.useCallback)((e=>{if(!i)return;const t=new URLSearchParams(a.location.search);t.set(i,e),a.replace({...a.location,search:t.toString()})}),[i,a])]}function k(e){const{defaultValue:t,queryString:n=!1,groupId:a}=e,i=d(e),[o,l]=(0,r.useState)((()=>function(e){let{defaultValue:t,tabValues:n}=e;if(0===n.length)throw new Error("Docusaurus error: the component requires at least one children component");if(t){if(!m({value:t,tabValues:n}))throw new Error(`Docusaurus error: The has a defaultValue "${t}" but none of its children has the corresponding value. Available values are: ${n.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return t}const a=n.find((e=>e.default))??n[0];if(!a)throw new Error("Unexpected error: 0 tabValues");return a.value}({defaultValue:t,tabValues:i}))),[s,p]=v({queryString:n,groupId:a}),[u,k]=function(e){let{groupId:t}=e;const n=function(e){return e?`docusaurus.tab.${e}`:null}(t),[a,i]=(0,c.Nk)(n);return[a,(0,r.useCallback)((e=>{n&&i.set(e)}),[n,i])]}({groupId:a}),y=(()=>{const e=s??u;return m({value:e,tabValues:i})?e:null})();(0,r.useLayoutEffect)((()=>{y&&l(y)}),[y]);return{selectedValue:o,selectValue:(0,r.useCallback)((e=>{if(!m({value:e,tabValues:i}))throw new Error(`Can't select invalid tab value=${e}`);l(e),p(e),k(e)}),[p,k,i]),tabValues:i}}var y=n(2389);const f="tabList__CuJ",h="tabItem_LNqP";function b(e){let{className:t,block:n,selectedValue:l,selectValue:s,tabValues:p}=e;const c=[],{blockElementScrollPositionUntilNextRender:u}=(0,o.o5)(),d=e=>{const t=e.currentTarget,n=c.indexOf(t),a=p[n].value;a!==l&&(u(t),s(a))},m=e=>{let t=null;switch(e.key){case"Enter":d(e);break;case"ArrowRight":{const n=c.indexOf(e.currentTarget)+1;t=c[n]??c[0];break}case"ArrowLeft":{const n=c.indexOf(e.currentTarget)-1;t=c[n]??c[c.length-1];break}}t?.focus()};return r.createElement("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,i.Z)("tabs",{"tabs--block":n},t)},p.map((e=>{let{value:t,label:n,attributes:o}=e;return r.createElement("li",(0,a.Z)({role:"tab",tabIndex:l===t?0:-1,"aria-selected":l===t,key:t,ref:e=>c.push(e),onKeyDown:m,onClick:d},o,{className:(0,i.Z)("tabs__item",h,o?.className,{"tabs__item--active":l===t})}),n??t)})))}function g(e){let{lazy:t,children:n,selectedValue:a}=e;const i=(Array.isArray(n)?n:[n]).filter(Boolean);if(t){const e=i.find((e=>e.props.value===a));return e?(0,r.cloneElement)(e,{className:"margin-top--md"}):null}return r.createElement("div",{className:"margin-top--md"},i.map(((e,t)=>(0,r.cloneElement)(e,{key:t,hidden:e.props.value!==a}))))}function N(e){const t=k(e);return r.createElement("div",{className:(0,i.Z)("tabs-container",f)},r.createElement(b,(0,a.Z)({},e,t)),r.createElement(g,(0,a.Z)({},e,t)))}function C(e){const t=(0,y.Z)();return r.createElement(N,(0,a.Z)({key:String(t)},e))}},8846:(e,t,n)=>{n.d(t,{Z:()=>l});var a=n(7294);const r="codeDescContainer_ie8f",i="desc_jyqI",o="example_eYlF";function l(e){let{children:t}=e,n=a.Children.toArray(t).filter((e=>e));return a.createElement("div",{className:r},a.createElement("div",{className:i},n[0]),a.createElement("div",{className:o},n[1]))}},2328:(e,t,n)=>{n.r(t),n.d(t,{assets:()=>u,contentTitle:()=>p,default:()=>v,frontMatter:()=>s,metadata:()=>c,toc:()=>d});var a=n(7462),r=(n(7294),n(3905)),i=n(8846),o=n(4866),l=n(5162);const s={},p="Wrappers & resolvers",c={unversionedId:"advanced/wrappers-resolvers",id:"advanced/wrappers-resolvers",title:"Wrappers & resolvers",description:"Stashbox uses so-called Wrapper and Resolver implementations to handle special resolution requests that none of the service registrations can fulfill. Functionalities like wrapper and unknown type resolution, cross-container requests, optional and default value injection are all built with resolvers.",source:"@site/docs/advanced/wrappers-resolvers.md",sourceDirName:"advanced",slug:"/advanced/wrappers-resolvers",permalink:"/stashbox/docs/advanced/wrappers-resolvers",draft:!1,editUrl:"https://github.com/z4kn4fein/stashbox/edit/master/docs/docs/advanced/wrappers-resolvers.md",tags:[],version:"current",lastUpdatedBy:"dependabot[bot]",lastUpdatedAt:1699538162,formattedLastUpdatedAt:"Nov 9, 2023",frontMatter:{},sidebar:"docs",previous:{title:"Decorators",permalink:"/stashbox/docs/advanced/decorators"},next:{title:"Child containers",permalink:"/stashbox/docs/advanced/child-containers"}},u={},d=[{value:"Pre-defined wrappers & resolvers",id:"pre-defined-wrappers--resolvers",level:2},{value:"Wrappers",id:"wrappers",level:2},{value:"Enumerable",id:"enumerable",level:3},{value:"Lazy",id:"lazy",level:3},{value:"Delegate",id:"delegate",level:3},{value:"Metadata & Tuple",id:"metadata--tuple",level:3},{value:"KeyValuePair & ReadOnlyKeyValue",id:"keyvaluepair--readonlykeyvalue",level:3},{value:"User-defined wrappers & resolvers",id:"user-defined-wrappers--resolvers",level:2},{value:"Visiting order",id:"visiting-order",level:2}],m={toc:d};function v(e){let{components:t,...n}=e;return(0,r.kt)("wrapper",(0,a.Z)({},m,n,{components:t,mdxType:"MDXLayout"}),(0,r.kt)("h1",{id:"wrappers--resolvers"},"Wrappers & resolvers"),(0,r.kt)("p",null,"Stashbox uses so-called ",(0,r.kt)("em",{parentName:"p"},"Wrapper")," and ",(0,r.kt)("em",{parentName:"p"},"Resolver")," implementations to handle special resolution requests that none of the ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-registration--registered-service"},"service registrations")," can fulfill. Functionalities like ",(0,r.kt)("a",{parentName:"p",href:"/docs/advanced/wrappers-resolvers#wrappers"},"wrapper")," and ",(0,r.kt)("a",{parentName:"p",href:"/docs/advanced/special-resolution-cases#unknown-type-resolution"},"unknown type")," resolution, ",(0,r.kt)("a",{parentName:"p",href:"/docs/advanced/child-containers"},"cross-container requests"),", ",(0,r.kt)("a",{parentName:"p",href:"/docs/advanced/special-resolution-cases#optional-value-injection"},"optional")," and ",(0,r.kt)("a",{parentName:"p",href:"/docs/advanced/special-resolution-cases#default-value-injection"},"default value")," injection are all built with resolvers."),(0,r.kt)("h2",{id:"pre-defined-wrappers--resolvers"},"Pre-defined wrappers & resolvers"),(0,r.kt)("ul",null,(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("inlineCode",{parentName:"li"},"EnumerableWrapper"),": Used to resolve a collection of services wrapped in one of the collection interfaces that a .NET ",(0,r.kt)("inlineCode",{parentName:"li"},"Array")," implements. (",(0,r.kt)("inlineCode",{parentName:"li"},"IEnumerable<>"),", ",(0,r.kt)("inlineCode",{parentName:"li"},"IList<>"),", ",(0,r.kt)("inlineCode",{parentName:"li"},"ICollection<>"),", ",(0,r.kt)("inlineCode",{parentName:"li"},"IReadOnlyList<>"),", ",(0,r.kt)("inlineCode",{parentName:"li"},"IReadOnlyCollection<>"),") "),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("inlineCode",{parentName:"li"},"LazyWrapper"),": Used to resolve services ",(0,r.kt)("a",{parentName:"li",href:"/docs/advanced/wrappers-resolvers#lazy"},"wrapped")," in ",(0,r.kt)("inlineCode",{parentName:"li"},"Lazy<>"),"."),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("inlineCode",{parentName:"li"},"FuncWrapper"),": Used to resolve services ",(0,r.kt)("a",{parentName:"li",href:"/docs/advanced/wrappers-resolvers#delegate"},"wrapped")," in a ",(0,r.kt)("inlineCode",{parentName:"li"},"Delegate")," that has a non-void return type like ",(0,r.kt)("inlineCode",{parentName:"li"},"Func<>"),"."),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("inlineCode",{parentName:"li"},"MetadataWrapper"),": Used to resolve services ",(0,r.kt)("a",{parentName:"li",href:"/docs/advanced/wrappers-resolvers#metadata--tuple"},"wrapped")," in ",(0,r.kt)("inlineCode",{parentName:"li"},"ValueTuple<,>"),", ",(0,r.kt)("inlineCode",{parentName:"li"},"Tuple<,>"),", or ",(0,r.kt)("inlineCode",{parentName:"li"},"Metadata<,>"),"."),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("inlineCode",{parentName:"li"},"KeyValueWrapper"),": Used to resolve services ",(0,r.kt)("a",{parentName:"li",href:"/docs/advanced/wrappers-resolvers#keyvaluepair--readonlykeyvalue"},"wrapped")," in ",(0,r.kt)("inlineCode",{parentName:"li"},"KeyValuePair<,>")," or ",(0,r.kt)("inlineCode",{parentName:"li"},"ReadOnlyKeyValue<,>"),"."),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("inlineCode",{parentName:"li"},"ServiceProviderResolver"),": Used to resolve the actual scope as ",(0,r.kt)("inlineCode",{parentName:"li"},"IServiceProvider")," when no other implementation is registered."),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("inlineCode",{parentName:"li"},"OptionalValueResolver"),": Used to resolve optional parameters."),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("inlineCode",{parentName:"li"},"DefaultValueResolver"),": Used to resolve default values."),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("inlineCode",{parentName:"li"},"ParentContainerResolver"),": Used to resolve services that are only registered in one of the parent containers."),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("inlineCode",{parentName:"li"},"UnknownTypeResolver"),": Used to resolve services that are not registered into the container.")),(0,r.kt)("h2",{id:"wrappers"},"Wrappers"),(0,r.kt)("p",null,"Stashbox can implicitly wrap your services into different data structures. All functionalities covered in the ",(0,r.kt)("a",{parentName:"p",href:"/docs/guides/service-resolution"},"service resolution")," are applied to the wrappers. Every wrapper request starts as a standard resolution; only the result is wrapped in the requested structure."),(0,r.kt)(i.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("h3",{id:"enumerable"},"Enumerable"),(0,r.kt)("p",null,"Stashbox can compose a collection from each implementation registered to a ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type"),". The requested type can be wrapped by any of the collection interfaces that a .NET ",(0,r.kt)("inlineCode",{parentName:"p"},"Array")," implements.")),(0,r.kt)("div",null,(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"IJob[] jobs = container.Resolve();\nIEnumerable jobs = container.Resolve>();\nIList jobs = container.Resolve>();\nICollection jobs = container.Resolve>();\nIReadOnlyList jobs = container.Resolve>();\nIReadOnlyCollection jobs = container.Resolve>();\n")))),(0,r.kt)(i.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("h3",{id:"lazy"},"Lazy"),(0,r.kt)("p",null,"When requesting ",(0,r.kt)("inlineCode",{parentName:"p"},"Lazy<>"),", the container implicitly constructs a new ",(0,r.kt)("inlineCode",{parentName:"p"},"Lazy<>")," instance with a factory delegate as its constructor argument used to instantiate the underlying service. ")),(0,r.kt)("div",null,(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register();\n\n// new Lazy(() => new DbBackup())\nLazy lazyJob = container.Resolve>();\nIJob job = lazyJob.Value;\n")))),(0,r.kt)(i.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("h3",{id:"delegate"},"Delegate"),(0,r.kt)("p",null,"When requesting a ",(0,r.kt)("inlineCode",{parentName:"p"},"Delegate"),", the container implicitly creates a factory used to instantiate the underlying service."),(0,r.kt)("p",null,"It's possible to request a delegate that expects some or all of the dependencies as delegate parameters.\nParameters are used for sub-dependencies as well, like: ",(0,r.kt)("inlineCode",{parentName:"p"},"(arg) => new A(new B(arg))")),(0,r.kt)("p",null,"When a dependency is not available as a parameter, it will be resolved from the container directly.")),(0,r.kt)("div",null,(0,r.kt)(o.Z,{mdxType:"Tabs"},(0,r.kt)(l.Z,{value:"Func",label:"Func",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register();\n\n// (conn, logger) => new DbBackup(conn, logger)\nFunc funcOfJob = container\n .Resolve>();\n \nIJob job = funcOfJob(config["connectionString"], new ConsoleLogger());\n'))),(0,r.kt)(l.Z,{value:"Custom delegate",label:"Custom delegate",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},'private delegate IJob JobFactory(string connectionString, ILogger logger);\n\ncontainer.Register();\n\nvar jobDelegate = container.Resolve();\nIJob job = jobDelegate(config["connectionString"], new ConsoleLogger());\n')))))),(0,r.kt)(i.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("h3",{id:"metadata--tuple"},"Metadata & Tuple"),(0,r.kt)("p",null,"With the ",(0,r.kt)("inlineCode",{parentName:"p"},".WithMetadata()")," registration option, you can attach additional information to a service.\nTo gather this information, you can request the service wrapped in either ",(0,r.kt)("inlineCode",{parentName:"p"},"Metadata<,>"),", ",(0,r.kt)("inlineCode",{parentName:"p"},"ValueTuple<,>"),", or ",(0,r.kt)("inlineCode",{parentName:"p"},"Tuple<,>"),"."),(0,r.kt)("p",null,(0,r.kt)("inlineCode",{parentName:"p"},"Metadata<,>")," is a type from the ",(0,r.kt)("inlineCode",{parentName:"p"},"Stashbox")," package, so you might prefer using ",(0,r.kt)("inlineCode",{parentName:"p"},"ValueTuple<,>")," or ",(0,r.kt)("inlineCode",{parentName:"p"},"Tuple<,>")," if you want to avoid referencing Stashbox in certain parts of your project."),(0,r.kt)("p",null,"You can also filter a collection of services by their metadata. Requesting ",(0,r.kt)("inlineCode",{parentName:"p"},"IEnumerable>")," will yield only those services that have the given type of metadata.")),(0,r.kt)("div",null,(0,r.kt)(o.Z,{mdxType:"Tabs"},(0,r.kt)(l.Z,{value:"Single service",label:"Single service",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register(options => options\n .WithMetadata("connection-string-to-db"));\n\nvar jobWithConnectionString = container.Resolve>();\n// prints: "connection-string-to-db"\nConsole.WriteLine(jobWithConnectionString.Data);\n\nvar alsoJobWithConnectionString = container.Resolve>();\n// prints: "connection-string-to-db"\nConsole.WriteLine(alsoJobWithConnectionString.Item2);\n\nvar stillJobWithConnectionString = container.Resolve>();\n// prints: "connection-string-to-db"\nConsole.WriteLine(stillJobWithConnectionString.Item2);\n'))),(0,r.kt)(l.Z,{value:"Collection filtering",label:"Collection filtering",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register(options => options\n .WithMetadata("meta-1"));\ncontainer.Register(options => options\n .WithMetadata("meta-2"));\ncontainer.Register(options => options\n .WithMetadata(5));\n\n// the result is: [Service1, Service2]\nvar servicesWithStringMetadata = container.Resolve[]>();\n\n// the result is: [Service3]\nvar servicesWithIntMetadata = container.Resolve[]>();\n')))))),(0,r.kt)("admonition",{type:"note"},(0,r.kt)("p",{parentName:"admonition"},"Metadata can also be a complex type e.g., an ",(0,r.kt)("inlineCode",{parentName:"p"},"IDictionary<,>"),".")),(0,r.kt)("admonition",{type:"info"},(0,r.kt)("p",{parentName:"admonition"},"When no service found for a particular metadata type, the container throws a ",(0,r.kt)("a",{parentName:"p",href:"/docs/diagnostics/validation#resolution-validation"},"ResolutionFailedException"),". In case of an ",(0,r.kt)("inlineCode",{parentName:"p"},"IEnumerable<>")," request, an empty collection will be returned for a non-existing metadata.")),(0,r.kt)(i.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("h3",{id:"keyvaluepair--readonlykeyvalue"},"KeyValuePair & ReadOnlyKeyValue"),(0,r.kt)("p",null,"With named registration, you can give your service unique identifiers. Requesting a service wrapped in a ",(0,r.kt)("inlineCode",{parentName:"p"},"KeyValuePair")," or ",(0,r.kt)("inlineCode",{parentName:"p"},"ReadOnlyKeyValue")," returns the requested service with its identifier as key."),(0,r.kt)("p",null,(0,r.kt)("inlineCode",{parentName:"p"},"ReadOnlyKeyValue<,>")," is a type from the ",(0,r.kt)("inlineCode",{parentName:"p"},"Stashbox")," package, so you might prefer using ",(0,r.kt)("inlineCode",{parentName:"p"},"KeyValuePair<,>")," if you want to avoid referencing Stashbox in certain parts of your project."),(0,r.kt)("p",null,"Requesting an ",(0,r.kt)("inlineCode",{parentName:"p"},"IEnumerable>")," will return all services of the requested type along their identifiers. When a service don't have an identifier the ",(0,r.kt)("inlineCode",{parentName:"p"},"Key")," will be set to ",(0,r.kt)("inlineCode",{parentName:"p"},"null"),".")),(0,r.kt)("div",null,(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register("FirstServiceId");\ncontainer.Register("SecondServiceId");\ncontainer.Register();\n\nvar serviceKeyValue1 = container\n .Resolve>("FirstServiceId");\n// prints: "FirstServiceId"\nConsole.WriteLine(serviceKeyValue1.Key);\n\nvar serviceKeyValue2 = container\n .Resolve>("SecondServiceId");\n// prints: "SecondServiceId"\nConsole.WriteLine(serviceKeyValue2.Key);\n\n// ["FirstServiceId": Service1, "SecondServiceId": Service2, null: Service3 ]\nvar servicesWithKeys = container.Resolve[]>();\n')))),(0,r.kt)("admonition",{type:"note"},(0,r.kt)("p",{parentName:"admonition"},"Wrappers can be composed e.g., ",(0,r.kt)("inlineCode",{parentName:"p"},"IEnumerable, string>>>"),".")),(0,r.kt)("h2",{id:"user-defined-wrappers--resolvers"},"User-defined wrappers & resolvers"),(0,r.kt)("p",null,"You can add support for more wrapper types by implementing the ",(0,r.kt)("inlineCode",{parentName:"p"},"IServiceWrapper")," interface."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"class CustomWrapper : IServiceWrapper\n{\n // this method is supposed to generate the expression for the given wrapper's \n // instantiation when it's selected by the container to resolve the actual service.\n public Expression WrapExpression(\n TypeInformation originalTypeInformation, \n TypeInformation wrappedTypeInformation, \n ServiceContext serviceContext)\n {\n // produce the expression for the wrapper.\n }\n\n // this method is called by the container to determine whether a \n // given requested type is wrapped by a supported wrapper type.\n public bool TryUnWrap(Type type, out Type unWrappedType)\n {\n // this is just a reference implementation of \n // un-wrapping a service from a given wrapper.\n if (!CanUnWrapServiceType(type))\n {\n unWrappedType = typeof(object);\n return false;\n }\n\n unWrappedType = UnWrapServiceType(type);\n return true;\n }\n}\n")),(0,r.kt)("p",null,"You can extend the functionality of the container by implementing the ",(0,r.kt)("inlineCode",{parentName:"p"},"IServiceResolver")," interface."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"class CustomResolver : IServiceResolver\n{\n // called to generate the expression for the given service\n // when this resolver is selected (through CanUseForResolution()) \n // to fulfill the request.\n public ServiceContext GetExpression(\n IResolutionStrategy resolutionStrategy,\n TypeInformation typeInfo,\n ResolutionContext resolutionContext)\n {\n var expression = GenerateExpression(); // resolution expression generation.\n return expression.AsServiceContext();\n }\n\n public bool CanUseForResolution(\n TypeInformation typeInfo,\n ResolutionContext resolutionContext)\n {\n // the predicate that determines whether the resolver \n // is able to resolve the requested service or not.\n return IsUsableFor(typeInfo);\n }\n}\n")),(0,r.kt)("p",null,"Then you can register your custom wrapper or resolver like this:"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterResolver(new CustomWrapper());\ncontainer.RegisterResolver(new CustomResolver());\n")),(0,r.kt)("h2",{id:"visiting-order"},"Visiting order"),(0,r.kt)("p",null,"Stashbox visits the wrappers and resolvers in the following order to satisfy the actual resolution request:"),(0,r.kt)("ol",null,(0,r.kt)("li",{parentName:"ol"},(0,r.kt)("inlineCode",{parentName:"li"},"EnumerableWrapper")),(0,r.kt)("li",{parentName:"ol"},(0,r.kt)("inlineCode",{parentName:"li"},"LazyWrapper")),(0,r.kt)("li",{parentName:"ol"},(0,r.kt)("inlineCode",{parentName:"li"},"FuncWrapper")),(0,r.kt)("li",{parentName:"ol"},(0,r.kt)("inlineCode",{parentName:"li"},"MetadataWrapper")),(0,r.kt)("li",{parentName:"ol"},(0,r.kt)("inlineCode",{parentName:"li"},"KeyValueWrapper")),(0,r.kt)("li",{parentName:"ol"},(0,r.kt)("strong",{parentName:"li"},"Custom, user-defined wrappers & resolvers")),(0,r.kt)("li",{parentName:"ol"},(0,r.kt)("inlineCode",{parentName:"li"},"ServiceProviderResolver")),(0,r.kt)("li",{parentName:"ol"},(0,r.kt)("inlineCode",{parentName:"li"},"OptionalValueResolver")),(0,r.kt)("li",{parentName:"ol"},(0,r.kt)("inlineCode",{parentName:"li"},"DefaultValueResolver")),(0,r.kt)("li",{parentName:"ol"},(0,r.kt)("inlineCode",{parentName:"li"},"ParentContainerResolver")),(0,r.kt)("li",{parentName:"ol"},(0,r.kt)("inlineCode",{parentName:"li"},"UnknownTypeResolver"))))}v.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/61ac6d0c.a648eb65.js b/assets/js/61ac6d0c.a648eb65.js deleted file mode 100644 index 2fc30841..00000000 --- a/assets/js/61ac6d0c.a648eb65.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkwebsite=self.webpackChunkwebsite||[]).push([[834],{3905:(e,t,n)=>{n.d(t,{Zo:()=>c,kt:()=>v});var a=n(7294);function r(e,t,n){return t in e?Object.defineProperty(e,t,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[t]=n,e}function i(e,t){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);t&&(a=a.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),n.push.apply(n,a)}return n}function o(e){for(var t=1;t=0||(r[n]=e[n]);return r}(e,t);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(e);for(a=0;a=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(r[n]=e[n])}return r}var s=a.createContext({}),p=function(e){var t=a.useContext(s),n=t;return e&&(n="function"==typeof e?e(t):o(o({},t),e)),n},c=function(e){var t=p(e.components);return a.createElement(s.Provider,{value:t},e.children)},u="mdxType",d={inlineCode:"code",wrapper:function(e){var t=e.children;return a.createElement(a.Fragment,{},t)}},m=a.forwardRef((function(e,t){var n=e.components,r=e.mdxType,i=e.originalType,s=e.parentName,c=l(e,["components","mdxType","originalType","parentName"]),u=p(n),m=r,v=u["".concat(s,".").concat(m)]||u[m]||d[m]||i;return n?a.createElement(v,o(o({ref:t},c),{},{components:n})):a.createElement(v,o({ref:t},c))}));function v(e,t){var n=arguments,r=t&&t.mdxType;if("string"==typeof e||r){var i=n.length,o=new Array(i);o[0]=m;var l={};for(var s in t)hasOwnProperty.call(t,s)&&(l[s]=t[s]);l.originalType=e,l[u]="string"==typeof e?e:r,o[1]=l;for(var p=2;p{n.d(t,{Z:()=>o});var a=n(7294),r=n(6010);const i="tabItem_Ymn6";function o(e){let{children:t,hidden:n,className:o}=e;return a.createElement("div",{role:"tabpanel",className:(0,r.Z)(i,o),hidden:n},t)}},4866:(e,t,n)=>{n.d(t,{Z:()=>C});var a=n(7462),r=n(7294),i=n(6010),o=n(2466),l=n(6550),s=n(1980),p=n(7392),c=n(12);function u(e){return function(e){return r.Children.map(e,(e=>{if(!e||(0,r.isValidElement)(e)&&function(e){const{props:t}=e;return!!t&&"object"==typeof t&&"value"in t}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}(e).map((e=>{let{props:{value:t,label:n,attributes:a,default:r}}=e;return{value:t,label:n,attributes:a,default:r}}))}function d(e){const{values:t,children:n}=e;return(0,r.useMemo)((()=>{const e=t??u(n);return function(e){const t=(0,p.l)(e,((e,t)=>e.value===t.value));if(t.length>0)throw new Error(`Docusaurus error: Duplicate values "${t.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[t,n])}function m(e){let{value:t,tabValues:n}=e;return n.some((e=>e.value===t))}function v(e){let{queryString:t=!1,groupId:n}=e;const a=(0,l.k6)(),i=function(e){let{queryString:t=!1,groupId:n}=e;if("string"==typeof t)return t;if(!1===t)return null;if(!0===t&&!n)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return n??null}({queryString:t,groupId:n});return[(0,s._X)(i),(0,r.useCallback)((e=>{if(!i)return;const t=new URLSearchParams(a.location.search);t.set(i,e),a.replace({...a.location,search:t.toString()})}),[i,a])]}function k(e){const{defaultValue:t,queryString:n=!1,groupId:a}=e,i=d(e),[o,l]=(0,r.useState)((()=>function(e){let{defaultValue:t,tabValues:n}=e;if(0===n.length)throw new Error("Docusaurus error: the component requires at least one children component");if(t){if(!m({value:t,tabValues:n}))throw new Error(`Docusaurus error: The has a defaultValue "${t}" but none of its children has the corresponding value. Available values are: ${n.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return t}const a=n.find((e=>e.default))??n[0];if(!a)throw new Error("Unexpected error: 0 tabValues");return a.value}({defaultValue:t,tabValues:i}))),[s,p]=v({queryString:n,groupId:a}),[u,k]=function(e){let{groupId:t}=e;const n=function(e){return e?`docusaurus.tab.${e}`:null}(t),[a,i]=(0,c.Nk)(n);return[a,(0,r.useCallback)((e=>{n&&i.set(e)}),[n,i])]}({groupId:a}),y=(()=>{const e=s??u;return m({value:e,tabValues:i})?e:null})();(0,r.useLayoutEffect)((()=>{y&&l(y)}),[y]);return{selectedValue:o,selectValue:(0,r.useCallback)((e=>{if(!m({value:e,tabValues:i}))throw new Error(`Can't select invalid tab value=${e}`);l(e),p(e),k(e)}),[p,k,i]),tabValues:i}}var y=n(2389);const f="tabList__CuJ",h="tabItem_LNqP";function b(e){let{className:t,block:n,selectedValue:l,selectValue:s,tabValues:p}=e;const c=[],{blockElementScrollPositionUntilNextRender:u}=(0,o.o5)(),d=e=>{const t=e.currentTarget,n=c.indexOf(t),a=p[n].value;a!==l&&(u(t),s(a))},m=e=>{let t=null;switch(e.key){case"Enter":d(e);break;case"ArrowRight":{const n=c.indexOf(e.currentTarget)+1;t=c[n]??c[0];break}case"ArrowLeft":{const n=c.indexOf(e.currentTarget)-1;t=c[n]??c[c.length-1];break}}t?.focus()};return r.createElement("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,i.Z)("tabs",{"tabs--block":n},t)},p.map((e=>{let{value:t,label:n,attributes:o}=e;return r.createElement("li",(0,a.Z)({role:"tab",tabIndex:l===t?0:-1,"aria-selected":l===t,key:t,ref:e=>c.push(e),onKeyDown:m,onClick:d},o,{className:(0,i.Z)("tabs__item",h,o?.className,{"tabs__item--active":l===t})}),n??t)})))}function g(e){let{lazy:t,children:n,selectedValue:a}=e;const i=(Array.isArray(n)?n:[n]).filter(Boolean);if(t){const e=i.find((e=>e.props.value===a));return e?(0,r.cloneElement)(e,{className:"margin-top--md"}):null}return r.createElement("div",{className:"margin-top--md"},i.map(((e,t)=>(0,r.cloneElement)(e,{key:t,hidden:e.props.value!==a}))))}function N(e){const t=k(e);return r.createElement("div",{className:(0,i.Z)("tabs-container",f)},r.createElement(b,(0,a.Z)({},e,t)),r.createElement(g,(0,a.Z)({},e,t)))}function C(e){const t=(0,y.Z)();return r.createElement(N,(0,a.Z)({key:String(t)},e))}},8846:(e,t,n)=>{n.d(t,{Z:()=>l});var a=n(7294);const r="codeDescContainer_ie8f",i="desc_jyqI",o="example_eYlF";function l(e){let{children:t}=e,n=a.Children.toArray(t).filter((e=>e));return a.createElement("div",{className:r},a.createElement("div",{className:i},n[0]),a.createElement("div",{className:o},n[1]))}},2328:(e,t,n)=>{n.r(t),n.d(t,{assets:()=>u,contentTitle:()=>p,default:()=>v,frontMatter:()=>s,metadata:()=>c,toc:()=>d});var a=n(7462),r=(n(7294),n(3905)),i=n(8846),o=n(4866),l=n(5162);const s={},p="Wrappers & resolvers",c={unversionedId:"advanced/wrappers-resolvers",id:"advanced/wrappers-resolvers",title:"Wrappers & resolvers",description:"Stashbox uses so-called Wrapper and Resolver implementations to handle special resolution requests that none of the service registrations can fulfill. Functionalities like wrapper and unknown type resolution, cross-container requests, optional and default value injection are all built with resolvers.",source:"@site/docs/advanced/wrappers-resolvers.md",sourceDirName:"advanced",slug:"/advanced/wrappers-resolvers",permalink:"/stashbox/docs/advanced/wrappers-resolvers",draft:!1,editUrl:"https://github.com/z4kn4fein/stashbox/edit/master/docs/docs/advanced/wrappers-resolvers.md",tags:[],version:"current",lastUpdatedBy:"dependabot[bot]",lastUpdatedAt:1697644420,formattedLastUpdatedAt:"Oct 18, 2023",frontMatter:{},sidebar:"docs",previous:{title:"Decorators",permalink:"/stashbox/docs/advanced/decorators"},next:{title:"Child containers",permalink:"/stashbox/docs/advanced/child-containers"}},u={},d=[{value:"Pre-defined wrappers & resolvers",id:"pre-defined-wrappers--resolvers",level:2},{value:"Wrappers",id:"wrappers",level:2},{value:"Enumerable",id:"enumerable",level:3},{value:"Lazy",id:"lazy",level:3},{value:"Delegate",id:"delegate",level:3},{value:"Metadata & Tuple",id:"metadata--tuple",level:3},{value:"KeyValuePair & ReadOnlyKeyValue",id:"keyvaluepair--readonlykeyvalue",level:3},{value:"User-defined wrappers & resolvers",id:"user-defined-wrappers--resolvers",level:2},{value:"Visiting order",id:"visiting-order",level:2}],m={toc:d};function v(e){let{components:t,...n}=e;return(0,r.kt)("wrapper",(0,a.Z)({},m,n,{components:t,mdxType:"MDXLayout"}),(0,r.kt)("h1",{id:"wrappers--resolvers"},"Wrappers & resolvers"),(0,r.kt)("p",null,"Stashbox uses so-called ",(0,r.kt)("em",{parentName:"p"},"Wrapper")," and ",(0,r.kt)("em",{parentName:"p"},"Resolver")," implementations to handle special resolution requests that none of the ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-registration--registered-service"},"service registrations")," can fulfill. Functionalities like ",(0,r.kt)("a",{parentName:"p",href:"/docs/advanced/wrappers-resolvers#wrappers"},"wrapper")," and ",(0,r.kt)("a",{parentName:"p",href:"/docs/advanced/special-resolution-cases#unknown-type-resolution"},"unknown type")," resolution, ",(0,r.kt)("a",{parentName:"p",href:"/docs/advanced/child-containers"},"cross-container requests"),", ",(0,r.kt)("a",{parentName:"p",href:"/docs/advanced/special-resolution-cases#optional-value-injection"},"optional")," and ",(0,r.kt)("a",{parentName:"p",href:"/docs/advanced/special-resolution-cases#default-value-injection"},"default value")," injection are all built with resolvers."),(0,r.kt)("h2",{id:"pre-defined-wrappers--resolvers"},"Pre-defined wrappers & resolvers"),(0,r.kt)("ul",null,(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("inlineCode",{parentName:"li"},"EnumerableWrapper"),": Used to resolve a collection of services wrapped in one of the collection interfaces that a .NET ",(0,r.kt)("inlineCode",{parentName:"li"},"Array")," implements. (",(0,r.kt)("inlineCode",{parentName:"li"},"IEnumerable<>"),", ",(0,r.kt)("inlineCode",{parentName:"li"},"IList<>"),", ",(0,r.kt)("inlineCode",{parentName:"li"},"ICollection<>"),", ",(0,r.kt)("inlineCode",{parentName:"li"},"IReadOnlyList<>"),", ",(0,r.kt)("inlineCode",{parentName:"li"},"IReadOnlyCollection<>"),") "),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("inlineCode",{parentName:"li"},"LazyWrapper"),": Used to resolve services ",(0,r.kt)("a",{parentName:"li",href:"/docs/advanced/wrappers-resolvers#lazy"},"wrapped")," in ",(0,r.kt)("inlineCode",{parentName:"li"},"Lazy<>"),"."),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("inlineCode",{parentName:"li"},"FuncWrapper"),": Used to resolve services ",(0,r.kt)("a",{parentName:"li",href:"/docs/advanced/wrappers-resolvers#delegate"},"wrapped")," in a ",(0,r.kt)("inlineCode",{parentName:"li"},"Delegate")," that has a non-void return type like ",(0,r.kt)("inlineCode",{parentName:"li"},"Func<>"),"."),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("inlineCode",{parentName:"li"},"MetadataWrapper"),": Used to resolve services ",(0,r.kt)("a",{parentName:"li",href:"/docs/advanced/wrappers-resolvers#metadata--tuple"},"wrapped")," in ",(0,r.kt)("inlineCode",{parentName:"li"},"ValueTuple<,>"),", ",(0,r.kt)("inlineCode",{parentName:"li"},"Tuple<,>"),", or ",(0,r.kt)("inlineCode",{parentName:"li"},"Metadata<,>"),"."),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("inlineCode",{parentName:"li"},"KeyValueWrapper"),": Used to resolve services ",(0,r.kt)("a",{parentName:"li",href:"/docs/advanced/wrappers-resolvers#keyvaluepair--readonlykeyvalue"},"wrapped")," in ",(0,r.kt)("inlineCode",{parentName:"li"},"KeyValuePair<,>")," or ",(0,r.kt)("inlineCode",{parentName:"li"},"ReadOnlyKeyValue<,>"),"."),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("inlineCode",{parentName:"li"},"ServiceProviderResolver"),": Used to resolve the actual scope as ",(0,r.kt)("inlineCode",{parentName:"li"},"IServiceProvider")," when no other implementation is registered."),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("inlineCode",{parentName:"li"},"OptionalValueResolver"),": Used to resolve optional parameters."),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("inlineCode",{parentName:"li"},"DefaultValueResolver"),": Used to resolve default values."),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("inlineCode",{parentName:"li"},"ParentContainerResolver"),": Used to resolve services that are only registered in one of the parent containers."),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("inlineCode",{parentName:"li"},"UnknownTypeResolver"),": Used to resolve services that are not registered into the container.")),(0,r.kt)("h2",{id:"wrappers"},"Wrappers"),(0,r.kt)("p",null,"Stashbox can implicitly wrap your services into different data structures. All functionalities covered in the ",(0,r.kt)("a",{parentName:"p",href:"/docs/guides/service-resolution"},"service resolution")," are applied to the wrappers. Every wrapper request starts as a standard resolution; only the result is wrapped in the requested structure."),(0,r.kt)(i.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("h3",{id:"enumerable"},"Enumerable"),(0,r.kt)("p",null,"Stashbox can compose a collection from each implementation registered to a ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type"),". The requested type can be wrapped by any of the collection interfaces that a .NET ",(0,r.kt)("inlineCode",{parentName:"p"},"Array")," implements.")),(0,r.kt)("div",null,(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"IJob[] jobs = container.Resolve();\nIEnumerable jobs = container.Resolve>();\nIList jobs = container.Resolve>();\nICollection jobs = container.Resolve>();\nIReadOnlyList jobs = container.Resolve>();\nIReadOnlyCollection jobs = container.Resolve>();\n")))),(0,r.kt)(i.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("h3",{id:"lazy"},"Lazy"),(0,r.kt)("p",null,"When requesting ",(0,r.kt)("inlineCode",{parentName:"p"},"Lazy<>"),", the container implicitly constructs a new ",(0,r.kt)("inlineCode",{parentName:"p"},"Lazy<>")," instance with a factory delegate as its constructor argument used to instantiate the underlying service. ")),(0,r.kt)("div",null,(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register();\n\n// new Lazy(() => new DbBackup())\nLazy lazyJob = container.Resolve>();\nIJob job = lazyJob.Value;\n")))),(0,r.kt)(i.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("h3",{id:"delegate"},"Delegate"),(0,r.kt)("p",null,"When requesting a ",(0,r.kt)("inlineCode",{parentName:"p"},"Delegate"),", the container implicitly creates a factory used to instantiate the underlying service."),(0,r.kt)("p",null,"It's possible to request a delegate that expects some or all of the dependencies as delegate parameters.\nParameters are used for sub-dependencies as well, like: ",(0,r.kt)("inlineCode",{parentName:"p"},"(arg) => new A(new B(arg))")),(0,r.kt)("p",null,"When a dependency is not available as a parameter, it will be resolved from the container directly.")),(0,r.kt)("div",null,(0,r.kt)(o.Z,{mdxType:"Tabs"},(0,r.kt)(l.Z,{value:"Func",label:"Func",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register();\n\n// (conn, logger) => new DbBackup(conn, logger)\nFunc funcOfJob = container\n .Resolve>();\n \nIJob job = funcOfJob(config["connectionString"], new ConsoleLogger());\n'))),(0,r.kt)(l.Z,{value:"Custom delegate",label:"Custom delegate",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},'private delegate IJob JobFactory(string connectionString, ILogger logger);\n\ncontainer.Register();\n\nvar jobDelegate = container.Resolve();\nIJob job = jobDelegate(config["connectionString"], new ConsoleLogger());\n')))))),(0,r.kt)(i.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("h3",{id:"metadata--tuple"},"Metadata & Tuple"),(0,r.kt)("p",null,"With the ",(0,r.kt)("inlineCode",{parentName:"p"},".WithMetadata()")," registration option, you can attach additional information to a service.\nTo gather this information, you can request the service wrapped in either ",(0,r.kt)("inlineCode",{parentName:"p"},"Metadata<,>"),", ",(0,r.kt)("inlineCode",{parentName:"p"},"ValueTuple<,>"),", or ",(0,r.kt)("inlineCode",{parentName:"p"},"Tuple<,>"),"."),(0,r.kt)("p",null,(0,r.kt)("inlineCode",{parentName:"p"},"Metadata<,>")," is a type from the ",(0,r.kt)("inlineCode",{parentName:"p"},"Stashbox")," package, so you might prefer using ",(0,r.kt)("inlineCode",{parentName:"p"},"ValueTuple<,>")," or ",(0,r.kt)("inlineCode",{parentName:"p"},"Tuple<,>")," if you want to avoid referencing Stashbox in certain parts of your project."),(0,r.kt)("p",null,"You can also filter a collection of services by their metadata. Requesting ",(0,r.kt)("inlineCode",{parentName:"p"},"IEnumerable>")," will yield only those services that have the given type of metadata.")),(0,r.kt)("div",null,(0,r.kt)(o.Z,{mdxType:"Tabs"},(0,r.kt)(l.Z,{value:"Single service",label:"Single service",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register(options => options\n .WithMetadata("connection-string-to-db"));\n\nvar jobWithConnectionString = container.Resolve>();\n// prints: "connection-string-to-db"\nConsole.WriteLine(jobWithConnectionString.Data);\n\nvar alsoJobWithConnectionString = container.Resolve>();\n// prints: "connection-string-to-db"\nConsole.WriteLine(alsoJobWithConnectionString.Item2);\n\nvar stillJobWithConnectionString = container.Resolve>();\n// prints: "connection-string-to-db"\nConsole.WriteLine(stillJobWithConnectionString.Item2);\n'))),(0,r.kt)(l.Z,{value:"Collection filtering",label:"Collection filtering",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register(options => options\n .WithMetadata("meta-1"));\ncontainer.Register(options => options\n .WithMetadata("meta-2"));\ncontainer.Register(options => options\n .WithMetadata(5));\n\n// the result is: [Service1, Service2]\nvar servicesWithStringMetadata = container.Resolve[]>();\n\n// the result is: [Service3]\nvar servicesWithIntMetadata = container.Resolve[]>();\n')))))),(0,r.kt)("admonition",{type:"note"},(0,r.kt)("p",{parentName:"admonition"},"Metadata can also be a complex type e.g., an ",(0,r.kt)("inlineCode",{parentName:"p"},"IDictionary<,>"),".")),(0,r.kt)("admonition",{type:"info"},(0,r.kt)("p",{parentName:"admonition"},"When no service found for a particular metadata type, the container throws a ",(0,r.kt)("a",{parentName:"p",href:"/docs/diagnostics/validation#resolution-validation"},"ResolutionFailedException"),". In case of an ",(0,r.kt)("inlineCode",{parentName:"p"},"IEnumerable<>")," request, an empty collection will be returned for a non-existing metadata.")),(0,r.kt)(i.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("h3",{id:"keyvaluepair--readonlykeyvalue"},"KeyValuePair & ReadOnlyKeyValue"),(0,r.kt)("p",null,"With named registration, you can give your service unique identifiers. Requesting a service wrapped in a ",(0,r.kt)("inlineCode",{parentName:"p"},"KeyValuePair")," or ",(0,r.kt)("inlineCode",{parentName:"p"},"ReadOnlyKeyValue")," returns the requested service with its identifier as key."),(0,r.kt)("p",null,(0,r.kt)("inlineCode",{parentName:"p"},"ReadOnlyKeyValue<,>")," is a type from the ",(0,r.kt)("inlineCode",{parentName:"p"},"Stashbox")," package, so you might prefer using ",(0,r.kt)("inlineCode",{parentName:"p"},"KeyValuePair<,>")," if you want to avoid referencing Stashbox in certain parts of your project."),(0,r.kt)("p",null,"Requesting an ",(0,r.kt)("inlineCode",{parentName:"p"},"IEnumerable>")," will return all services of the requested type along their identifiers. When a service don't have an identifier the ",(0,r.kt)("inlineCode",{parentName:"p"},"Key")," will be set to ",(0,r.kt)("inlineCode",{parentName:"p"},"null"),".")),(0,r.kt)("div",null,(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register("FirstServiceId");\ncontainer.Register("SecondServiceId");\ncontainer.Register();\n\nvar serviceKeyValue1 = container\n .Resolve>("FirstServiceId");\n// prints: "FirstServiceId"\nConsole.WriteLine(serviceKeyValue1.Key);\n\nvar serviceKeyValue2 = container\n .Resolve>("SecondServiceId");\n// prints: "SecondServiceId"\nConsole.WriteLine(serviceKeyValue2.Key);\n\n// ["FirstServiceId": Service1, "SecondServiceId": Service2, null: Service3 ]\nvar servicesWithKeys = container.Resolve[]>();\n')))),(0,r.kt)("admonition",{type:"note"},(0,r.kt)("p",{parentName:"admonition"},"Wrappers can be composed e.g., ",(0,r.kt)("inlineCode",{parentName:"p"},"IEnumerable, string>>>"),".")),(0,r.kt)("h2",{id:"user-defined-wrappers--resolvers"},"User-defined wrappers & resolvers"),(0,r.kt)("p",null,"You can add support for more wrapper types by implementing the ",(0,r.kt)("inlineCode",{parentName:"p"},"IServiceWrapper")," interface."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"class CustomWrapper : IServiceWrapper\n{\n // this method is supposed to generate the expression for the given wrapper's \n // instantiation when it's selected by the container to resolve the actual service.\n public Expression WrapExpression(\n TypeInformation originalTypeInformation, \n TypeInformation wrappedTypeInformation, \n ServiceContext serviceContext)\n {\n // produce the expression for the wrapper.\n }\n\n // this method is called by the container to determine whether a \n // given requested type is wrapped by a supported wrapper type.\n public bool TryUnWrap(Type type, out Type unWrappedType)\n {\n // this is just a reference implementation of \n // un-wrapping a service from a given wrapper.\n if (!CanUnWrapServiceType(type))\n {\n unWrappedType = typeof(object);\n return false;\n }\n\n unWrappedType = UnWrapServiceType(type);\n return true;\n }\n}\n")),(0,r.kt)("p",null,"You can extend the functionality of the container by implementing the ",(0,r.kt)("inlineCode",{parentName:"p"},"IServiceResolver")," interface."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"class CustomResolver : IServiceResolver\n{\n // called to generate the expression for the given service\n // when this resolver is selected (through CanUseForResolution()) \n // to fulfill the request.\n public ServiceContext GetExpression(\n IResolutionStrategy resolutionStrategy,\n TypeInformation typeInfo,\n ResolutionContext resolutionContext)\n {\n var expression = GenerateExpression(); // resolution expression generation.\n return expression.AsServiceContext();\n }\n\n public bool CanUseForResolution(\n TypeInformation typeInfo,\n ResolutionContext resolutionContext)\n {\n // the predicate that determines whether the resolver \n // is able to resolve the requested service or not.\n return IsUsableFor(typeInfo);\n }\n}\n")),(0,r.kt)("p",null,"Then you can register your custom wrapper or resolver like this:"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterResolver(new CustomWrapper());\ncontainer.RegisterResolver(new CustomResolver());\n")),(0,r.kt)("h2",{id:"visiting-order"},"Visiting order"),(0,r.kt)("p",null,"Stashbox visits the wrappers and resolvers in the following order to satisfy the actual resolution request:"),(0,r.kt)("ol",null,(0,r.kt)("li",{parentName:"ol"},(0,r.kt)("inlineCode",{parentName:"li"},"EnumerableWrapper")),(0,r.kt)("li",{parentName:"ol"},(0,r.kt)("inlineCode",{parentName:"li"},"LazyWrapper")),(0,r.kt)("li",{parentName:"ol"},(0,r.kt)("inlineCode",{parentName:"li"},"FuncWrapper")),(0,r.kt)("li",{parentName:"ol"},(0,r.kt)("inlineCode",{parentName:"li"},"MetadataWrapper")),(0,r.kt)("li",{parentName:"ol"},(0,r.kt)("inlineCode",{parentName:"li"},"KeyValueWrapper")),(0,r.kt)("li",{parentName:"ol"},(0,r.kt)("strong",{parentName:"li"},"Custom, user-defined wrappers & resolvers")),(0,r.kt)("li",{parentName:"ol"},(0,r.kt)("inlineCode",{parentName:"li"},"ServiceProviderResolver")),(0,r.kt)("li",{parentName:"ol"},(0,r.kt)("inlineCode",{parentName:"li"},"OptionalValueResolver")),(0,r.kt)("li",{parentName:"ol"},(0,r.kt)("inlineCode",{parentName:"li"},"DefaultValueResolver")),(0,r.kt)("li",{parentName:"ol"},(0,r.kt)("inlineCode",{parentName:"li"},"ParentContainerResolver")),(0,r.kt)("li",{parentName:"ol"},(0,r.kt)("inlineCode",{parentName:"li"},"UnknownTypeResolver"))))}v.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/6d4ed487.97626423.js b/assets/js/6d4ed487.97626423.js new file mode 100644 index 00000000..b1223d1b --- /dev/null +++ b/assets/js/6d4ed487.97626423.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkwebsite=self.webpackChunkwebsite||[]).push([[302],{3905:(e,n,t)=>{t.d(n,{Zo:()=>c,kt:()=>m});var a=t(7294);function r(e,n,t){return n in e?Object.defineProperty(e,n,{value:t,enumerable:!0,configurable:!0,writable:!0}):e[n]=t,e}function l(e,n){var t=Object.keys(e);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);n&&(a=a.filter((function(n){return Object.getOwnPropertyDescriptor(e,n).enumerable}))),t.push.apply(t,a)}return t}function i(e){for(var n=1;n=0||(r[t]=e[t]);return r}(e,n);if(Object.getOwnPropertySymbols){var l=Object.getOwnPropertySymbols(e);for(a=0;a=0||Object.prototype.propertyIsEnumerable.call(e,t)&&(r[t]=e[t])}return r}var s=a.createContext({}),d=function(e){var n=a.useContext(s),t=n;return e&&(t="function"==typeof e?e(n):i(i({},n),e)),t},c=function(e){var n=d(e.components);return a.createElement(s.Provider,{value:n},e.children)},p="mdxType",u={inlineCode:"code",wrapper:function(e){var n=e.children;return a.createElement(a.Fragment,{},n)}},v=a.forwardRef((function(e,n){var t=e.components,r=e.mdxType,l=e.originalType,s=e.parentName,c=o(e,["components","mdxType","originalType","parentName"]),p=d(t),v=r,m=p["".concat(s,".").concat(v)]||p[v]||u[v]||l;return t?a.createElement(m,i(i({ref:n},c),{},{components:t})):a.createElement(m,i({ref:n},c))}));function m(e,n){var t=arguments,r=n&&n.mdxType;if("string"==typeof e||r){var l=t.length,i=new Array(l);i[0]=v;var o={};for(var s in n)hasOwnProperty.call(n,s)&&(o[s]=n[s]);o.originalType=e,o[p]="string"==typeof e?e:r,i[1]=o;for(var d=2;d{t.d(n,{Z:()=>i});var a=t(7294),r=t(6010);const l="tabItem_Ymn6";function i(e){let{children:n,hidden:t,className:i}=e;return a.createElement("div",{role:"tabpanel",className:(0,r.Z)(l,i),hidden:t},n)}},4866:(e,n,t)=>{t.d(n,{Z:()=>I});var a=t(7462),r=t(7294),l=t(6010),i=t(2466),o=t(6550),s=t(1980),d=t(7392),c=t(12);function p(e){return function(e){return r.Children.map(e,(e=>{if(!e||(0,r.isValidElement)(e)&&function(e){const{props:n}=e;return!!n&&"object"==typeof n&&"value"in n}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}(e).map((e=>{let{props:{value:n,label:t,attributes:a,default:r}}=e;return{value:n,label:t,attributes:a,default:r}}))}function u(e){const{values:n,children:t}=e;return(0,r.useMemo)((()=>{const e=n??p(t);return function(e){const n=(0,d.l)(e,((e,n)=>e.value===n.value));if(n.length>0)throw new Error(`Docusaurus error: Duplicate values "${n.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[n,t])}function v(e){let{value:n,tabValues:t}=e;return t.some((e=>e.value===n))}function m(e){let{queryString:n=!1,groupId:t}=e;const a=(0,o.k6)(),l=function(e){let{queryString:n=!1,groupId:t}=e;if("string"==typeof n)return n;if(!1===n)return null;if(!0===n&&!t)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return t??null}({queryString:n,groupId:t});return[(0,s._X)(l),(0,r.useCallback)((e=>{if(!l)return;const n=new URLSearchParams(a.location.search);n.set(l,e),a.replace({...a.location,search:n.toString()})}),[l,a])]}function f(e){const{defaultValue:n,queryString:t=!1,groupId:a}=e,l=u(e),[i,o]=(0,r.useState)((()=>function(e){let{defaultValue:n,tabValues:t}=e;if(0===t.length)throw new Error("Docusaurus error: the component requires at least one children component");if(n){if(!v({value:n,tabValues:t}))throw new Error(`Docusaurus error: The has a defaultValue "${n}" but none of its children has the corresponding value. Available values are: ${t.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return n}const a=t.find((e=>e.default))??t[0];if(!a)throw new Error("Unexpected error: 0 tabValues");return a.value}({defaultValue:n,tabValues:l}))),[s,d]=m({queryString:t,groupId:a}),[p,f]=function(e){let{groupId:n}=e;const t=function(e){return e?`docusaurus.tab.${e}`:null}(n),[a,l]=(0,c.Nk)(t);return[a,(0,r.useCallback)((e=>{t&&l.set(e)}),[t,l])]}({groupId:a}),g=(()=>{const e=s??p;return v({value:e,tabValues:l})?e:null})();(0,r.useLayoutEffect)((()=>{g&&o(g)}),[g]);return{selectedValue:i,selectValue:(0,r.useCallback)((e=>{if(!v({value:e,tabValues:l}))throw new Error(`Can't select invalid tab value=${e}`);o(e),d(e),f(e)}),[d,f,l]),tabValues:l}}var g=t(2389);const h="tabList__CuJ",E="tabItem_LNqP";function y(e){let{className:n,block:t,selectedValue:o,selectValue:s,tabValues:d}=e;const c=[],{blockElementScrollPositionUntilNextRender:p}=(0,i.o5)(),u=e=>{const n=e.currentTarget,t=c.indexOf(n),a=d[t].value;a!==o&&(p(n),s(a))},v=e=>{let n=null;switch(e.key){case"Enter":u(e);break;case"ArrowRight":{const t=c.indexOf(e.currentTarget)+1;n=c[t]??c[0];break}case"ArrowLeft":{const t=c.indexOf(e.currentTarget)-1;n=c[t]??c[c.length-1];break}}n?.focus()};return r.createElement("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,l.Z)("tabs",{"tabs--block":t},n)},d.map((e=>{let{value:n,label:t,attributes:i}=e;return r.createElement("li",(0,a.Z)({role:"tab",tabIndex:o===n?0:-1,"aria-selected":o===n,key:n,ref:e=>c.push(e),onKeyDown:v,onClick:u},i,{className:(0,l.Z)("tabs__item",E,i?.className,{"tabs__item--active":o===n})}),t??n)})))}function b(e){let{lazy:n,children:t,selectedValue:a}=e;const l=(Array.isArray(t)?t:[t]).filter(Boolean);if(n){const e=l.find((e=>e.props.value===a));return e?(0,r.cloneElement)(e,{className:"margin-top--md"}):null}return r.createElement("div",{className:"margin-top--md"},l.map(((e,n)=>(0,r.cloneElement)(e,{key:n,hidden:e.props.value!==a}))))}function k(e){const n=f(e);return r.createElement("div",{className:(0,l.Z)("tabs-container",h)},r.createElement(y,(0,a.Z)({},e,n)),r.createElement(b,(0,a.Z)({},e,n)))}function I(e){const n=(0,g.Z)();return r.createElement(k,(0,a.Z)({key:String(n)},e))}},8846:(e,n,t)=>{t.d(n,{Z:()=>o});var a=t(7294);const r="codeDescContainer_ie8f",l="desc_jyqI",i="example_eYlF";function o(e){let{children:n}=e,t=a.Children.toArray(n).filter((e=>e));return a.createElement("div",{className:r},a.createElement("div",{className:l},t[0]),a.createElement("div",{className:i},t[1]))}},7956:(e,n,t)=>{t.r(n),t.d(n,{assets:()=>p,contentTitle:()=>d,default:()=>m,frontMatter:()=>s,metadata:()=>c,toc:()=>u});var a=t(7462),r=(t(7294),t(3905)),l=t(8846),i=t(4866),o=t(5162);const s={},d="Generics",c={unversionedId:"advanced/generics",id:"advanced/generics",title:"Generics",description:"This section is about how Stashbox handles various usage scenarios that involve .NET Generic types. Including the registration of open-generic and closed-generic types, generic decorators, conditions based on generic constraints, and variance.",source:"@site/docs/advanced/generics.md",sourceDirName:"advanced",slug:"/advanced/generics",permalink:"/stashbox/docs/advanced/generics",draft:!1,editUrl:"https://github.com/z4kn4fein/stashbox/edit/master/docs/docs/advanced/generics.md",tags:[],version:"current",lastUpdatedBy:"dependabot[bot]",lastUpdatedAt:1699538162,formattedLastUpdatedAt:"Nov 9, 2023",frontMatter:{},sidebar:"docs",previous:{title:"Container configuration",permalink:"/stashbox/docs/configuration/container-configuration"},next:{title:"Decorators",permalink:"/stashbox/docs/advanced/decorators"}},p={},u=[{value:"Closed-generics",id:"closed-generics",level:2},{value:"Open-generics",id:"open-generics",level:2},{value:"Generic constraints",id:"generic-constraints",level:2},{value:"Variance",id:"variance",level:2}],v={toc:u};function m(e){let{components:n,...t}=e;return(0,r.kt)("wrapper",(0,a.Z)({},v,t,{components:n,mdxType:"MDXLayout"}),(0,r.kt)("h1",{id:"generics"},"Generics"),(0,r.kt)("p",null,"This section is about how Stashbox handles various usage scenarios that involve .NET Generic types. Including the registration of open-generic and closed-generic types, ",(0,r.kt)("a",{parentName:"p",href:"/docs/advanced/decorators#generic-decorators"},"generic decorators"),", conditions based on generic constraints, and variance."),(0,r.kt)("h2",{id:"closed-generics"},"Closed-generics"),(0,r.kt)(l.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("p",null,"The registration of a closed-generic type does not differ from registering a simple non-generic service."),(0,r.kt)("p",null,"You have all options available that you saw at the ",(0,r.kt)("a",{parentName:"p",href:"/docs/guides/basics"},"basic")," and ",(0,r.kt)("a",{parentName:"p",href:"/docs/guides/advanced-registration"},"advanced registration")," flows.")),(0,r.kt)("div",null,(0,r.kt)(i.Z,{groupId:"generic-runtime-apis",mdxType:"Tabs"},(0,r.kt)(o.Z,{value:"Generic API",label:"Generic API",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register, UserValidator>();\nIValidator validator = container.Resolve>();\n"))),(0,r.kt)(o.Z,{value:"Runtime type API",label:"Runtime type API",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(typeof(IValidator), typeof(UserValidator));\nobject validator = container.Resolve(typeof(IValidator));\n")))))),(0,r.kt)("h2",{id:"open-generics"},"Open-generics"),(0,r.kt)("p",null,"The registration of an open-generic type differs from registering a closed-generic one as C# doesn't allow the usage of open-generic types in generic method parameters. We have to get a runtime type from the open-generic type first with ",(0,r.kt)("inlineCode",{parentName:"p"},"typeof()"),"."),(0,r.kt)(l.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("p",null,"Open-generic types could help in such scenarios where you have generic interface-implementation pairs with numerous generic parameter variations. The registration of those different versions would look like this: ")),(0,r.kt)("div",null,(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register, Validator>();\ncontainer.Register, Validator>();\ncontainer.Register, Validator>();\n// and so on...\n")))),(0,r.kt)(l.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("p",null,"Rather than doing that, you can register your type's generic definition and let Stashbox bind the type parameters for you. When a matching closed ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type")," is requested, the container will construct an equivalent closed-generic implementation.")),(0,r.kt)("div",null,(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(typeof(IValidator<>), typeof(Validator<>));\n// Validator will be returned.\nIValidator userValidator = container.Resolve>();\n// Validator will be returned.\nIValidator roleValidator = container.Resolve>();\n")))),(0,r.kt)(l.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("p",null,"A registered closed-generic type always has priority over an open-generic type at service selection.")),(0,r.kt)("div",null,(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register, UserValidator>();\ncontainer.Register(typeof(IValidator<>), typeof(Validator<>));\n// UserValidator will be returned.\nIValidator validator = container.Resolve>();\n")))),(0,r.kt)("h2",{id:"generic-constraints"},"Generic constraints"),(0,r.kt)("p",null,"In the following examples, you can see how the container handles generic constraints during service resolution. Constraints can be used for ",(0,r.kt)("a",{parentName:"p",href:"/docs/guides/service-resolution#conditional-resolution"},"conditional resolution")," including collection filters. "),(0,r.kt)(i.Z,{mdxType:"Tabs"},(0,r.kt)(o.Z,{value:"Conditional resolution",label:"Conditional resolution",mdxType:"TabItem"},(0,r.kt)("p",null,"The container chooses ",(0,r.kt)("inlineCode",{parentName:"p"},"UpdatedEventHandler")," because it is the only one that has a constraint satisfied by the requested ",(0,r.kt)("inlineCode",{parentName:"p"},"UserUpdatedEvent")," generic parameter as it's implementing ",(0,r.kt)("inlineCode",{parentName:"p"},"IUpdatedEvent"),"."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"interface IEventHandler { }\n\n// event interfaces\ninterface IUpdatedEvent { }\ninterface ICreatedEvent { }\n\n// event handlers\nclass UpdatedEventHandler : IEventHandler where TEvent : IUpdatedEvent { }\nclass CreatedEventHandler : IEventHandler where TEvent : ICreatedEvent { }\n\n// event implementation\nclass UserUpdatedEvent : IUpdatedEvent { }\n\nusing var container = new StashboxContainer();\n\ncontainer.RegisterTypesAs(typeof(IEventHandler<>), new[] \n { \n typeof(UpdateEventHandler<>), \n typeof(CreateEventHandler<>) \n });\n\n// eventHandler will be UpdatedEventHandler\nIEventHandler eventHandler = container.Resolve>();\n"))),(0,r.kt)(o.Z,{value:"Collection filter",label:"Collection filter",mdxType:"TabItem"},(0,r.kt)("p",null,"This example shows how the container is filtering out those services from the returned collection that does not satisfy the given generic constraint needed to create the closed generic type."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"interface IEventHandler { }\n\n// event interfaces\ninterface IUpdatedEvent { }\ninterface ICreatedEvent { }\n\n// event handlers\nclass UpdatedEventHandler : IEventHandler where TEvent : IUpdatedEvent { }\nclass CreatedEventHandler : IEventHandler where TEvent : ICreatedEvent { }\n\n// event implementation\nclass UserUpdatedEvent : IUpdatedEvent { }\n\nusing var container = new StashboxContainer();\n\ncontainer.RegisterTypesAs(typeof(IEventHandler<>), new[] \n { \n typeof(UpdateEventHandler<>), \n typeof(CreateEventHandler<>) \n });\n\n// eventHandlers will contain only UpdatedEventHandler\nIEnumerable> eventHandlers = container.ResolveAll>();\n")))),(0,r.kt)("h2",{id:"variance"},"Variance"),(0,r.kt)("p",null,"Since .NET Framework 4.0, C# supports ",(0,r.kt)("a",{parentName:"p",href:"https://docs.microsoft.com/en-us/dotnet/csharp/programming-guide/concepts/covariance-contravariance/"},"covariance and contravariance")," in generic interfaces and delegates and allows implicit conversion of generic type parameters. In this section, we'll focus on variance in generic interfaces. "),(0,r.kt)("p",null,(0,r.kt)("a",{parentName:"p",href:"https://docs.microsoft.com/en-us/dotnet/csharp/programming-guide/concepts/covariance-contravariance/creating-variant-generic-interfaces"},"Here")," you can read more about how to create variant generic interfaces, and the following example will show how you can use them with Stashbox."),(0,r.kt)(i.Z,{mdxType:"Tabs"},(0,r.kt)(o.Z,{value:"Contravariance",label:"Contravariance",mdxType:"TabItem"},(0,r.kt)("p",null,(0,r.kt)("strong",{parentName:"p"},"Contravariance")," only allows argument types that are less derived than that defined by the generic parameters. You can declare a generic type parameter contravariant by using the ",(0,r.kt)("inlineCode",{parentName:"p"},"in")," keyword."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"// contravariant generic event handler interface\ninterface IEventHandler { } \n\n// event interfaces\ninterface IGeneralEvent { }\ninterface IUpdatedEvent : IGeneralEvent { }\n\n// event handlers\nclass GeneralEventHandler : IEventHandler { }\nclass UpdatedEventHandler : IEventHandler { }\n\ncontainer.Register, GeneralEventHandler>();\ncontainer.Register, UpdatedEventHandler>();\n\n// eventHandlers contain both GeneralEventHandler and UpdatedEventHandler\nIEnumerable> eventHandlers = container.ResolveAll>();\n")),(0,r.kt)("p",null,"Despite the fact that only ",(0,r.kt)("inlineCode",{parentName:"p"},"IEventHandler")," implementations were requested, the result contains both ",(0,r.kt)("inlineCode",{parentName:"p"},"GeneralEventHandler")," and ",(0,r.kt)("inlineCode",{parentName:"p"},"UpdatedEventHandler"),". As ",(0,r.kt)("inlineCode",{parentName:"p"},"TEvent")," is declared ",(0,r.kt)("strong",{parentName:"p"},"contravariant")," with the ",(0,r.kt)("inlineCode",{parentName:"p"},"in")," keyword, and ",(0,r.kt)("inlineCode",{parentName:"p"},"IGeneralEvent")," is less derived than ",(0,r.kt)("inlineCode",{parentName:"p"},"IUpdatedEvent"),", ",(0,r.kt)("inlineCode",{parentName:"p"},"IEventHandler")," implementations can be part of ",(0,r.kt)("inlineCode",{parentName:"p"},"IEventHandler")," collections."),(0,r.kt)("p",null,"If we request ",(0,r.kt)("inlineCode",{parentName:"p"},"IEventHandler"),", only ",(0,r.kt)("inlineCode",{parentName:"p"},"GeneralEventHandler")," would be returned, because ",(0,r.kt)("inlineCode",{parentName:"p"},"IUpdatedEvent")," is more derived, so ",(0,r.kt)("inlineCode",{parentName:"p"},"IEventHandler")," implementations are not fit into ",(0,r.kt)("inlineCode",{parentName:"p"},"IEventHandler")," collections. ")),(0,r.kt)(o.Z,{value:"Covariance",label:"Covariance",mdxType:"TabItem"},(0,r.kt)("p",null,(0,r.kt)("strong",{parentName:"p"},"Covariance")," only allows argument types that are more derived than that defined by the generic parameters. You can declare a generic type parameter covariant by using the ",(0,r.kt)("inlineCode",{parentName:"p"},"out")," keyword."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"// covariant generic event handler interface\ninterface IEventHandler { } \n\n// event interfaces\ninterface IGeneralEvent { }\ninterface IUpdatedEvent : IGeneralEvent { }\n\n// event handlers\nclass GeneralEventHandler : IEventHandler { }\nclass UpdatedEventHandler : IEventHandler { }\n\ncontainer.Register, GeneralEventHandler>();\ncontainer.Register, UpdatedEventHandler>();\n\n// eventHandlers contain both GeneralEventHandler and UpdatedEventHandler\nIEnumerable> eventHandlers = container.ResolveAll>();\n")),(0,r.kt)("p",null,"Despite the fact that only ",(0,r.kt)("inlineCode",{parentName:"p"},"IEventHandler")," implementations were requested, the result contains both ",(0,r.kt)("inlineCode",{parentName:"p"},"GeneralEventHandler")," and ",(0,r.kt)("inlineCode",{parentName:"p"},"UpdatedEventHandler"),". As ",(0,r.kt)("inlineCode",{parentName:"p"},"TEvent")," is declared ",(0,r.kt)("strong",{parentName:"p"},"covariant")," with the ",(0,r.kt)("inlineCode",{parentName:"p"},"out")," keyword, and ",(0,r.kt)("inlineCode",{parentName:"p"},"IUpdatedEvent")," is more derived than ",(0,r.kt)("inlineCode",{parentName:"p"},"IGeneralEvent"),", ",(0,r.kt)("inlineCode",{parentName:"p"},"IEventHandler")," implementations can be part of ",(0,r.kt)("inlineCode",{parentName:"p"},"IEventHandler")," collections."),(0,r.kt)("p",null,"If we request ",(0,r.kt)("inlineCode",{parentName:"p"},"IEventHandler"),", only ",(0,r.kt)("inlineCode",{parentName:"p"},"UpdatedEventHandler")," would be returned, because ",(0,r.kt)("inlineCode",{parentName:"p"},"IGeneralEvent")," is less derived, so ",(0,r.kt)("inlineCode",{parentName:"p"},"IEventHandler")," implementations are not fit into ",(0,r.kt)("inlineCode",{parentName:"p"},"IEventHandler")," collections."))))}m.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/6d4ed487.af3bba9a.js b/assets/js/6d4ed487.af3bba9a.js deleted file mode 100644 index 2d95cf76..00000000 --- a/assets/js/6d4ed487.af3bba9a.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkwebsite=self.webpackChunkwebsite||[]).push([[302],{3905:(e,n,t)=>{t.d(n,{Zo:()=>c,kt:()=>m});var a=t(7294);function r(e,n,t){return n in e?Object.defineProperty(e,n,{value:t,enumerable:!0,configurable:!0,writable:!0}):e[n]=t,e}function l(e,n){var t=Object.keys(e);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);n&&(a=a.filter((function(n){return Object.getOwnPropertyDescriptor(e,n).enumerable}))),t.push.apply(t,a)}return t}function i(e){for(var n=1;n=0||(r[t]=e[t]);return r}(e,n);if(Object.getOwnPropertySymbols){var l=Object.getOwnPropertySymbols(e);for(a=0;a=0||Object.prototype.propertyIsEnumerable.call(e,t)&&(r[t]=e[t])}return r}var s=a.createContext({}),d=function(e){var n=a.useContext(s),t=n;return e&&(t="function"==typeof e?e(n):i(i({},n),e)),t},c=function(e){var n=d(e.components);return a.createElement(s.Provider,{value:n},e.children)},p="mdxType",u={inlineCode:"code",wrapper:function(e){var n=e.children;return a.createElement(a.Fragment,{},n)}},v=a.forwardRef((function(e,n){var t=e.components,r=e.mdxType,l=e.originalType,s=e.parentName,c=o(e,["components","mdxType","originalType","parentName"]),p=d(t),v=r,m=p["".concat(s,".").concat(v)]||p[v]||u[v]||l;return t?a.createElement(m,i(i({ref:n},c),{},{components:t})):a.createElement(m,i({ref:n},c))}));function m(e,n){var t=arguments,r=n&&n.mdxType;if("string"==typeof e||r){var l=t.length,i=new Array(l);i[0]=v;var o={};for(var s in n)hasOwnProperty.call(n,s)&&(o[s]=n[s]);o.originalType=e,o[p]="string"==typeof e?e:r,i[1]=o;for(var d=2;d{t.d(n,{Z:()=>i});var a=t(7294),r=t(6010);const l="tabItem_Ymn6";function i(e){let{children:n,hidden:t,className:i}=e;return a.createElement("div",{role:"tabpanel",className:(0,r.Z)(l,i),hidden:t},n)}},4866:(e,n,t)=>{t.d(n,{Z:()=>I});var a=t(7462),r=t(7294),l=t(6010),i=t(2466),o=t(6550),s=t(1980),d=t(7392),c=t(12);function p(e){return function(e){return r.Children.map(e,(e=>{if(!e||(0,r.isValidElement)(e)&&function(e){const{props:n}=e;return!!n&&"object"==typeof n&&"value"in n}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}(e).map((e=>{let{props:{value:n,label:t,attributes:a,default:r}}=e;return{value:n,label:t,attributes:a,default:r}}))}function u(e){const{values:n,children:t}=e;return(0,r.useMemo)((()=>{const e=n??p(t);return function(e){const n=(0,d.l)(e,((e,n)=>e.value===n.value));if(n.length>0)throw new Error(`Docusaurus error: Duplicate values "${n.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[n,t])}function v(e){let{value:n,tabValues:t}=e;return t.some((e=>e.value===n))}function m(e){let{queryString:n=!1,groupId:t}=e;const a=(0,o.k6)(),l=function(e){let{queryString:n=!1,groupId:t}=e;if("string"==typeof n)return n;if(!1===n)return null;if(!0===n&&!t)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return t??null}({queryString:n,groupId:t});return[(0,s._X)(l),(0,r.useCallback)((e=>{if(!l)return;const n=new URLSearchParams(a.location.search);n.set(l,e),a.replace({...a.location,search:n.toString()})}),[l,a])]}function f(e){const{defaultValue:n,queryString:t=!1,groupId:a}=e,l=u(e),[i,o]=(0,r.useState)((()=>function(e){let{defaultValue:n,tabValues:t}=e;if(0===t.length)throw new Error("Docusaurus error: the component requires at least one children component");if(n){if(!v({value:n,tabValues:t}))throw new Error(`Docusaurus error: The has a defaultValue "${n}" but none of its children has the corresponding value. Available values are: ${t.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return n}const a=t.find((e=>e.default))??t[0];if(!a)throw new Error("Unexpected error: 0 tabValues");return a.value}({defaultValue:n,tabValues:l}))),[s,d]=m({queryString:t,groupId:a}),[p,f]=function(e){let{groupId:n}=e;const t=function(e){return e?`docusaurus.tab.${e}`:null}(n),[a,l]=(0,c.Nk)(t);return[a,(0,r.useCallback)((e=>{t&&l.set(e)}),[t,l])]}({groupId:a}),g=(()=>{const e=s??p;return v({value:e,tabValues:l})?e:null})();(0,r.useLayoutEffect)((()=>{g&&o(g)}),[g]);return{selectedValue:i,selectValue:(0,r.useCallback)((e=>{if(!v({value:e,tabValues:l}))throw new Error(`Can't select invalid tab value=${e}`);o(e),d(e),f(e)}),[d,f,l]),tabValues:l}}var g=t(2389);const h="tabList__CuJ",E="tabItem_LNqP";function y(e){let{className:n,block:t,selectedValue:o,selectValue:s,tabValues:d}=e;const c=[],{blockElementScrollPositionUntilNextRender:p}=(0,i.o5)(),u=e=>{const n=e.currentTarget,t=c.indexOf(n),a=d[t].value;a!==o&&(p(n),s(a))},v=e=>{let n=null;switch(e.key){case"Enter":u(e);break;case"ArrowRight":{const t=c.indexOf(e.currentTarget)+1;n=c[t]??c[0];break}case"ArrowLeft":{const t=c.indexOf(e.currentTarget)-1;n=c[t]??c[c.length-1];break}}n?.focus()};return r.createElement("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,l.Z)("tabs",{"tabs--block":t},n)},d.map((e=>{let{value:n,label:t,attributes:i}=e;return r.createElement("li",(0,a.Z)({role:"tab",tabIndex:o===n?0:-1,"aria-selected":o===n,key:n,ref:e=>c.push(e),onKeyDown:v,onClick:u},i,{className:(0,l.Z)("tabs__item",E,i?.className,{"tabs__item--active":o===n})}),t??n)})))}function b(e){let{lazy:n,children:t,selectedValue:a}=e;const l=(Array.isArray(t)?t:[t]).filter(Boolean);if(n){const e=l.find((e=>e.props.value===a));return e?(0,r.cloneElement)(e,{className:"margin-top--md"}):null}return r.createElement("div",{className:"margin-top--md"},l.map(((e,n)=>(0,r.cloneElement)(e,{key:n,hidden:e.props.value!==a}))))}function k(e){const n=f(e);return r.createElement("div",{className:(0,l.Z)("tabs-container",h)},r.createElement(y,(0,a.Z)({},e,n)),r.createElement(b,(0,a.Z)({},e,n)))}function I(e){const n=(0,g.Z)();return r.createElement(k,(0,a.Z)({key:String(n)},e))}},8846:(e,n,t)=>{t.d(n,{Z:()=>o});var a=t(7294);const r="codeDescContainer_ie8f",l="desc_jyqI",i="example_eYlF";function o(e){let{children:n}=e,t=a.Children.toArray(n).filter((e=>e));return a.createElement("div",{className:r},a.createElement("div",{className:l},t[0]),a.createElement("div",{className:i},t[1]))}},7956:(e,n,t)=>{t.r(n),t.d(n,{assets:()=>p,contentTitle:()=>d,default:()=>m,frontMatter:()=>s,metadata:()=>c,toc:()=>u});var a=t(7462),r=(t(7294),t(3905)),l=t(8846),i=t(4866),o=t(5162);const s={},d="Generics",c={unversionedId:"advanced/generics",id:"advanced/generics",title:"Generics",description:"This section is about how Stashbox handles various usage scenarios that involve .NET Generic types. Including the registration of open-generic and closed-generic types, generic decorators, conditions based on generic constraints, and variance.",source:"@site/docs/advanced/generics.md",sourceDirName:"advanced",slug:"/advanced/generics",permalink:"/stashbox/docs/advanced/generics",draft:!1,editUrl:"https://github.com/z4kn4fein/stashbox/edit/master/docs/docs/advanced/generics.md",tags:[],version:"current",lastUpdatedBy:"dependabot[bot]",lastUpdatedAt:1697644420,formattedLastUpdatedAt:"Oct 18, 2023",frontMatter:{},sidebar:"docs",previous:{title:"Container configuration",permalink:"/stashbox/docs/configuration/container-configuration"},next:{title:"Decorators",permalink:"/stashbox/docs/advanced/decorators"}},p={},u=[{value:"Closed-generics",id:"closed-generics",level:2},{value:"Open-generics",id:"open-generics",level:2},{value:"Generic constraints",id:"generic-constraints",level:2},{value:"Variance",id:"variance",level:2}],v={toc:u};function m(e){let{components:n,...t}=e;return(0,r.kt)("wrapper",(0,a.Z)({},v,t,{components:n,mdxType:"MDXLayout"}),(0,r.kt)("h1",{id:"generics"},"Generics"),(0,r.kt)("p",null,"This section is about how Stashbox handles various usage scenarios that involve .NET Generic types. Including the registration of open-generic and closed-generic types, ",(0,r.kt)("a",{parentName:"p",href:"/docs/advanced/decorators#generic-decorators"},"generic decorators"),", conditions based on generic constraints, and variance."),(0,r.kt)("h2",{id:"closed-generics"},"Closed-generics"),(0,r.kt)(l.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("p",null,"The registration of a closed-generic type does not differ from registering a simple non-generic service."),(0,r.kt)("p",null,"You have all options available that you saw at the ",(0,r.kt)("a",{parentName:"p",href:"/docs/guides/basics"},"basic")," and ",(0,r.kt)("a",{parentName:"p",href:"/docs/guides/advanced-registration"},"advanced registration")," flows.")),(0,r.kt)("div",null,(0,r.kt)(i.Z,{groupId:"generic-runtime-apis",mdxType:"Tabs"},(0,r.kt)(o.Z,{value:"Generic API",label:"Generic API",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register, UserValidator>();\nIValidator validator = container.Resolve>();\n"))),(0,r.kt)(o.Z,{value:"Runtime type API",label:"Runtime type API",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(typeof(IValidator), typeof(UserValidator));\nobject validator = container.Resolve(typeof(IValidator));\n")))))),(0,r.kt)("h2",{id:"open-generics"},"Open-generics"),(0,r.kt)("p",null,"The registration of an open-generic type differs from registering a closed-generic one as C# doesn't allow the usage of open-generic types in generic method parameters. We have to get a runtime type from the open-generic type first with ",(0,r.kt)("inlineCode",{parentName:"p"},"typeof()"),"."),(0,r.kt)(l.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("p",null,"Open-generic types could help in such scenarios where you have generic interface-implementation pairs with numerous generic parameter variations. The registration of those different versions would look like this: ")),(0,r.kt)("div",null,(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register, Validator>();\ncontainer.Register, Validator>();\ncontainer.Register, Validator>();\n// and so on...\n")))),(0,r.kt)(l.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("p",null,"Rather than doing that, you can register your type's generic definition and let Stashbox bind the type parameters for you. When a matching closed ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type")," is requested, the container will construct an equivalent closed-generic implementation.")),(0,r.kt)("div",null,(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(typeof(IValidator<>), typeof(Validator<>));\n// Validator will be returned.\nIValidator userValidator = container.Resolve>();\n// Validator will be returned.\nIValidator roleValidator = container.Resolve>();\n")))),(0,r.kt)(l.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("p",null,"A registered closed-generic type always has priority over an open-generic type at service selection.")),(0,r.kt)("div",null,(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register, UserValidator>();\ncontainer.Register(typeof(IValidator<>), typeof(Validator<>));\n// UserValidator will be returned.\nIValidator validator = container.Resolve>();\n")))),(0,r.kt)("h2",{id:"generic-constraints"},"Generic constraints"),(0,r.kt)("p",null,"In the following examples, you can see how the container handles generic constraints during service resolution. Constraints can be used for ",(0,r.kt)("a",{parentName:"p",href:"/docs/guides/service-resolution#conditional-resolution"},"conditional resolution")," including collection filters. "),(0,r.kt)(i.Z,{mdxType:"Tabs"},(0,r.kt)(o.Z,{value:"Conditional resolution",label:"Conditional resolution",mdxType:"TabItem"},(0,r.kt)("p",null,"The container chooses ",(0,r.kt)("inlineCode",{parentName:"p"},"UpdatedEventHandler")," because it is the only one that has a constraint satisfied by the requested ",(0,r.kt)("inlineCode",{parentName:"p"},"UserUpdatedEvent")," generic parameter as it's implementing ",(0,r.kt)("inlineCode",{parentName:"p"},"IUpdatedEvent"),"."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"interface IEventHandler { }\n\n// event interfaces\ninterface IUpdatedEvent { }\ninterface ICreatedEvent { }\n\n// event handlers\nclass UpdatedEventHandler : IEventHandler where TEvent : IUpdatedEvent { }\nclass CreatedEventHandler : IEventHandler where TEvent : ICreatedEvent { }\n\n// event implementation\nclass UserUpdatedEvent : IUpdatedEvent { }\n\nusing var container = new StashboxContainer();\n\ncontainer.RegisterTypesAs(typeof(IEventHandler<>), new[] \n { \n typeof(UpdateEventHandler<>), \n typeof(CreateEventHandler<>) \n });\n\n// eventHandler will be UpdatedEventHandler\nIEventHandler eventHandler = container.Resolve>();\n"))),(0,r.kt)(o.Z,{value:"Collection filter",label:"Collection filter",mdxType:"TabItem"},(0,r.kt)("p",null,"This example shows how the container is filtering out those services from the returned collection that does not satisfy the given generic constraint needed to create the closed generic type."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"interface IEventHandler { }\n\n// event interfaces\ninterface IUpdatedEvent { }\ninterface ICreatedEvent { }\n\n// event handlers\nclass UpdatedEventHandler : IEventHandler where TEvent : IUpdatedEvent { }\nclass CreatedEventHandler : IEventHandler where TEvent : ICreatedEvent { }\n\n// event implementation\nclass UserUpdatedEvent : IUpdatedEvent { }\n\nusing var container = new StashboxContainer();\n\ncontainer.RegisterTypesAs(typeof(IEventHandler<>), new[] \n { \n typeof(UpdateEventHandler<>), \n typeof(CreateEventHandler<>) \n });\n\n// eventHandlers will contain only UpdatedEventHandler\nIEnumerable> eventHandlers = container.ResolveAll>();\n")))),(0,r.kt)("h2",{id:"variance"},"Variance"),(0,r.kt)("p",null,"Since .NET Framework 4.0, C# supports ",(0,r.kt)("a",{parentName:"p",href:"https://docs.microsoft.com/en-us/dotnet/csharp/programming-guide/concepts/covariance-contravariance/"},"covariance and contravariance")," in generic interfaces and delegates and allows implicit conversion of generic type parameters. In this section, we'll focus on variance in generic interfaces. "),(0,r.kt)("p",null,(0,r.kt)("a",{parentName:"p",href:"https://docs.microsoft.com/en-us/dotnet/csharp/programming-guide/concepts/covariance-contravariance/creating-variant-generic-interfaces"},"Here")," you can read more about how to create variant generic interfaces, and the following example will show how you can use them with Stashbox."),(0,r.kt)(i.Z,{mdxType:"Tabs"},(0,r.kt)(o.Z,{value:"Contravariance",label:"Contravariance",mdxType:"TabItem"},(0,r.kt)("p",null,(0,r.kt)("strong",{parentName:"p"},"Contravariance")," only allows argument types that are less derived than that defined by the generic parameters. You can declare a generic type parameter contravariant by using the ",(0,r.kt)("inlineCode",{parentName:"p"},"in")," keyword."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"// contravariant generic event handler interface\ninterface IEventHandler { } \n\n// event interfaces\ninterface IGeneralEvent { }\ninterface IUpdatedEvent : IGeneralEvent { }\n\n// event handlers\nclass GeneralEventHandler : IEventHandler { }\nclass UpdatedEventHandler : IEventHandler { }\n\ncontainer.Register, GeneralEventHandler>();\ncontainer.Register, UpdatedEventHandler>();\n\n// eventHandlers contain both GeneralEventHandler and UpdatedEventHandler\nIEnumerable> eventHandlers = container.ResolveAll>();\n")),(0,r.kt)("p",null,"Despite the fact that only ",(0,r.kt)("inlineCode",{parentName:"p"},"IEventHandler")," implementations were requested, the result contains both ",(0,r.kt)("inlineCode",{parentName:"p"},"GeneralEventHandler")," and ",(0,r.kt)("inlineCode",{parentName:"p"},"UpdatedEventHandler"),". As ",(0,r.kt)("inlineCode",{parentName:"p"},"TEvent")," is declared ",(0,r.kt)("strong",{parentName:"p"},"contravariant")," with the ",(0,r.kt)("inlineCode",{parentName:"p"},"in")," keyword, and ",(0,r.kt)("inlineCode",{parentName:"p"},"IGeneralEvent")," is less derived than ",(0,r.kt)("inlineCode",{parentName:"p"},"IUpdatedEvent"),", ",(0,r.kt)("inlineCode",{parentName:"p"},"IEventHandler")," implementations can be part of ",(0,r.kt)("inlineCode",{parentName:"p"},"IEventHandler")," collections."),(0,r.kt)("p",null,"If we request ",(0,r.kt)("inlineCode",{parentName:"p"},"IEventHandler"),", only ",(0,r.kt)("inlineCode",{parentName:"p"},"GeneralEventHandler")," would be returned, because ",(0,r.kt)("inlineCode",{parentName:"p"},"IUpdatedEvent")," is more derived, so ",(0,r.kt)("inlineCode",{parentName:"p"},"IEventHandler")," implementations are not fit into ",(0,r.kt)("inlineCode",{parentName:"p"},"IEventHandler")," collections. ")),(0,r.kt)(o.Z,{value:"Covariance",label:"Covariance",mdxType:"TabItem"},(0,r.kt)("p",null,(0,r.kt)("strong",{parentName:"p"},"Covariance")," only allows argument types that are more derived than that defined by the generic parameters. You can declare a generic type parameter covariant by using the ",(0,r.kt)("inlineCode",{parentName:"p"},"out")," keyword."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"// covariant generic event handler interface\ninterface IEventHandler { } \n\n// event interfaces\ninterface IGeneralEvent { }\ninterface IUpdatedEvent : IGeneralEvent { }\n\n// event handlers\nclass GeneralEventHandler : IEventHandler { }\nclass UpdatedEventHandler : IEventHandler { }\n\ncontainer.Register, GeneralEventHandler>();\ncontainer.Register, UpdatedEventHandler>();\n\n// eventHandlers contain both GeneralEventHandler and UpdatedEventHandler\nIEnumerable> eventHandlers = container.ResolveAll>();\n")),(0,r.kt)("p",null,"Despite the fact that only ",(0,r.kt)("inlineCode",{parentName:"p"},"IEventHandler")," implementations were requested, the result contains both ",(0,r.kt)("inlineCode",{parentName:"p"},"GeneralEventHandler")," and ",(0,r.kt)("inlineCode",{parentName:"p"},"UpdatedEventHandler"),". As ",(0,r.kt)("inlineCode",{parentName:"p"},"TEvent")," is declared ",(0,r.kt)("strong",{parentName:"p"},"covariant")," with the ",(0,r.kt)("inlineCode",{parentName:"p"},"out")," keyword, and ",(0,r.kt)("inlineCode",{parentName:"p"},"IUpdatedEvent")," is more derived than ",(0,r.kt)("inlineCode",{parentName:"p"},"IGeneralEvent"),", ",(0,r.kt)("inlineCode",{parentName:"p"},"IEventHandler")," implementations can be part of ",(0,r.kt)("inlineCode",{parentName:"p"},"IEventHandler")," collections."),(0,r.kt)("p",null,"If we request ",(0,r.kt)("inlineCode",{parentName:"p"},"IEventHandler"),", only ",(0,r.kt)("inlineCode",{parentName:"p"},"UpdatedEventHandler")," would be returned, because ",(0,r.kt)("inlineCode",{parentName:"p"},"IGeneralEvent")," is less derived, so ",(0,r.kt)("inlineCode",{parentName:"p"},"IEventHandler")," implementations are not fit into ",(0,r.kt)("inlineCode",{parentName:"p"},"IEventHandler")," collections."))))}m.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/74fdf922.deb27988.js b/assets/js/74fdf922.deb27988.js deleted file mode 100644 index ea9bcdcf..00000000 --- a/assets/js/74fdf922.deb27988.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkwebsite=self.webpackChunkwebsite||[]).push([[783],{3905:(e,n,t)=>{t.d(n,{Zo:()=>p,kt:()=>b});var a=t(7294);function s(e,n,t){return n in e?Object.defineProperty(e,n,{value:t,enumerable:!0,configurable:!0,writable:!0}):e[n]=t,e}function o(e,n){var t=Object.keys(e);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);n&&(a=a.filter((function(n){return Object.getOwnPropertyDescriptor(e,n).enumerable}))),t.push.apply(t,a)}return t}function i(e){for(var n=1;n=0||(s[t]=e[t]);return s}(e,n);if(Object.getOwnPropertySymbols){var o=Object.getOwnPropertySymbols(e);for(a=0;a=0||Object.prototype.propertyIsEnumerable.call(e,t)&&(s[t]=e[t])}return s}var l=a.createContext({}),c=function(e){var n=a.useContext(l),t=n;return e&&(t="function"==typeof e?e(n):i(i({},n),e)),t},p=function(e){var n=c(e.components);return a.createElement(l.Provider,{value:n},e.children)},u="mdxType",d={inlineCode:"code",wrapper:function(e){var n=e.children;return a.createElement(a.Fragment,{},n)}},m=a.forwardRef((function(e,n){var t=e.components,s=e.mdxType,o=e.originalType,l=e.parentName,p=r(e,["components","mdxType","originalType","parentName"]),u=c(t),m=s,b=u["".concat(l,".").concat(m)]||u[m]||d[m]||o;return t?a.createElement(b,i(i({ref:n},p),{},{components:t})):a.createElement(b,i({ref:n},p))}));function b(e,n){var t=arguments,s=n&&n.mdxType;if("string"==typeof e||s){var o=t.length,i=new Array(o);i[0]=m;var r={};for(var l in n)hasOwnProperty.call(n,l)&&(r[l]=n[l]);r.originalType=e,r[u]="string"==typeof e?e:s,i[1]=r;for(var c=2;c{t.d(n,{Z:()=>i});var a=t(7294),s=t(6010);const o="tabItem_Ymn6";function i(e){let{children:n,hidden:t,className:i}=e;return a.createElement("div",{role:"tabpanel",className:(0,s.Z)(o,i),hidden:t},n)}},4866:(e,n,t)=>{t.d(n,{Z:()=>I});var a=t(7462),s=t(7294),o=t(6010),i=t(2466),r=t(6550),l=t(1980),c=t(7392),p=t(12);function u(e){return function(e){return s.Children.map(e,(e=>{if(!e||(0,s.isValidElement)(e)&&function(e){const{props:n}=e;return!!n&&"object"==typeof n&&"value"in n}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}(e).map((e=>{let{props:{value:n,label:t,attributes:a,default:s}}=e;return{value:n,label:t,attributes:a,default:s}}))}function d(e){const{values:n,children:t}=e;return(0,s.useMemo)((()=>{const e=n??u(t);return function(e){const n=(0,c.l)(e,((e,n)=>e.value===n.value));if(n.length>0)throw new Error(`Docusaurus error: Duplicate values "${n.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[n,t])}function m(e){let{value:n,tabValues:t}=e;return t.some((e=>e.value===n))}function b(e){let{queryString:n=!1,groupId:t}=e;const a=(0,r.k6)(),o=function(e){let{queryString:n=!1,groupId:t}=e;if("string"==typeof n)return n;if(!1===n)return null;if(!0===n&&!t)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return t??null}({queryString:n,groupId:t});return[(0,l._X)(o),(0,s.useCallback)((e=>{if(!o)return;const n=new URLSearchParams(a.location.search);n.set(o,e),a.replace({...a.location,search:n.toString()})}),[o,a])]}function g(e){const{defaultValue:n,queryString:t=!1,groupId:a}=e,o=d(e),[i,r]=(0,s.useState)((()=>function(e){let{defaultValue:n,tabValues:t}=e;if(0===t.length)throw new Error("Docusaurus error: the component requires at least one children component");if(n){if(!m({value:n,tabValues:t}))throw new Error(`Docusaurus error: The has a defaultValue "${n}" but none of its children has the corresponding value. Available values are: ${t.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return n}const a=t.find((e=>e.default))??t[0];if(!a)throw new Error("Unexpected error: 0 tabValues");return a.value}({defaultValue:n,tabValues:o}))),[l,c]=b({queryString:t,groupId:a}),[u,g]=function(e){let{groupId:n}=e;const t=function(e){return e?`docusaurus.tab.${e}`:null}(n),[a,o]=(0,p.Nk)(t);return[a,(0,s.useCallback)((e=>{t&&o.set(e)}),[t,o])]}({groupId:a}),h=(()=>{const e=l??u;return m({value:e,tabValues:o})?e:null})();(0,s.useLayoutEffect)((()=>{h&&r(h)}),[h]);return{selectedValue:i,selectValue:(0,s.useCallback)((e=>{if(!m({value:e,tabValues:o}))throw new Error(`Can't select invalid tab value=${e}`);r(e),c(e),g(e)}),[c,g,o]),tabValues:o}}var h=t(2389);const k="tabList__CuJ",v="tabItem_LNqP";function f(e){let{className:n,block:t,selectedValue:r,selectValue:l,tabValues:c}=e;const p=[],{blockElementScrollPositionUntilNextRender:u}=(0,i.o5)(),d=e=>{const n=e.currentTarget,t=p.indexOf(n),a=c[t].value;a!==r&&(u(n),l(a))},m=e=>{let n=null;switch(e.key){case"Enter":d(e);break;case"ArrowRight":{const t=p.indexOf(e.currentTarget)+1;n=p[t]??p[0];break}case"ArrowLeft":{const t=p.indexOf(e.currentTarget)-1;n=p[t]??p[p.length-1];break}}n?.focus()};return s.createElement("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,o.Z)("tabs",{"tabs--block":t},n)},c.map((e=>{let{value:n,label:t,attributes:i}=e;return s.createElement("li",(0,a.Z)({role:"tab",tabIndex:r===n?0:-1,"aria-selected":r===n,key:n,ref:e=>p.push(e),onKeyDown:m,onClick:d},i,{className:(0,o.Z)("tabs__item",v,i?.className,{"tabs__item--active":r===n})}),t??n)})))}function y(e){let{lazy:n,children:t,selectedValue:a}=e;const o=(Array.isArray(t)?t:[t]).filter(Boolean);if(n){const e=o.find((e=>e.props.value===a));return e?(0,s.cloneElement)(e,{className:"margin-top--md"}):null}return s.createElement("div",{className:"margin-top--md"},o.map(((e,n)=>(0,s.cloneElement)(e,{key:n,hidden:e.props.value!==a}))))}function w(e){const n=g(e);return s.createElement("div",{className:(0,o.Z)("tabs-container",k)},s.createElement(f,(0,a.Z)({},e,n)),s.createElement(y,(0,a.Z)({},e,n)))}function I(e){const n=(0,h.Z)();return s.createElement(w,(0,a.Z)({key:String(n)},e))}},8846:(e,n,t)=>{t.d(n,{Z:()=>r});var a=t(7294);const s="codeDescContainer_ie8f",o="desc_jyqI",i="example_eYlF";function r(e){let{children:n}=e,t=a.Children.toArray(n).filter((e=>e));return a.createElement("div",{className:s},a.createElement("div",{className:o},t[0]),a.createElement("div",{className:i},t[1]))}},3422:(e,n,t)=>{t.r(n),t.d(n,{assets:()=>u,contentTitle:()=>c,default:()=>b,frontMatter:()=>l,metadata:()=>p,toc:()=>d});var a=t(7462),s=(t(7294),t(3905)),o=t(8846),i=t(4866),r=t(5162);const l={},c="Scopes",p={unversionedId:"guides/scopes",id:"guides/scopes",title:"Scopes",description:"A scope is Stashbox's implementation of the unit-of-work pattern; it encapsulates a given unit used to resolve and store instances required for a given work. When a scoped service is resolved or injected, the scope ensures that it gets instantiated only once within the scope's lifetime. When the work is finished, the scope cleans up the resources by disposing each tracked disposable instance.",source:"@site/docs/guides/scopes.md",sourceDirName:"guides",slug:"/guides/scopes",permalink:"/stashbox/docs/guides/scopes",draft:!1,editUrl:"https://github.com/z4kn4fein/stashbox/edit/master/docs/docs/guides/scopes.md",tags:[],version:"current",lastUpdatedBy:"dependabot[bot]",lastUpdatedAt:1697644420,formattedLastUpdatedAt:"Oct 18, 2023",frontMatter:{},sidebar:"docs",previous:{title:"Lifetimes",permalink:"/stashbox/docs/guides/lifetimes"},next:{title:"Registration configuration",permalink:"/stashbox/docs/configuration/registration-configuration"}},u={},d=[{value:"Creating a scope",id:"creating-a-scope",level:2},{value:"Named scopes",id:"named-scopes",level:2},{value:"Service as scope",id:"service-as-scope",level:2},{value:"Put instance to a scope",id:"put-instance-to-a-scope",level:2},{value:"Disposal",id:"disposal",level:2},{value:"Async disposal",id:"async-disposal",level:3},{value:"Finalizer delegate",id:"finalizer-delegate",level:3}],m={toc:d};function b(e){let{components:n,...t}=e;return(0,s.kt)("wrapper",(0,a.Z)({},m,t,{components:n,mdxType:"MDXLayout"}),(0,s.kt)("h1",{id:"scopes"},"Scopes"),(0,s.kt)("p",null,"A scope is Stashbox's implementation of the unit-of-work pattern; it encapsulates a given unit used to resolve and store instances required for a given work. When a scoped service is resolved or injected, the scope ensures that it gets instantiated only once within the scope's lifetime. When the work is finished, the scope cleans up the resources by disposing each tracked disposable instance."),(0,s.kt)("p",null,"A web application is a fair usage example for scopes as it has a well-defined execution unit that can be bound to a scope - the HTTP request. Every request could have its unique scope attached to the request's lifetime. When a request ends, the scope gets closed, and all the scoped instances will be disposed."),(0,s.kt)("h2",{id:"creating-a-scope"},"Creating a scope"),(0,s.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,s.kt)("div",null,(0,s.kt)("p",null,"You can create a scope from the container by calling its ",(0,s.kt)("inlineCode",{parentName:"p"},".BeginScope()")," method."),(0,s.kt)("p",null,"Scopes can be ",(0,s.kt)("strong",{parentName:"p"},"nested"),", which means you can create sub-scopes from existing ones with their ",(0,s.kt)("inlineCode",{parentName:"p"},".BeginScope()")," method. "),(0,s.kt)("p",null,"Scoped service instances are not shared across parent and sub-scope relations."),(0,s.kt)("p",null,"Nested scopes can be ",(0,s.kt)("strong",{parentName:"p"},"attached to their parent's lifetime"),", which means when a parent gets disposed all child scopes attached to it will be disposed."),(0,s.kt)("p",null,"Scopes are ",(0,s.kt)("inlineCode",{parentName:"p"},"IDisposable"),"; they track all ",(0,s.kt)("inlineCode",{parentName:"p"},"IDisposable")," instances they resolved. Calling their ",(0,s.kt)("inlineCode",{parentName:"p"},"Dispose()")," method or wrapping them in ",(0,s.kt)("inlineCode",{parentName:"p"},"using")," statements is a crucial part of their service's lifetime management.")),(0,s.kt)("div",null,(0,s.kt)(i.Z,{mdxType:"Tabs"},(0,s.kt)(r.Z,{value:"Create",label:"Create",mdxType:"TabItem"},(0,s.kt)("pre",null,(0,s.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterScoped();\n\n// create the scope with using so it'll be auto disposed.\nusing (var scope = container.BeginScope())\n{\n IJob job = scope.Resolve();\n IJob jobAgain = scope.Resolve();\n // job and jobAgain are created in the \n // same scope, so they are the same instance.\n}\n"))),(0,s.kt)(r.Z,{value:"Nested",label:"Nested",mdxType:"TabItem"},(0,s.kt)("pre",null,(0,s.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterScoped();\n\nusing (var parent = container.BeginScope())\n{\n IJob job = parent.Resolve();\n IJob jobAgain = parent.Resolve();\n // job and jobAgain are created in the \n // same scope, so they are the same instance.\n\n // create a sub-scope.\n using var sub = parent.BeginScope();\n\n IJob subJob = sub.Resolve();\n // subJob is a new instance created in the sub-scope, \n // differs from either job and jobAgain.\n}\n"))),(0,s.kt)(r.Z,{value:"Nested attached",label:"Nested attached",mdxType:"TabItem"},(0,s.kt)("pre",null,(0,s.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterScoped();\n\nvar parent = container.BeginScope();\nvar sub = parent.BeginScope(attachToParent: true);\n\n// sub will also be disposed with the scope.\nscope.Dispose(); \n")))))),(0,s.kt)("h2",{id:"named-scopes"},"Named scopes"),(0,s.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,s.kt)("div",null,(0,s.kt)("p",null,"There might be cases where you don't want to use a service globally across every scope, only in specific ones. "),(0,s.kt)("p",null,"For this reason, you can differentiate specific scope groups from other scopes with a ",(0,s.kt)("strong",{parentName:"p"},"name"),"."),(0,s.kt)("p",null,"You can set a service's lifetime to ",(0,s.kt)("a",{parentName:"p",href:"/docs/guides/lifetimes#named-scope-lifetime"},"named scope lifetime")," initialized with the ",(0,s.kt)("strong",{parentName:"p"},"scope's name")," to mark it usable only for that named scope."),(0,s.kt)("pre",null,(0,s.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register(options => \n options.InNamedScope("DbScope"));\n\ncontainer.Register(options => \n options.InNamedScope("DbScope"));\n\ncontainer.Register(options => \n options.InNamedScope("DbSubScope"));\n\ncontainer.Register(options => \n options.InNamedScope("StorageScope"));\n')),(0,s.kt)("admonition",{type:"note"},(0,s.kt)("p",{parentName:"admonition"},"Services with named scope lifetime are ",(0,s.kt)("strong",{parentName:"p"},"shared across parent and sub-scope relations"),".")),(0,s.kt)("p",null,"If you request a name-scoped service from an un-named scope, you'll get an error or no result (depending on the configuration) because those services are selectable only by named scopes with a matching name.")),(0,s.kt)("div",null,(0,s.kt)("pre",null,(0,s.kt)("code",{parentName:"pre",className:"language-cs"},'using (var dbScope = container.BeginScope("DbScope"))\n{ \n // DbBackup and DbCleanup will be returned.\n IEnumerable jobs = dbScope.ResolveAll();\n\n // create a sub-scope of dbScope.\n using var sub = dbScope.BeginScope();\n\n // DbBackup and DbCleanup will be returned from the named parent scope.\n IEnumerable jobs = sub.ResolveAll();\n\n // create a named sub-scope.\n using var namedSub = dbScope.BeginScope("DbSubScope");\n // DbIndexRebuild will be returned from the named sub-scope.\n IEnumerable jobs = namedSub.ResolveAll();\n}\n\nusing (var storageScope = container.BeginScope("StorageScope"))\n{\n // StorageCleanup will be returned.\n IJob job = storageScope.Resolve();\n}\n\n// create a common scope without a name.\nusing (var unNamed = container.BeginScope())\n{\n // empty result as there\'s no service registered without named scope.\n IEnumerable jobs = unNamed.ResolveAll();\n\n // throws an exception because there\'s no unnamed service registered.\n IJob job = unNamed.Resolve();\n}\n')))),(0,s.kt)("h2",{id:"service-as-scope"},"Service as scope"),(0,s.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,s.kt)("div",null,(0,s.kt)("p",null,"You can configure a service to behave like a nested named scope. At the resolution of this kind of service, a new dedicated named scope is created implicitly for managing the service's dependencies. "),(0,s.kt)("p",null,"With this feature, you can organize your dependencies around logical groups (named scopes) instead of individual services."),(0,s.kt)("p",null,"Using ",(0,s.kt)("inlineCode",{parentName:"p"},"InScopeDefinedBy()"),", you can bind services to a defined scope without giving it a name. In this case, the defining service's ",(0,s.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"implementation type")," is used for naming the scope."),(0,s.kt)("admonition",{type:"note"},(0,s.kt)("p",{parentName:"admonition"},"The lifetime of the defined scope is attached to the current scope that was used to create the service."))),(0,s.kt)("div",null,(0,s.kt)(i.Z,{mdxType:"Tabs"},(0,s.kt)(r.Z,{value:"Define named",label:"Define named",mdxType:"TabItem"},(0,s.kt)("pre",null,(0,s.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register(options => options\n .DefinesScope("DbBackupScope"));\ncontainer.Register(options => options\n .InNamedScope("DbBackupScope"));\ncontainer.Register();\n\nvar scope = container.BeginScope();\n\n// DbBackup will create a named scope with the name "DbBackupScope".\n// the named scope will select ConsoleLogger as it\'s \n// bound to the named scope\'s identifier.\nIJob job = scope.Resolve();\n\n// this will dispose the implicitly created named scope by DbBackup.\nscope.Dispose(); \n'))),(0,s.kt)(r.Z,{value:"Define typed",label:"Define typed",mdxType:"TabItem"},(0,s.kt)("pre",null,(0,s.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .DefinesScope());\ncontainer.Register(options => options\n .InScopeDefinedBy());\ncontainer.Register();\n\nvar scope = container.BeginScope();\n\n// DbBackup will create a named scope with the name typeof(DbBackup).\n// the named scope will select ConsoleLogger as it's \n// bound to the named scope's identifier.\nIJob job = scope.Resolve();\n\n// this will dispose the implicitly created named scope by DbBackup.\nscope.Dispose(); \n")))))),(0,s.kt)("h2",{id:"put-instance-to-a-scope"},"Put instance to a scope"),(0,s.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,s.kt)("div",null,(0,s.kt)("p",null,"You can add an already instantiated service to a scope. The instance's lifetime will be tracked by the given scope.")),(0,s.kt)("div",null,(0,s.kt)("pre",null,(0,s.kt)("code",{parentName:"pre",className:"language-cs"},"using var scope = container.BeginScope();\nscope.PutInstanceInScope(new DbBackup());\n")))),(0,s.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,s.kt)("div",null,(0,s.kt)("p",null,"You can disable the tracking by passing ",(0,s.kt)("inlineCode",{parentName:"p"},"true")," for the ",(0,s.kt)("inlineCode",{parentName:"p"},"withoutDisposalTracking")," parameter. In this case, only the strong reference to the instance is dropped when the scope is disposed.")),(0,s.kt)("div",null,(0,s.kt)("pre",null,(0,s.kt)("code",{parentName:"pre",className:"language-cs"},"using var scope = container.BeginScope();\nscope.PutInstanceInScope(new DbBackup(), withoutDisposalTracking: true);\n")))),(0,s.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,s.kt)("div",null,(0,s.kt)("p",null,"You can also give your instance a name to use it like a ",(0,s.kt)("a",{parentName:"p",href:"/docs/guides/basics#named-registration"},"named registration"),":")),(0,s.kt)("div",null,(0,s.kt)("pre",null,(0,s.kt)("code",{parentName:"pre",className:"language-cs"},'using var scope = container.BeginScope();\nscope.PutInstanceInScope(new DbBackup(), false, name: "DbBackup");\n')))),(0,s.kt)("admonition",{type:"note"},(0,s.kt)("p",{parentName:"admonition"},"Instances put to a scope will take precedence over existing registrations with the same ",(0,s.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type"),".")),(0,s.kt)("h2",{id:"disposal"},"Disposal"),(0,s.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,s.kt)("div",null,(0,s.kt)("p",null,"The currently resolving scope tracks services that implement either ",(0,s.kt)("inlineCode",{parentName:"p"},"IDisposable")," or ",(0,s.kt)("inlineCode",{parentName:"p"},"IAsyncDisposable"),". This means that when the scope is disposed, all the tracked disposable instances will be disposed with it."),(0,s.kt)("admonition",{type:"note"},(0,s.kt)("p",{parentName:"admonition"},"Disposing the container will dispose all the singleton instances and their dependencies."))),(0,s.kt)("div",null,(0,s.kt)(i.Z,{groupId:"lifetime-dispose",mdxType:"Tabs"},(0,s.kt)(r.Z,{value:"Using",label:"Using",mdxType:"TabItem"},(0,s.kt)("pre",null,(0,s.kt)("code",{parentName:"pre",className:"language-cs"},"using (var scope = container.BeginScope())\n{\n var disposable = scope.Resolve();\n} // 'disposable' will be disposed when \n // the using statement ends.\n"))),(0,s.kt)(r.Z,{value:"Dispose",label:"Dispose",mdxType:"TabItem"},(0,s.kt)("pre",null,(0,s.kt)("code",{parentName:"pre",className:"language-cs"},"var scope = container.BeginScope();\nvar disposable = scope.Resolve();\n\n// 'disposable' will be disposed with the scope.\nscope.Dispose();\n")))))),(0,s.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,s.kt)("div",null,(0,s.kt)("p",null,"You can disable the disposal tracking on a ",(0,s.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-registration--registered-service"},"service registration")," with the ",(0,s.kt)("inlineCode",{parentName:"p"},".WithoutDisposalTracking()")," option.")),(0,s.kt)("div",null,(0,s.kt)("pre",null,(0,s.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => \n options.WithoutDisposalTracking());\n")))),(0,s.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,s.kt)("div",null,(0,s.kt)("h3",{id:"async-disposal"},"Async disposal"),(0,s.kt)("p",null,"As the container and its scopes implement the ",(0,s.kt)("inlineCode",{parentName:"p"},"IAsyncDisposable")," interface, you can dispose them asynchronously when they are used in an ",(0,s.kt)("inlineCode",{parentName:"p"},"async")," context."),(0,s.kt)("p",null,"Calling ",(0,s.kt)("inlineCode",{parentName:"p"},"DisposeAsync")," disposes both ",(0,s.kt)("inlineCode",{parentName:"p"},"IDisposable")," and ",(0,s.kt)("inlineCode",{parentName:"p"},"IAsyncDisposable")," instances; however, calling ",(0,s.kt)("inlineCode",{parentName:"p"},"Dispose")," only disposes ",(0,s.kt)("inlineCode",{parentName:"p"},"IDisposable")," instances.")),(0,s.kt)("div",null,(0,s.kt)(i.Z,{groupId:"lifetime-dispose",mdxType:"Tabs"},(0,s.kt)(r.Z,{value:"Using",label:"Using",mdxType:"TabItem"},(0,s.kt)("pre",null,(0,s.kt)("code",{parentName:"pre",className:"language-cs"},"await using (var scope = container.BeginScope())\n{\n var disposable = scope.Resolve();\n} // 'disposable' will be disposed asynchronously \n // when the using statement ends.\n"))),(0,s.kt)(r.Z,{value:"Dispose",label:"Dispose",mdxType:"TabItem"},(0,s.kt)("pre",null,(0,s.kt)("code",{parentName:"pre",className:"language-cs"},"var scope = container.BeginScope();\nvar disposable = scope.Resolve();\n\n// 'disposable' will be disposed asynchronously with the scope.\nawait scope.DisposeAsync();\n")))))),(0,s.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,s.kt)("div",null,(0,s.kt)("h3",{id:"finalizer-delegate"},"Finalizer delegate"),(0,s.kt)("p",null,"During ",(0,s.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-registration--registered-service"},"service registration"),", you can set a custom finalizer delegate that will be invoked at the service's disposal.")),(0,s.kt)("div",null,(0,s.kt)("pre",null,(0,s.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => \n options.WithFinalizer(backup => \n backup.CloseDbConnection()));\n")))))}b.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/74fdf922.e9d5d220.js b/assets/js/74fdf922.e9d5d220.js new file mode 100644 index 00000000..473b3738 --- /dev/null +++ b/assets/js/74fdf922.e9d5d220.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkwebsite=self.webpackChunkwebsite||[]).push([[783],{3905:(e,n,t)=>{t.d(n,{Zo:()=>p,kt:()=>b});var a=t(7294);function s(e,n,t){return n in e?Object.defineProperty(e,n,{value:t,enumerable:!0,configurable:!0,writable:!0}):e[n]=t,e}function o(e,n){var t=Object.keys(e);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);n&&(a=a.filter((function(n){return Object.getOwnPropertyDescriptor(e,n).enumerable}))),t.push.apply(t,a)}return t}function i(e){for(var n=1;n=0||(s[t]=e[t]);return s}(e,n);if(Object.getOwnPropertySymbols){var o=Object.getOwnPropertySymbols(e);for(a=0;a=0||Object.prototype.propertyIsEnumerable.call(e,t)&&(s[t]=e[t])}return s}var l=a.createContext({}),c=function(e){var n=a.useContext(l),t=n;return e&&(t="function"==typeof e?e(n):i(i({},n),e)),t},p=function(e){var n=c(e.components);return a.createElement(l.Provider,{value:n},e.children)},u="mdxType",d={inlineCode:"code",wrapper:function(e){var n=e.children;return a.createElement(a.Fragment,{},n)}},m=a.forwardRef((function(e,n){var t=e.components,s=e.mdxType,o=e.originalType,l=e.parentName,p=r(e,["components","mdxType","originalType","parentName"]),u=c(t),m=s,b=u["".concat(l,".").concat(m)]||u[m]||d[m]||o;return t?a.createElement(b,i(i({ref:n},p),{},{components:t})):a.createElement(b,i({ref:n},p))}));function b(e,n){var t=arguments,s=n&&n.mdxType;if("string"==typeof e||s){var o=t.length,i=new Array(o);i[0]=m;var r={};for(var l in n)hasOwnProperty.call(n,l)&&(r[l]=n[l]);r.originalType=e,r[u]="string"==typeof e?e:s,i[1]=r;for(var c=2;c{t.d(n,{Z:()=>i});var a=t(7294),s=t(6010);const o="tabItem_Ymn6";function i(e){let{children:n,hidden:t,className:i}=e;return a.createElement("div",{role:"tabpanel",className:(0,s.Z)(o,i),hidden:t},n)}},4866:(e,n,t)=>{t.d(n,{Z:()=>I});var a=t(7462),s=t(7294),o=t(6010),i=t(2466),r=t(6550),l=t(1980),c=t(7392),p=t(12);function u(e){return function(e){return s.Children.map(e,(e=>{if(!e||(0,s.isValidElement)(e)&&function(e){const{props:n}=e;return!!n&&"object"==typeof n&&"value"in n}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}(e).map((e=>{let{props:{value:n,label:t,attributes:a,default:s}}=e;return{value:n,label:t,attributes:a,default:s}}))}function d(e){const{values:n,children:t}=e;return(0,s.useMemo)((()=>{const e=n??u(t);return function(e){const n=(0,c.l)(e,((e,n)=>e.value===n.value));if(n.length>0)throw new Error(`Docusaurus error: Duplicate values "${n.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[n,t])}function m(e){let{value:n,tabValues:t}=e;return t.some((e=>e.value===n))}function b(e){let{queryString:n=!1,groupId:t}=e;const a=(0,r.k6)(),o=function(e){let{queryString:n=!1,groupId:t}=e;if("string"==typeof n)return n;if(!1===n)return null;if(!0===n&&!t)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return t??null}({queryString:n,groupId:t});return[(0,l._X)(o),(0,s.useCallback)((e=>{if(!o)return;const n=new URLSearchParams(a.location.search);n.set(o,e),a.replace({...a.location,search:n.toString()})}),[o,a])]}function g(e){const{defaultValue:n,queryString:t=!1,groupId:a}=e,o=d(e),[i,r]=(0,s.useState)((()=>function(e){let{defaultValue:n,tabValues:t}=e;if(0===t.length)throw new Error("Docusaurus error: the component requires at least one children component");if(n){if(!m({value:n,tabValues:t}))throw new Error(`Docusaurus error: The has a defaultValue "${n}" but none of its children has the corresponding value. Available values are: ${t.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return n}const a=t.find((e=>e.default))??t[0];if(!a)throw new Error("Unexpected error: 0 tabValues");return a.value}({defaultValue:n,tabValues:o}))),[l,c]=b({queryString:t,groupId:a}),[u,g]=function(e){let{groupId:n}=e;const t=function(e){return e?`docusaurus.tab.${e}`:null}(n),[a,o]=(0,p.Nk)(t);return[a,(0,s.useCallback)((e=>{t&&o.set(e)}),[t,o])]}({groupId:a}),h=(()=>{const e=l??u;return m({value:e,tabValues:o})?e:null})();(0,s.useLayoutEffect)((()=>{h&&r(h)}),[h]);return{selectedValue:i,selectValue:(0,s.useCallback)((e=>{if(!m({value:e,tabValues:o}))throw new Error(`Can't select invalid tab value=${e}`);r(e),c(e),g(e)}),[c,g,o]),tabValues:o}}var h=t(2389);const k="tabList__CuJ",v="tabItem_LNqP";function f(e){let{className:n,block:t,selectedValue:r,selectValue:l,tabValues:c}=e;const p=[],{blockElementScrollPositionUntilNextRender:u}=(0,i.o5)(),d=e=>{const n=e.currentTarget,t=p.indexOf(n),a=c[t].value;a!==r&&(u(n),l(a))},m=e=>{let n=null;switch(e.key){case"Enter":d(e);break;case"ArrowRight":{const t=p.indexOf(e.currentTarget)+1;n=p[t]??p[0];break}case"ArrowLeft":{const t=p.indexOf(e.currentTarget)-1;n=p[t]??p[p.length-1];break}}n?.focus()};return s.createElement("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,o.Z)("tabs",{"tabs--block":t},n)},c.map((e=>{let{value:n,label:t,attributes:i}=e;return s.createElement("li",(0,a.Z)({role:"tab",tabIndex:r===n?0:-1,"aria-selected":r===n,key:n,ref:e=>p.push(e),onKeyDown:m,onClick:d},i,{className:(0,o.Z)("tabs__item",v,i?.className,{"tabs__item--active":r===n})}),t??n)})))}function y(e){let{lazy:n,children:t,selectedValue:a}=e;const o=(Array.isArray(t)?t:[t]).filter(Boolean);if(n){const e=o.find((e=>e.props.value===a));return e?(0,s.cloneElement)(e,{className:"margin-top--md"}):null}return s.createElement("div",{className:"margin-top--md"},o.map(((e,n)=>(0,s.cloneElement)(e,{key:n,hidden:e.props.value!==a}))))}function w(e){const n=g(e);return s.createElement("div",{className:(0,o.Z)("tabs-container",k)},s.createElement(f,(0,a.Z)({},e,n)),s.createElement(y,(0,a.Z)({},e,n)))}function I(e){const n=(0,h.Z)();return s.createElement(w,(0,a.Z)({key:String(n)},e))}},8846:(e,n,t)=>{t.d(n,{Z:()=>r});var a=t(7294);const s="codeDescContainer_ie8f",o="desc_jyqI",i="example_eYlF";function r(e){let{children:n}=e,t=a.Children.toArray(n).filter((e=>e));return a.createElement("div",{className:s},a.createElement("div",{className:o},t[0]),a.createElement("div",{className:i},t[1]))}},3422:(e,n,t)=>{t.r(n),t.d(n,{assets:()=>u,contentTitle:()=>c,default:()=>b,frontMatter:()=>l,metadata:()=>p,toc:()=>d});var a=t(7462),s=(t(7294),t(3905)),o=t(8846),i=t(4866),r=t(5162);const l={},c="Scopes",p={unversionedId:"guides/scopes",id:"guides/scopes",title:"Scopes",description:"A scope is Stashbox's implementation of the unit-of-work pattern; it encapsulates a given unit used to resolve and store instances required for a given work. When a scoped service is resolved or injected, the scope ensures that it gets instantiated only once within the scope's lifetime. When the work is finished, the scope cleans up the resources by disposing each tracked disposable instance.",source:"@site/docs/guides/scopes.md",sourceDirName:"guides",slug:"/guides/scopes",permalink:"/stashbox/docs/guides/scopes",draft:!1,editUrl:"https://github.com/z4kn4fein/stashbox/edit/master/docs/docs/guides/scopes.md",tags:[],version:"current",lastUpdatedBy:"dependabot[bot]",lastUpdatedAt:1699538162,formattedLastUpdatedAt:"Nov 9, 2023",frontMatter:{},sidebar:"docs",previous:{title:"Lifetimes",permalink:"/stashbox/docs/guides/lifetimes"},next:{title:"Registration configuration",permalink:"/stashbox/docs/configuration/registration-configuration"}},u={},d=[{value:"Creating a scope",id:"creating-a-scope",level:2},{value:"Named scopes",id:"named-scopes",level:2},{value:"Service as scope",id:"service-as-scope",level:2},{value:"Put instance to a scope",id:"put-instance-to-a-scope",level:2},{value:"Disposal",id:"disposal",level:2},{value:"Async disposal",id:"async-disposal",level:3},{value:"Finalizer delegate",id:"finalizer-delegate",level:3}],m={toc:d};function b(e){let{components:n,...t}=e;return(0,s.kt)("wrapper",(0,a.Z)({},m,t,{components:n,mdxType:"MDXLayout"}),(0,s.kt)("h1",{id:"scopes"},"Scopes"),(0,s.kt)("p",null,"A scope is Stashbox's implementation of the unit-of-work pattern; it encapsulates a given unit used to resolve and store instances required for a given work. When a scoped service is resolved or injected, the scope ensures that it gets instantiated only once within the scope's lifetime. When the work is finished, the scope cleans up the resources by disposing each tracked disposable instance."),(0,s.kt)("p",null,"A web application is a fair usage example for scopes as it has a well-defined execution unit that can be bound to a scope - the HTTP request. Every request could have its unique scope attached to the request's lifetime. When a request ends, the scope gets closed, and all the scoped instances will be disposed."),(0,s.kt)("h2",{id:"creating-a-scope"},"Creating a scope"),(0,s.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,s.kt)("div",null,(0,s.kt)("p",null,"You can create a scope from the container by calling its ",(0,s.kt)("inlineCode",{parentName:"p"},".BeginScope()")," method."),(0,s.kt)("p",null,"Scopes can be ",(0,s.kt)("strong",{parentName:"p"},"nested"),", which means you can create sub-scopes from existing ones with their ",(0,s.kt)("inlineCode",{parentName:"p"},".BeginScope()")," method. "),(0,s.kt)("p",null,"Scoped service instances are not shared across parent and sub-scope relations."),(0,s.kt)("p",null,"Nested scopes can be ",(0,s.kt)("strong",{parentName:"p"},"attached to their parent's lifetime"),", which means when a parent gets disposed all child scopes attached to it will be disposed."),(0,s.kt)("p",null,"Scopes are ",(0,s.kt)("inlineCode",{parentName:"p"},"IDisposable"),"; they track all ",(0,s.kt)("inlineCode",{parentName:"p"},"IDisposable")," instances they resolved. Calling their ",(0,s.kt)("inlineCode",{parentName:"p"},"Dispose()")," method or wrapping them in ",(0,s.kt)("inlineCode",{parentName:"p"},"using")," statements is a crucial part of their service's lifetime management.")),(0,s.kt)("div",null,(0,s.kt)(i.Z,{mdxType:"Tabs"},(0,s.kt)(r.Z,{value:"Create",label:"Create",mdxType:"TabItem"},(0,s.kt)("pre",null,(0,s.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterScoped();\n\n// create the scope with using so it'll be auto disposed.\nusing (var scope = container.BeginScope())\n{\n IJob job = scope.Resolve();\n IJob jobAgain = scope.Resolve();\n // job and jobAgain are created in the \n // same scope, so they are the same instance.\n}\n"))),(0,s.kt)(r.Z,{value:"Nested",label:"Nested",mdxType:"TabItem"},(0,s.kt)("pre",null,(0,s.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterScoped();\n\nusing (var parent = container.BeginScope())\n{\n IJob job = parent.Resolve();\n IJob jobAgain = parent.Resolve();\n // job and jobAgain are created in the \n // same scope, so they are the same instance.\n\n // create a sub-scope.\n using var sub = parent.BeginScope();\n\n IJob subJob = sub.Resolve();\n // subJob is a new instance created in the sub-scope, \n // differs from either job and jobAgain.\n}\n"))),(0,s.kt)(r.Z,{value:"Nested attached",label:"Nested attached",mdxType:"TabItem"},(0,s.kt)("pre",null,(0,s.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterScoped();\n\nvar parent = container.BeginScope();\nvar sub = parent.BeginScope(attachToParent: true);\n\n// sub will also be disposed with the scope.\nscope.Dispose(); \n")))))),(0,s.kt)("h2",{id:"named-scopes"},"Named scopes"),(0,s.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,s.kt)("div",null,(0,s.kt)("p",null,"There might be cases where you don't want to use a service globally across every scope, only in specific ones. "),(0,s.kt)("p",null,"For this reason, you can differentiate specific scope groups from other scopes with a ",(0,s.kt)("strong",{parentName:"p"},"name"),"."),(0,s.kt)("p",null,"You can set a service's lifetime to ",(0,s.kt)("a",{parentName:"p",href:"/docs/guides/lifetimes#named-scope-lifetime"},"named scope lifetime")," initialized with the ",(0,s.kt)("strong",{parentName:"p"},"scope's name")," to mark it usable only for that named scope."),(0,s.kt)("pre",null,(0,s.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register(options => \n options.InNamedScope("DbScope"));\n\ncontainer.Register(options => \n options.InNamedScope("DbScope"));\n\ncontainer.Register(options => \n options.InNamedScope("DbSubScope"));\n\ncontainer.Register(options => \n options.InNamedScope("StorageScope"));\n')),(0,s.kt)("admonition",{type:"note"},(0,s.kt)("p",{parentName:"admonition"},"Services with named scope lifetime are ",(0,s.kt)("strong",{parentName:"p"},"shared across parent and sub-scope relations"),".")),(0,s.kt)("p",null,"If you request a name-scoped service from an un-named scope, you'll get an error or no result (depending on the configuration) because those services are selectable only by named scopes with a matching name.")),(0,s.kt)("div",null,(0,s.kt)("pre",null,(0,s.kt)("code",{parentName:"pre",className:"language-cs"},'using (var dbScope = container.BeginScope("DbScope"))\n{ \n // DbBackup and DbCleanup will be returned.\n IEnumerable jobs = dbScope.ResolveAll();\n\n // create a sub-scope of dbScope.\n using var sub = dbScope.BeginScope();\n\n // DbBackup and DbCleanup will be returned from the named parent scope.\n IEnumerable jobs = sub.ResolveAll();\n\n // create a named sub-scope.\n using var namedSub = dbScope.BeginScope("DbSubScope");\n // DbIndexRebuild will be returned from the named sub-scope.\n IEnumerable jobs = namedSub.ResolveAll();\n}\n\nusing (var storageScope = container.BeginScope("StorageScope"))\n{\n // StorageCleanup will be returned.\n IJob job = storageScope.Resolve();\n}\n\n// create a common scope without a name.\nusing (var unNamed = container.BeginScope())\n{\n // empty result as there\'s no service registered without named scope.\n IEnumerable jobs = unNamed.ResolveAll();\n\n // throws an exception because there\'s no unnamed service registered.\n IJob job = unNamed.Resolve();\n}\n')))),(0,s.kt)("h2",{id:"service-as-scope"},"Service as scope"),(0,s.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,s.kt)("div",null,(0,s.kt)("p",null,"You can configure a service to behave like a nested named scope. At the resolution of this kind of service, a new dedicated named scope is created implicitly for managing the service's dependencies. "),(0,s.kt)("p",null,"With this feature, you can organize your dependencies around logical groups (named scopes) instead of individual services."),(0,s.kt)("p",null,"Using ",(0,s.kt)("inlineCode",{parentName:"p"},"InScopeDefinedBy()"),", you can bind services to a defined scope without giving it a name. In this case, the defining service's ",(0,s.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"implementation type")," is used for naming the scope."),(0,s.kt)("admonition",{type:"note"},(0,s.kt)("p",{parentName:"admonition"},"The lifetime of the defined scope is attached to the current scope that was used to create the service."))),(0,s.kt)("div",null,(0,s.kt)(i.Z,{mdxType:"Tabs"},(0,s.kt)(r.Z,{value:"Define named",label:"Define named",mdxType:"TabItem"},(0,s.kt)("pre",null,(0,s.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register(options => options\n .DefinesScope("DbBackupScope"));\ncontainer.Register(options => options\n .InNamedScope("DbBackupScope"));\ncontainer.Register();\n\nvar scope = container.BeginScope();\n\n// DbBackup will create a named scope with the name "DbBackupScope".\n// the named scope will select ConsoleLogger as it\'s \n// bound to the named scope\'s identifier.\nIJob job = scope.Resolve();\n\n// this will dispose the implicitly created named scope by DbBackup.\nscope.Dispose(); \n'))),(0,s.kt)(r.Z,{value:"Define typed",label:"Define typed",mdxType:"TabItem"},(0,s.kt)("pre",null,(0,s.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .DefinesScope());\ncontainer.Register(options => options\n .InScopeDefinedBy());\ncontainer.Register();\n\nvar scope = container.BeginScope();\n\n// DbBackup will create a named scope with the name typeof(DbBackup).\n// the named scope will select ConsoleLogger as it's \n// bound to the named scope's identifier.\nIJob job = scope.Resolve();\n\n// this will dispose the implicitly created named scope by DbBackup.\nscope.Dispose(); \n")))))),(0,s.kt)("h2",{id:"put-instance-to-a-scope"},"Put instance to a scope"),(0,s.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,s.kt)("div",null,(0,s.kt)("p",null,"You can add an already instantiated service to a scope. The instance's lifetime will be tracked by the given scope.")),(0,s.kt)("div",null,(0,s.kt)("pre",null,(0,s.kt)("code",{parentName:"pre",className:"language-cs"},"using var scope = container.BeginScope();\nscope.PutInstanceInScope(new DbBackup());\n")))),(0,s.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,s.kt)("div",null,(0,s.kt)("p",null,"You can disable the tracking by passing ",(0,s.kt)("inlineCode",{parentName:"p"},"true")," for the ",(0,s.kt)("inlineCode",{parentName:"p"},"withoutDisposalTracking")," parameter. In this case, only the strong reference to the instance is dropped when the scope is disposed.")),(0,s.kt)("div",null,(0,s.kt)("pre",null,(0,s.kt)("code",{parentName:"pre",className:"language-cs"},"using var scope = container.BeginScope();\nscope.PutInstanceInScope(new DbBackup(), withoutDisposalTracking: true);\n")))),(0,s.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,s.kt)("div",null,(0,s.kt)("p",null,"You can also give your instance a name to use it like a ",(0,s.kt)("a",{parentName:"p",href:"/docs/guides/basics#named-registration"},"named registration"),":")),(0,s.kt)("div",null,(0,s.kt)("pre",null,(0,s.kt)("code",{parentName:"pre",className:"language-cs"},'using var scope = container.BeginScope();\nscope.PutInstanceInScope(new DbBackup(), false, name: "DbBackup");\n')))),(0,s.kt)("admonition",{type:"note"},(0,s.kt)("p",{parentName:"admonition"},"Instances put to a scope will take precedence over existing registrations with the same ",(0,s.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type"),".")),(0,s.kt)("h2",{id:"disposal"},"Disposal"),(0,s.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,s.kt)("div",null,(0,s.kt)("p",null,"The currently resolving scope tracks services that implement either ",(0,s.kt)("inlineCode",{parentName:"p"},"IDisposable")," or ",(0,s.kt)("inlineCode",{parentName:"p"},"IAsyncDisposable"),". This means that when the scope is disposed, all the tracked disposable instances will be disposed with it."),(0,s.kt)("admonition",{type:"note"},(0,s.kt)("p",{parentName:"admonition"},"Disposing the container will dispose all the singleton instances and their dependencies."))),(0,s.kt)("div",null,(0,s.kt)(i.Z,{groupId:"lifetime-dispose",mdxType:"Tabs"},(0,s.kt)(r.Z,{value:"Using",label:"Using",mdxType:"TabItem"},(0,s.kt)("pre",null,(0,s.kt)("code",{parentName:"pre",className:"language-cs"},"using (var scope = container.BeginScope())\n{\n var disposable = scope.Resolve();\n} // 'disposable' will be disposed when \n // the using statement ends.\n"))),(0,s.kt)(r.Z,{value:"Dispose",label:"Dispose",mdxType:"TabItem"},(0,s.kt)("pre",null,(0,s.kt)("code",{parentName:"pre",className:"language-cs"},"var scope = container.BeginScope();\nvar disposable = scope.Resolve();\n\n// 'disposable' will be disposed with the scope.\nscope.Dispose();\n")))))),(0,s.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,s.kt)("div",null,(0,s.kt)("p",null,"You can disable the disposal tracking on a ",(0,s.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-registration--registered-service"},"service registration")," with the ",(0,s.kt)("inlineCode",{parentName:"p"},".WithoutDisposalTracking()")," option.")),(0,s.kt)("div",null,(0,s.kt)("pre",null,(0,s.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => \n options.WithoutDisposalTracking());\n")))),(0,s.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,s.kt)("div",null,(0,s.kt)("h3",{id:"async-disposal"},"Async disposal"),(0,s.kt)("p",null,"As the container and its scopes implement the ",(0,s.kt)("inlineCode",{parentName:"p"},"IAsyncDisposable")," interface, you can dispose them asynchronously when they are used in an ",(0,s.kt)("inlineCode",{parentName:"p"},"async")," context."),(0,s.kt)("p",null,"Calling ",(0,s.kt)("inlineCode",{parentName:"p"},"DisposeAsync")," disposes both ",(0,s.kt)("inlineCode",{parentName:"p"},"IDisposable")," and ",(0,s.kt)("inlineCode",{parentName:"p"},"IAsyncDisposable")," instances; however, calling ",(0,s.kt)("inlineCode",{parentName:"p"},"Dispose")," only disposes ",(0,s.kt)("inlineCode",{parentName:"p"},"IDisposable")," instances.")),(0,s.kt)("div",null,(0,s.kt)(i.Z,{groupId:"lifetime-dispose",mdxType:"Tabs"},(0,s.kt)(r.Z,{value:"Using",label:"Using",mdxType:"TabItem"},(0,s.kt)("pre",null,(0,s.kt)("code",{parentName:"pre",className:"language-cs"},"await using (var scope = container.BeginScope())\n{\n var disposable = scope.Resolve();\n} // 'disposable' will be disposed asynchronously \n // when the using statement ends.\n"))),(0,s.kt)(r.Z,{value:"Dispose",label:"Dispose",mdxType:"TabItem"},(0,s.kt)("pre",null,(0,s.kt)("code",{parentName:"pre",className:"language-cs"},"var scope = container.BeginScope();\nvar disposable = scope.Resolve();\n\n// 'disposable' will be disposed asynchronously with the scope.\nawait scope.DisposeAsync();\n")))))),(0,s.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,s.kt)("div",null,(0,s.kt)("h3",{id:"finalizer-delegate"},"Finalizer delegate"),(0,s.kt)("p",null,"During ",(0,s.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-registration--registered-service"},"service registration"),", you can set a custom finalizer delegate that will be invoked at the service's disposal.")),(0,s.kt)("div",null,(0,s.kt)("pre",null,(0,s.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => \n options.WithFinalizer(backup => \n backup.CloseDbConnection()));\n")))))}b.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/7a96ca3d.18e419c9.js b/assets/js/7a96ca3d.18e419c9.js new file mode 100644 index 00000000..8ad93903 --- /dev/null +++ b/assets/js/7a96ca3d.18e419c9.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkwebsite=self.webpackChunkwebsite||[]).push([[835],{3905:(t,e,a)=>{a.d(e,{Zo:()=>u,kt:()=>d});var r=a(7294);function n(t,e,a){return e in t?Object.defineProperty(t,e,{value:a,enumerable:!0,configurable:!0,writable:!0}):t[e]=a,t}function s(t,e){var a=Object.keys(t);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(t);e&&(r=r.filter((function(e){return Object.getOwnPropertyDescriptor(t,e).enumerable}))),a.push.apply(a,r)}return a}function o(t){for(var e=1;e=0||(n[a]=t[a]);return n}(t,e);if(Object.getOwnPropertySymbols){var s=Object.getOwnPropertySymbols(t);for(r=0;r=0||Object.prototype.propertyIsEnumerable.call(t,a)&&(n[a]=t[a])}return n}var l=r.createContext({}),p=function(t){var e=r.useContext(l),a=e;return t&&(a="function"==typeof t?t(e):o(o({},e),t)),a},u=function(t){var e=p(t.components);return r.createElement(l.Provider,{value:e},t.children)},c="mdxType",m={inlineCode:"code",wrapper:function(t){var e=t.children;return r.createElement(r.Fragment,{},e)}},h=r.forwardRef((function(t,e){var a=t.components,n=t.mdxType,s=t.originalType,l=t.parentName,u=i(t,["components","mdxType","originalType","parentName"]),c=p(a),h=n,d=c["".concat(l,".").concat(h)]||c[h]||m[h]||s;return a?r.createElement(d,o(o({ref:e},u),{},{components:a})):r.createElement(d,o({ref:e},u))}));function d(t,e){var a=arguments,n=e&&e.mdxType;if("string"==typeof t||n){var s=a.length,o=new Array(s);o[0]=h;var i={};for(var l in e)hasOwnProperty.call(e,l)&&(i[l]=e[l]);i.originalType=t,i[c]="string"==typeof t?t:n,o[1]=i;for(var p=2;p{a.r(e),a.d(e,{assets:()=>l,contentTitle:()=>o,default:()=>c,frontMatter:()=>s,metadata:()=>i,toc:()=>p});var r=a(7462),n=(a(7294),a(3905));const s={title:"Overview"},o="Stashbox",i={unversionedId:"getting-started/overview",id:"getting-started/overview",title:"Overview",description:"Appveyor Build Status",source:"@site/docs/getting-started/overview.md",sourceDirName:"getting-started",slug:"/getting-started/overview",permalink:"/stashbox/docs/getting-started/overview",draft:!1,editUrl:"https://github.com/z4kn4fein/stashbox/edit/master/docs/docs/getting-started/overview.md",tags:[],version:"current",lastUpdatedBy:"dependabot[bot]",lastUpdatedAt:1699538162,formattedLastUpdatedAt:"Nov 9, 2023",frontMatter:{title:"Overview"},sidebar:"docs",next:{title:"Introduction",permalink:"/stashbox/docs/getting-started/introduction"}},l={},p=[{value:"Core attributes",id:"core-attributes",level:2},{value:"Supported platforms",id:"supported-platforms",level:2},{value:"Contact & support",id:"contact--support",level:2},{value:"License",id:"license",level:2}],u={toc:p};function c(t){let{components:e,...a}=t;return(0,n.kt)("wrapper",(0,r.Z)({},u,a,{components:e,mdxType:"MDXLayout"}),(0,n.kt)("h1",{id:"stashbox"},"Stashbox"),(0,n.kt)("p",null,(0,n.kt)("a",{parentName:"p",href:"https://ci.appveyor.com/project/pcsajtai/stashbox/branch/master"},(0,n.kt)("img",{parentName:"a",src:"https://img.shields.io/appveyor/build/pcsajtai/stashbox?logo=appveyor&logoColor=white",alt:"Appveyor Build Status"})),"\n",(0,n.kt)("a",{parentName:"p",href:"https://github.com/z4kn4fein/stashbox/actions/workflows/linux-macOS-CI.yml"},(0,n.kt)("img",{parentName:"a",src:"https://img.shields.io/github/actions/workflow/status/z4kn4fein/stashbox/linux-macOS-CI.yml?logo=GitHub&branch=master",alt:"GitHub Workflow Status"})),"\n",(0,n.kt)("a",{parentName:"p",href:"https://sonarcloud.io/project/overview?id=z4kn4fein_stashbox"},(0,n.kt)("img",{parentName:"a",src:"https://img.shields.io/sonar/tests/z4kn4fein_stashbox?compact_message&logo=sonarcloud&server=https%3A%2F%2Fsonarcloud.io",alt:"Sonar Tests"})),"\n",(0,n.kt)("a",{parentName:"p",href:"https://sonarcloud.io/project/overview?id=z4kn4fein_stashbox"},(0,n.kt)("img",{parentName:"a",src:"https://img.shields.io/sonar/coverage/z4kn4fein_stashbox?logo=SonarCloud&server=https%3A%2F%2Fsonarcloud.io",alt:"Sonar Coverage"})),"\n",(0,n.kt)("a",{parentName:"p",href:"https://sonarcloud.io/project/overview?id=z4kn4fein_stashbox"},(0,n.kt)("img",{parentName:"a",src:"https://img.shields.io/sonar/quality_gate/z4kn4fein_stashbox?logo=sonarcloud&server=https%3A%2F%2Fsonarcloud.io",alt:"Sonar Quality Gate"})),"\n",(0,n.kt)("a",{parentName:"p",href:"https://github.com/dotnet/sourcelink"},(0,n.kt)("img",{parentName:"a",src:"https://img.shields.io/badge/sourcelink-enabled-brightgreen.svg",alt:"Sourcelink"}))),(0,n.kt)("p",null,"Stashbox is a lightweight, fast, and portable dependency injection framework for .NET-based solutions. It encourages the building of loosely coupled applications and simplifies the construction of hierarchical object structures. It can be integrated easily with .NET Core, Generic Host, ASP.NET, Xamarin, and many other applications."),(0,n.kt)("p",null,"These are the latest available stable and pre-release versions:"),(0,n.kt)("table",null,(0,n.kt)("thead",{parentName:"table"},(0,n.kt)("tr",{parentName:"thead"},(0,n.kt)("th",{parentName:"tr",align:null},"Github (stable)"),(0,n.kt)("th",{parentName:"tr",align:null},"NuGet (stable)"),(0,n.kt)("th",{parentName:"tr",align:null},"Fuget (stable)"),(0,n.kt)("th",{parentName:"tr",align:null},"NuGet (daily)"))),(0,n.kt)("tbody",{parentName:"table"},(0,n.kt)("tr",{parentName:"tbody"},(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"https://github.com/z4kn4fein/stashbox/releases"},(0,n.kt)("img",{parentName:"a",src:"https://img.shields.io/github/release/z4kn4fein/stashbox.svg",alt:"Github release"}))),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"https://www.nuget.org/packages/Stashbox/"},(0,n.kt)("img",{parentName:"a",src:"https://buildstats.info/nuget/Stashbox",alt:"NuGet Version"}))),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"https://www.fuget.org/packages/Stashbox"},(0,n.kt)("img",{parentName:"a",src:"https://www.fuget.org/packages/Stashbox/badge.svg?v=5.12.2",alt:"Stashbox on fuget.org"}))),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"https://www.nuget.org/packages/Stashbox/"},(0,n.kt)("img",{parentName:"a",src:"https://img.shields.io/nuget/vpre/Stashbox",alt:"Nuget pre-release"})))))),(0,n.kt)("h2",{id:"core-attributes"},"Core attributes"),(0,n.kt)("ul",null,(0,n.kt)("li",{parentName:"ul"},"\ud83d\ude80 Fast, thread-safe, and lock-free operations."),(0,n.kt)("li",{parentName:"ul"},"\u26a1\ufe0f Easy-to-use Fluent configuration API."),(0,n.kt)("li",{parentName:"ul"},"\u267b\ufe0f Small memory footprint."),(0,n.kt)("li",{parentName:"ul"},"\ud83d\udd04 Tracks the dependency tree for cycles. "),(0,n.kt)("li",{parentName:"ul"},"\ud83d\udea8 Detects and warns about misconfigurations."),(0,n.kt)("li",{parentName:"ul"},"\ud83d\udd25 Gives fast feedback on registration/resolution issues.")),(0,n.kt)("h2",{id:"supported-platforms"},"Supported platforms"),(0,n.kt)("ul",null,(0,n.kt)("li",{parentName:"ul"},".NET 5+"),(0,n.kt)("li",{parentName:"ul"},".NET Standard 2.0+"),(0,n.kt)("li",{parentName:"ul"},".NET Framework 4.5+"),(0,n.kt)("li",{parentName:"ul"},"Mono"),(0,n.kt)("li",{parentName:"ul"},"Universal Windows Platform"),(0,n.kt)("li",{parentName:"ul"},"Xamarin (Android/iOS/Mac)"),(0,n.kt)("li",{parentName:"ul"},"Unity")),(0,n.kt)("h2",{id:"contact--support"},"Contact & support"),(0,n.kt)("ul",null,(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("a",{parentName:"li",href:"https://gitter.im/z4kn4fein/stashbox"},(0,n.kt)("img",{parentName:"a",src:"https://img.shields.io/gitter/room/z4kn4fein/stashbox.svg",alt:"Join the chat at https://gitter.im/z4kn4fein/stashbox"}))," ",(0,n.kt)("a",{parentName:"li",href:"https://3vj.short.gy/stashbox-slack"},(0,n.kt)("img",{parentName:"a",src:"https://img.shields.io/badge/chat-on%20slack-orange.svg?style=flat",alt:"Slack"}))),(0,n.kt)("li",{parentName:"ul"},"Create a ",(0,n.kt)("a",{parentName:"li",href:"https://github.com/z4kn4fein/stashbox/issues"},"GitHub issue")," for bug reports and feature requests."),(0,n.kt)("li",{parentName:"ul"},"Start a ",(0,n.kt)("a",{parentName:"li",href:"https://github.com/z4kn4fein/stashbox/discussions"},"GitHub discussion")," for your questions and ideas."),(0,n.kt)("li",{parentName:"ul"},"Add a \u2b50\ufe0f ",(0,n.kt)("a",{parentName:"li",href:"https://github.com/z4kn4fein/stashbox"},"on GitHub")," to support the project!")),(0,n.kt)("h2",{id:"license"},"License"),(0,n.kt)("p",null,"This project is licensed under the ",(0,n.kt)("a",{parentName:"p",href:"https://github.com/z4kn4fein/stashbox/blob/master/LICENSE"},"MIT license"),"."))}c.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/7a96ca3d.b2b825bb.js b/assets/js/7a96ca3d.b2b825bb.js deleted file mode 100644 index fb53b924..00000000 --- a/assets/js/7a96ca3d.b2b825bb.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkwebsite=self.webpackChunkwebsite||[]).push([[835],{3905:(t,e,a)=>{a.d(e,{Zo:()=>u,kt:()=>d});var r=a(7294);function n(t,e,a){return e in t?Object.defineProperty(t,e,{value:a,enumerable:!0,configurable:!0,writable:!0}):t[e]=a,t}function s(t,e){var a=Object.keys(t);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(t);e&&(r=r.filter((function(e){return Object.getOwnPropertyDescriptor(t,e).enumerable}))),a.push.apply(a,r)}return a}function o(t){for(var e=1;e=0||(n[a]=t[a]);return n}(t,e);if(Object.getOwnPropertySymbols){var s=Object.getOwnPropertySymbols(t);for(r=0;r=0||Object.prototype.propertyIsEnumerable.call(t,a)&&(n[a]=t[a])}return n}var l=r.createContext({}),p=function(t){var e=r.useContext(l),a=e;return t&&(a="function"==typeof t?t(e):o(o({},e),t)),a},u=function(t){var e=p(t.components);return r.createElement(l.Provider,{value:e},t.children)},c="mdxType",m={inlineCode:"code",wrapper:function(t){var e=t.children;return r.createElement(r.Fragment,{},e)}},h=r.forwardRef((function(t,e){var a=t.components,n=t.mdxType,s=t.originalType,l=t.parentName,u=i(t,["components","mdxType","originalType","parentName"]),c=p(a),h=n,d=c["".concat(l,".").concat(h)]||c[h]||m[h]||s;return a?r.createElement(d,o(o({ref:e},u),{},{components:a})):r.createElement(d,o({ref:e},u))}));function d(t,e){var a=arguments,n=e&&e.mdxType;if("string"==typeof t||n){var s=a.length,o=new Array(s);o[0]=h;var i={};for(var l in e)hasOwnProperty.call(e,l)&&(i[l]=e[l]);i.originalType=t,i[c]="string"==typeof t?t:n,o[1]=i;for(var p=2;p{a.r(e),a.d(e,{assets:()=>l,contentTitle:()=>o,default:()=>c,frontMatter:()=>s,metadata:()=>i,toc:()=>p});var r=a(7462),n=(a(7294),a(3905));const s={title:"Overview"},o="Stashbox",i={unversionedId:"getting-started/overview",id:"getting-started/overview",title:"Overview",description:"Appveyor Build Status",source:"@site/docs/getting-started/overview.md",sourceDirName:"getting-started",slug:"/getting-started/overview",permalink:"/stashbox/docs/getting-started/overview",draft:!1,editUrl:"https://github.com/z4kn4fein/stashbox/edit/master/docs/docs/getting-started/overview.md",tags:[],version:"current",lastUpdatedBy:"dependabot[bot]",lastUpdatedAt:1697644420,formattedLastUpdatedAt:"Oct 18, 2023",frontMatter:{title:"Overview"},sidebar:"docs",next:{title:"Introduction",permalink:"/stashbox/docs/getting-started/introduction"}},l={},p=[{value:"Core attributes",id:"core-attributes",level:2},{value:"Supported platforms",id:"supported-platforms",level:2},{value:"Contact & support",id:"contact--support",level:2},{value:"License",id:"license",level:2}],u={toc:p};function c(t){let{components:e,...a}=t;return(0,n.kt)("wrapper",(0,r.Z)({},u,a,{components:e,mdxType:"MDXLayout"}),(0,n.kt)("h1",{id:"stashbox"},"Stashbox"),(0,n.kt)("p",null,(0,n.kt)("a",{parentName:"p",href:"https://ci.appveyor.com/project/pcsajtai/stashbox/branch/master"},(0,n.kt)("img",{parentName:"a",src:"https://img.shields.io/appveyor/build/pcsajtai/stashbox?logo=appveyor&logoColor=white",alt:"Appveyor Build Status"})),"\n",(0,n.kt)("a",{parentName:"p",href:"https://github.com/z4kn4fein/stashbox/actions/workflows/linux-macOS-CI.yml"},(0,n.kt)("img",{parentName:"a",src:"https://img.shields.io/github/actions/workflow/status/z4kn4fein/stashbox/linux-macOS-CI.yml?logo=GitHub&branch=master",alt:"GitHub Workflow Status"})),"\n",(0,n.kt)("a",{parentName:"p",href:"https://sonarcloud.io/project/overview?id=z4kn4fein_stashbox"},(0,n.kt)("img",{parentName:"a",src:"https://img.shields.io/sonar/tests/z4kn4fein_stashbox?compact_message&logo=sonarcloud&server=https%3A%2F%2Fsonarcloud.io",alt:"Sonar Tests"})),"\n",(0,n.kt)("a",{parentName:"p",href:"https://sonarcloud.io/project/overview?id=z4kn4fein_stashbox"},(0,n.kt)("img",{parentName:"a",src:"https://img.shields.io/sonar/coverage/z4kn4fein_stashbox?logo=SonarCloud&server=https%3A%2F%2Fsonarcloud.io",alt:"Sonar Coverage"})),"\n",(0,n.kt)("a",{parentName:"p",href:"https://sonarcloud.io/project/overview?id=z4kn4fein_stashbox"},(0,n.kt)("img",{parentName:"a",src:"https://img.shields.io/sonar/quality_gate/z4kn4fein_stashbox?logo=sonarcloud&server=https%3A%2F%2Fsonarcloud.io",alt:"Sonar Quality Gate"})),"\n",(0,n.kt)("a",{parentName:"p",href:"https://github.com/dotnet/sourcelink"},(0,n.kt)("img",{parentName:"a",src:"https://img.shields.io/badge/sourcelink-enabled-brightgreen.svg",alt:"Sourcelink"}))),(0,n.kt)("p",null,"Stashbox is a lightweight, fast, and portable dependency injection framework for .NET-based solutions. It encourages the building of loosely coupled applications and simplifies the construction of hierarchical object structures. It can be integrated easily with .NET Core, Generic Host, ASP.NET, Xamarin, and many other applications."),(0,n.kt)("p",null,"These are the latest available stable and pre-release versions:"),(0,n.kt)("table",null,(0,n.kt)("thead",{parentName:"table"},(0,n.kt)("tr",{parentName:"thead"},(0,n.kt)("th",{parentName:"tr",align:null},"Github (stable)"),(0,n.kt)("th",{parentName:"tr",align:null},"NuGet (stable)"),(0,n.kt)("th",{parentName:"tr",align:null},"Fuget (stable)"),(0,n.kt)("th",{parentName:"tr",align:null},"NuGet (daily)"))),(0,n.kt)("tbody",{parentName:"table"},(0,n.kt)("tr",{parentName:"tbody"},(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"https://github.com/z4kn4fein/stashbox/releases"},(0,n.kt)("img",{parentName:"a",src:"https://img.shields.io/github/release/z4kn4fein/stashbox.svg",alt:"Github release"}))),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"https://www.nuget.org/packages/Stashbox/"},(0,n.kt)("img",{parentName:"a",src:"https://buildstats.info/nuget/Stashbox",alt:"NuGet Version"}))),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"https://www.fuget.org/packages/Stashbox"},(0,n.kt)("img",{parentName:"a",src:"https://www.fuget.org/packages/Stashbox/badge.svg?v=5.12.2",alt:"Stashbox on fuget.org"}))),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"https://www.nuget.org/packages/Stashbox/"},(0,n.kt)("img",{parentName:"a",src:"https://img.shields.io/nuget/vpre/Stashbox",alt:"Nuget pre-release"})))))),(0,n.kt)("h2",{id:"core-attributes"},"Core attributes"),(0,n.kt)("ul",null,(0,n.kt)("li",{parentName:"ul"},"\ud83d\ude80 Fast, thread-safe, and lock-free operations."),(0,n.kt)("li",{parentName:"ul"},"\u26a1\ufe0f Easy-to-use Fluent configuration API."),(0,n.kt)("li",{parentName:"ul"},"\u267b\ufe0f Small memory footprint."),(0,n.kt)("li",{parentName:"ul"},"\ud83d\udd04 Tracks the dependency tree for cycles. "),(0,n.kt)("li",{parentName:"ul"},"\ud83d\udea8 Detects and warns about misconfigurations."),(0,n.kt)("li",{parentName:"ul"},"\ud83d\udd25 Gives fast feedback on registration/resolution issues.")),(0,n.kt)("h2",{id:"supported-platforms"},"Supported platforms"),(0,n.kt)("ul",null,(0,n.kt)("li",{parentName:"ul"},".NET 5+"),(0,n.kt)("li",{parentName:"ul"},".NET Standard 2.0+"),(0,n.kt)("li",{parentName:"ul"},".NET Framework 4.5+"),(0,n.kt)("li",{parentName:"ul"},"Mono"),(0,n.kt)("li",{parentName:"ul"},"Universal Windows Platform"),(0,n.kt)("li",{parentName:"ul"},"Xamarin (Android/iOS/Mac)"),(0,n.kt)("li",{parentName:"ul"},"Unity")),(0,n.kt)("h2",{id:"contact--support"},"Contact & support"),(0,n.kt)("ul",null,(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("a",{parentName:"li",href:"https://gitter.im/z4kn4fein/stashbox"},(0,n.kt)("img",{parentName:"a",src:"https://img.shields.io/gitter/room/z4kn4fein/stashbox.svg",alt:"Join the chat at https://gitter.im/z4kn4fein/stashbox"}))," ",(0,n.kt)("a",{parentName:"li",href:"https://3vj.short.gy/stashbox-slack"},(0,n.kt)("img",{parentName:"a",src:"https://img.shields.io/badge/chat-on%20slack-orange.svg?style=flat",alt:"Slack"}))),(0,n.kt)("li",{parentName:"ul"},"Create a ",(0,n.kt)("a",{parentName:"li",href:"https://github.com/z4kn4fein/stashbox/issues"},"GitHub issue")," for bug reports and feature requests."),(0,n.kt)("li",{parentName:"ul"},"Start a ",(0,n.kt)("a",{parentName:"li",href:"https://github.com/z4kn4fein/stashbox/discussions"},"GitHub discussion")," for your questions and ideas."),(0,n.kt)("li",{parentName:"ul"},"Add a \u2b50\ufe0f ",(0,n.kt)("a",{parentName:"li",href:"https://github.com/z4kn4fein/stashbox"},"on GitHub")," to support the project!")),(0,n.kt)("h2",{id:"license"},"License"),(0,n.kt)("p",null,"This project is licensed under the ",(0,n.kt)("a",{parentName:"p",href:"https://github.com/z4kn4fein/stashbox/blob/master/LICENSE"},"MIT license"),"."))}c.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/851a7a9e.1667941e.js b/assets/js/851a7a9e.1667941e.js new file mode 100644 index 00000000..17eb6ff3 --- /dev/null +++ b/assets/js/851a7a9e.1667941e.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkwebsite=self.webpackChunkwebsite||[]).push([[153],{3905:(e,t,r)=>{r.d(t,{Zo:()=>d,kt:()=>h});var n=r(7294);function o(e,t,r){return t in e?Object.defineProperty(e,t,{value:r,enumerable:!0,configurable:!0,writable:!0}):e[t]=r,e}function a(e,t){var r=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);t&&(n=n.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),r.push.apply(r,n)}return r}function s(e){for(var t=1;t=0||(o[r]=e[r]);return o}(e,t);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(e,r)&&(o[r]=e[r])}return o}var c=n.createContext({}),l=function(e){var t=n.useContext(c),r=t;return e&&(r="function"==typeof e?e(t):s(s({},t),e)),r},d=function(e){var t=l(e.components);return n.createElement(c.Provider,{value:t},e.children)},p="mdxType",u={inlineCode:"code",wrapper:function(e){var t=e.children;return n.createElement(n.Fragment,{},t)}},v=n.forwardRef((function(e,t){var r=e.components,o=e.mdxType,a=e.originalType,c=e.parentName,d=i(e,["components","mdxType","originalType","parentName"]),p=l(r),v=o,h=p["".concat(c,".").concat(v)]||p[v]||u[v]||a;return r?n.createElement(h,s(s({ref:t},d),{},{components:r})):n.createElement(h,s({ref:t},d))}));function h(e,t){var r=arguments,o=t&&t.mdxType;if("string"==typeof e||o){var a=r.length,s=new Array(a);s[0]=v;var i={};for(var c in t)hasOwnProperty.call(t,c)&&(i[c]=t[c]);i.originalType=e,i[p]="string"==typeof e?e:o,s[1]=i;for(var l=2;l{r.d(t,{Z:()=>s});var n=r(7294),o=r(6010);const a="tabItem_Ymn6";function s(e){let{children:t,hidden:r,className:s}=e;return n.createElement("div",{role:"tabpanel",className:(0,o.Z)(a,s),hidden:r},t)}},4866:(e,t,r)=>{r.d(t,{Z:()=>w});var n=r(7462),o=r(7294),a=r(6010),s=r(2466),i=r(6550),c=r(1980),l=r(7392),d=r(12);function p(e){return function(e){return o.Children.map(e,(e=>{if(!e||(0,o.isValidElement)(e)&&function(e){const{props:t}=e;return!!t&&"object"==typeof t&&"value"in t}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}(e).map((e=>{let{props:{value:t,label:r,attributes:n,default:o}}=e;return{value:t,label:r,attributes:n,default:o}}))}function u(e){const{values:t,children:r}=e;return(0,o.useMemo)((()=>{const e=t??p(r);return function(e){const t=(0,l.l)(e,((e,t)=>e.value===t.value));if(t.length>0)throw new Error(`Docusaurus error: Duplicate values "${t.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[t,r])}function v(e){let{value:t,tabValues:r}=e;return r.some((e=>e.value===t))}function h(e){let{queryString:t=!1,groupId:r}=e;const n=(0,i.k6)(),a=function(e){let{queryString:t=!1,groupId:r}=e;if("string"==typeof t)return t;if(!1===t)return null;if(!0===t&&!r)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return r??null}({queryString:t,groupId:r});return[(0,c._X)(a),(0,o.useCallback)((e=>{if(!a)return;const t=new URLSearchParams(n.location.search);t.set(a,e),n.replace({...n.location,search:t.toString()})}),[a,n])]}function g(e){const{defaultValue:t,queryString:r=!1,groupId:n}=e,a=u(e),[s,i]=(0,o.useState)((()=>function(e){let{defaultValue:t,tabValues:r}=e;if(0===r.length)throw new Error("Docusaurus error: the component requires at least one children component");if(t){if(!v({value:t,tabValues:r}))throw new Error(`Docusaurus error: The has a defaultValue "${t}" but none of its children has the corresponding value. Available values are: ${r.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return t}const n=r.find((e=>e.default))??r[0];if(!n)throw new Error("Unexpected error: 0 tabValues");return n.value}({defaultValue:t,tabValues:a}))),[c,l]=h({queryString:r,groupId:n}),[p,g]=function(e){let{groupId:t}=e;const r=function(e){return e?`docusaurus.tab.${e}`:null}(t),[n,a]=(0,d.Nk)(r);return[n,(0,o.useCallback)((e=>{r&&a.set(e)}),[r,a])]}({groupId:n}),m=(()=>{const e=c??p;return v({value:e,tabValues:a})?e:null})();(0,o.useLayoutEffect)((()=>{m&&i(m)}),[m]);return{selectedValue:s,selectValue:(0,o.useCallback)((e=>{if(!v({value:e,tabValues:a}))throw new Error(`Can't select invalid tab value=${e}`);i(e),l(e),g(e)}),[l,g,a]),tabValues:a}}var m=r(2389);const f="tabList__CuJ",E="tabItem_LNqP";function P(e){let{className:t,block:r,selectedValue:i,selectValue:c,tabValues:l}=e;const d=[],{blockElementScrollPositionUntilNextRender:p}=(0,s.o5)(),u=e=>{const t=e.currentTarget,r=d.indexOf(t),n=l[r].value;n!==i&&(p(t),c(n))},v=e=>{let t=null;switch(e.key){case"Enter":u(e);break;case"ArrowRight":{const r=d.indexOf(e.currentTarget)+1;t=d[r]??d[0];break}case"ArrowLeft":{const r=d.indexOf(e.currentTarget)-1;t=d[r]??d[d.length-1];break}}t?.focus()};return o.createElement("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,a.Z)("tabs",{"tabs--block":r},t)},l.map((e=>{let{value:t,label:r,attributes:s}=e;return o.createElement("li",(0,n.Z)({role:"tab",tabIndex:i===t?0:-1,"aria-selected":i===t,key:t,ref:e=>d.push(e),onKeyDown:v,onClick:u},s,{className:(0,a.Z)("tabs__item",E,s?.className,{"tabs__item--active":i===t})}),r??t)})))}function b(e){let{lazy:t,children:r,selectedValue:n}=e;const a=(Array.isArray(r)?r:[r]).filter(Boolean);if(t){const e=a.find((e=>e.props.value===n));return e?(0,o.cloneElement)(e,{className:"margin-top--md"}):null}return o.createElement("div",{className:"margin-top--md"},a.map(((e,t)=>(0,o.cloneElement)(e,{key:t,hidden:e.props.value!==n}))))}function y(e){const t=g(e);return o.createElement("div",{className:(0,a.Z)("tabs-container",f)},o.createElement(P,(0,n.Z)({},e,t)),o.createElement(b,(0,n.Z)({},e,t)))}function w(e){const t=(0,m.Z)();return o.createElement(y,(0,n.Z)({key:String(t)},e))}},8846:(e,t,r)=>{r.d(t,{Z:()=>i});var n=r(7294);const o="codeDescContainer_ie8f",a="desc_jyqI",s="example_eYlF";function i(e){let{children:t}=e,r=n.Children.toArray(t).filter((e=>e));return n.createElement("div",{className:o},n.createElement("div",{className:a},r[0]),n.createElement("div",{className:s},r[1]))}},1212:(e,t,r)=>{r.r(t),r.d(t,{assets:()=>p,contentTitle:()=>l,default:()=>h,frontMatter:()=>c,metadata:()=>d,toc:()=>u});var n=r(7462),o=(r(7294),r(3905)),a=r(8846),s=r(4866),i=r(5162);const c={},l="Decorators",d={unversionedId:"advanced/decorators",id:"advanced/decorators",title:"Decorators",description:"Stashbox supports decorator service registration to take advantage of the Decorator pattern. This pattern is used to extend the functionality of a class without changing its implementation. This is also what the Open\u2013closed principle stands for; services should be open for extension but closed for modification.",source:"@site/docs/advanced/decorators.md",sourceDirName:"advanced",slug:"/advanced/decorators",permalink:"/stashbox/docs/advanced/decorators",draft:!1,editUrl:"https://github.com/z4kn4fein/stashbox/edit/master/docs/docs/advanced/decorators.md",tags:[],version:"current",lastUpdatedBy:"dependabot[bot]",lastUpdatedAt:1699538162,formattedLastUpdatedAt:"Nov 9, 2023",frontMatter:{},sidebar:"docs",previous:{title:"Generics",permalink:"/stashbox/docs/advanced/generics"},next:{title:"Wrappers & resolvers",permalink:"/stashbox/docs/advanced/wrappers-resolvers"}},p={},u=[{value:"Simple use-case",id:"simple-use-case",level:2},{value:"Multiple decorators",id:"multiple-decorators",level:2},{value:"Conditional decoration",id:"conditional-decoration",level:2},{value:"Generic decorators",id:"generic-decorators",level:2},{value:"Composite pattern",id:"composite-pattern",level:2},{value:"Decorating multiple services",id:"decorating-multiple-services",level:2},{value:"Lifetime",id:"lifetime",level:2},{value:"Wrappers",id:"wrappers",level:2},{value:"Interception",id:"interception",level:2}],v={toc:u};function h(e){let{components:t,...r}=e;return(0,o.kt)("wrapper",(0,n.Z)({},v,r,{components:t,mdxType:"MDXLayout"}),(0,o.kt)("h1",{id:"decorators"},"Decorators"),(0,o.kt)("p",null,"Stashbox supports decorator ",(0,o.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-registration--registered-service"},"service registration")," to take advantage of the ",(0,o.kt)("a",{parentName:"p",href:"https://en.wikipedia.org/wiki/Decorator_pattern"},"Decorator pattern"),". This pattern is used to extend the functionality of a class without changing its implementation. This is also what the ",(0,o.kt)("a",{parentName:"p",href:"https://en.wikipedia.org/wiki/Open%E2%80%93closed_principle"},"Open\u2013closed principle")," stands for; services should be open for extension but closed for modification."),(0,o.kt)("h2",{id:"simple-use-case"},"Simple use-case"),(0,o.kt)("p",null,"We define an ",(0,o.kt)("inlineCode",{parentName:"p"},"IEventProcessor")," service used to process ",(0,o.kt)("inlineCode",{parentName:"p"},"Event")," entities. Then we'll decorate this service with additional validation capabilities:"),(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"class Event { }\nclass UpdateEvent : Event { }\n\ninterface IEventProcessor\n{\n void ProcessEvent(Event @event);\n}\n\ninterface IEventValidator\n{\n bool IsValid(Event @event);\n}\n\nclass EventValidator : IEventValidator\n{\n public bool IsValid(Event @event) { /* do the actual validation. */ }\n}\n\nclass GeneralEventProcessor : IEventProcessor\n{\n public void ProcessEvent(Event @event)\n {\n // suppose this method is processing the given event.\n this.DoTheActualProcessing(@event);\n }\n}\n\nclass ValidatorProcessor : IEventProcessor\n{\n private readonly IEventProcessor nextProcessor;\n private readonly IEventValidator eventValidator;\n public ValidatorProcessor(IEventProcessor eventProcessor, IEventValidator eventValidator)\n {\n this.nextProcessor = eventProcessor;\n this.eventValidator = eventValidator;\n }\n\n public void ProcessEvent(Event @event)\n {\n // validate the event first.\n if (!this.eventValidator.IsValid(@event))\n throw new InvalidEventException();\n\n // if everything is ok, call the next processor.\n this.nextProcessor.ProcessEvent(@event);\n }\n}\n\nusing var container = new StashboxContainer();\n\ncontainer.Register();\ncontainer.Register();\ncontainer.RegisterDecorator();\n\n// new ValidatorProcessor(new GeneralEventProcessor(), new EventValidator())\nvar eventProcessor = container.Resolve();\n\n// process the event.\neventProcessor.ProcessEvent(new UpdateEvent());\n")),(0,o.kt)("p",null,"The ",(0,o.kt)("inlineCode",{parentName:"p"},"GeneralEventProcessor")," is an implementation of ",(0,o.kt)("inlineCode",{parentName:"p"},"IEventProcessor")," and does the actual event processing logic. It does not have any other responsibilities. Rather than putting the event validation's burden onto its shoulder, we create a different service for validation purposes. Instead of injecting the validator into the ",(0,o.kt)("inlineCode",{parentName:"p"},"GeneralEventProcessor")," directly, we let another ",(0,o.kt)("inlineCode",{parentName:"p"},"IEventProcessor")," decorate it like an ",(0,o.kt)("em",{parentName:"p"},"event processing pipeline")," that validates the event as a first step."),(0,o.kt)("h2",{id:"multiple-decorators"},"Multiple decorators"),(0,o.kt)(a.Z,{mdxType:"CodeDescPanel"},(0,o.kt)("div",null,(0,o.kt)("p",null,"You have the option to register multiple decorators for a service to extend its functionality."),(0,o.kt)("p",null,"The decoration order will be the same as the registration order of the decorators. The first registered decorator will decorate the service itself. Then, all the subsequent decorators will wrap the already decorated service.")),(0,o.kt)("div",null,(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register();\ncontainer.RegisterDecorator();\ncontainer.RegisterDecorator();\n\n// new ValidatorProcessor(new LoggerProcessor(new GeneralProcessor()));\nvar processor = container.Resolve(); \n")))),(0,o.kt)("h2",{id:"conditional-decoration"},"Conditional decoration"),(0,o.kt)("p",null,"With ",(0,o.kt)("a",{parentName:"p",href:"/docs/guides/service-resolution#conditional-resolution"},"conditional resolution")," you can control which decorator should be selected to decorate a given service."),(0,o.kt)(s.Z,{mdxType:"Tabs"},(0,o.kt)(i.Z,{value:"Decoretee",label:"Decoretee",mdxType:"TabItem"},(0,o.kt)("p",null,"You have the option to set which decorator should be selected for a given implementation. For a single type filter, you can use the ",(0,o.kt)("inlineCode",{parentName:"p"},".WhenDecoratedServiceIs()")," configuration option. To select more types, you can use the more generic ",(0,o.kt)("inlineCode",{parentName:"p"},".When()")," option. "),(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register();\ncontainer.Register();\n\ncontainer.RegisterDecorator(options => options\n // select when CustomProcessor or GeneralProcessor is resolved.\n .WhenDecoratedServiceIs()\n .WhenDecoratedServiceIs());\n\ncontainer.RegisterDecorator(options => options\n // select only when GeneralProcessor is resolved.\n .WhenDecoratedServiceIs());\n\n// [\n// new ValidatorProcessor(new LoggerProcessor(new GeneralProcessor())),\n// new LoggerProcessor(new CustomProcessor())\n// ]\nvar processors = container.ResolveAll();\n"))),(0,o.kt)(i.Z,{value:"Named",label:"Named",mdxType:"TabItem"},(0,o.kt)("p",null,"You can filter for service names to control the decorator selection."),(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register("General");\ncontainer.Register("Custom");\n\ncontainer.RegisterDecorator(options => options\n // select when CustomProcessor or GeneralProcessor is resolved.\n .WhenDecoratedServiceIs("General")\n .WhenDecoratedServiceIs("Custom"));\n\ncontainer.RegisterDecorator(options => options\n // select only when GeneralProcessor is resolved.\n .WhenDecoratedServiceIs("General"));\n\n// new ValidatorProcessor(new LoggerProcessor(new GeneralProcessor()))\nvar general = container.Resolve("General");\n\n// new LoggerProcessor(new CustomProcessor())\nvar custom = container.Resolve("Custom");\n'))),(0,o.kt)(i.Z,{value:"Attribute",label:"Attribute",mdxType:"TabItem"},(0,o.kt)("p",null,"You can use your custom attributes to control the decorator selection. With ",(0,o.kt)("strong",{parentName:"p"},"class attributes"),", you can mark your classes for decoration."),(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"class LogAttribute : Attribute { }\nclass ValidateAttribute : Attribute { }\n\n[Log, Validate]\nclass GeneralProcessor : IEventProcessor { }\n\n[Log]\nclass CustomProcessor : IEventProcessor { }\n\ncontainer.Register();\ncontainer.Register();\n\ncontainer.RegisterDecorator(options => options\n // select when the resolving class has 'LogAttribute'.\n .WhenDecoratedServiceHas());\n\ncontainer.RegisterDecorator(options => options\n // select when the resolving class has 'ValidateAttribute'.\n .WhenDecoratedServiceHas());\n\n// [\n// new ValidatorProcessor(new LoggerProcessor(new GeneralProcessor())),\n// new LoggerProcessor(new CustomProcessor())\n// ]\nvar processors = container.ResolveAll();\n")),(0,o.kt)("p",null,"You can also mark your dependencies for decoration with ",(0,o.kt)("strong",{parentName:"p"},"property / field / parameter attributes"),". "),(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"class LogAttribute : Attribute { }\nclass ValidateAttribute : Attribute { }\n\nclass ProcessorExecutor\n{\n public ProcessorExecutor([Log, Validate]IEventProcessor eventProcessor)\n { }\n}\n\ncontainer.Register();\ncontainer.Register();\n\ncontainer.RegisterDecorator(options => options\n // select when the resolving dependency has 'LogAttribute'.\n .WhenHas());\n\ncontainer.RegisterDecorator(options => options\n // select when the resolving dependency has 'ValidateAttribute'.\n .WhenHas());\n\n// new ProcessorExecutor(new ValidatorProcessor(new LoggerProcessor(new GeneralProcessor())))\nvar executor = container.ResolveAll();\n")))),(0,o.kt)("h2",{id:"generic-decorators"},"Generic decorators"),(0,o.kt)("p",null,"Stashbox supports the registration of open-generic decorators, which allows the extension of open-generic services.\nInspection of ",(0,o.kt)("a",{parentName:"p",href:"/docs/advanced/generics#generic-constraints"},"generic parameter constraints")," and ",(0,o.kt)("a",{parentName:"p",href:"/docs/advanced/generics#variance"},"variance handling")," is supported on generic decorators also."),(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"interface IEventProcessor\n{\n void ProcessEvent(TEvent @event);\n}\n\nclass GeneralEventProcessor : IEventProcessor\n{\n public void ProcessEvent(TEvent @event) { /* suppose this method is processing the given event.*/ }\n}\n\nclass ValidatorProcessor : IEventProcessor\n{\n private readonly IEventProcessor nextProcessor;\n\n public ValidatorProcessor(IEventProcessor eventProcessor)\n {\n this.nextProcessor = eventProcessor;\n }\n\n public void ProcessEvent(TEvent @event)\n {\n // validate the event first.\n if (!this.IsValid(@event))\n throw new InvalidEventException();\n\n // if everything is ok, call the next processor.\n this.nextProcessor.ProcessEvent(@event);\n }\n}\n\nusing var container = new StashboxContainer();\ncontainer.Register(typeof(IEventProcessor<>), typeof(GeneralEventProcessor<>));\ncontainer.RegisterDecorator(typeof(IEventProcessor<>), typeof(ValidatorProcessor<>));\n\n// new ValidatorProcessor(new GeneralEventProcessor())\nvar eventProcessor = container.Resolve>();\n\n// process the event.\neventProcessor.ProcessEvent(new UpdateEvent());\n")),(0,o.kt)("h2",{id:"composite-pattern"},"Composite pattern"),(0,o.kt)("p",null,"The ",(0,o.kt)("a",{parentName:"p",href:"https://en.wikipedia.org/wiki/Composite_pattern"},"Composite pattern")," allows a group of objects to be treated the same way as a single instance of the same type. It's useful when you want to use the functionality of multiple instances behind the same interface. You can achieve this by registering a decorator that takes a collection of the same service as a dependency. "),(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"public class CompositeValidator : IEventValidator\n{\n private readonly IEnumerable> validators;\n\n public CompositeValidator(IEnumerable> validators)\n {\n this.validators = validators;\n }\n\n public bool IsValid(TEvent @event)\n {\n return this.validators.All(validator => validator.IsValid(@event));\n }\n}\n\ncontainer.Register(typeof(IEventValidator<>), typeof(EventValidator<>));\ncontainer.Register(typeof(IEventValidator<>), typeof(AnotherEventValidator<>));\ncontainer.RegisterDecorator(typeof(IEventValidator<>), typeof(CompositeValidator<>));\n")),(0,o.kt)("h2",{id:"decorating-multiple-services"},"Decorating multiple services"),(0,o.kt)("p",null,"You have the option to organize similar decorating functionalities for different interfaces into the same decorator class.\nIn this example, we would like to validate a given ",(0,o.kt)("inlineCode",{parentName:"p"},"Event")," right before publishing and also before processing. "),(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"public class EventValidator : IEventProcessor, IEventPublisher\n{\n private readonly IEventProcessor processor;\n private readonly IEventPublisher publisher;\n private readonly IEventValidator validator;\n\n public CompositeValidator(IEventProcessor processor, \n IEventPublisher publisher, \n IEventValidator validator)\n {\n this.processor = processor;\n this.publisher = publisher;\n this.validator = validator;\n }\n\n public void ProcessEvent(TEvent @event)\n {\n // validate the event first.\n if (!this.validator.IsValid(@event))\n throw new InvalidEventException();\n\n // if everything is ok, call the processor.\n this.processor.ProcessEvent(@event);\n }\n\n public void PublishEvent(TEvent @event)\n {\n // validate the event first.\n if (!this.validator.IsValid(@event))\n throw new InvalidEventException();\n\n // if everything is ok, call the publisher.\n this.publisher.PublishEvent(@event);\n }\n}\n\ncontainer.Register(typeof(IEventProcessor<>), typeof(EventProcessor<>));\ncontainer.Register(typeof(IEventPublisher<>), typeof(EventPublisher<>));\ncontainer.Register(typeof(IEventValidator<>), typeof(EventValidator<>));\n\n// without specifying the interface type, the container binds this registration to all of its implemented types\ncontainer.RegisterDecorator(typeof(EventValidator<>));\n")),(0,o.kt)("admonition",{type:"info"},(0,o.kt)("p",{parentName:"admonition"},"You can also use the ",(0,o.kt)("a",{parentName:"p",href:"/docs/guides/advanced-registration#binding-to-multiple-services"},"Binding to multiple services")," options.")),(0,o.kt)("h2",{id:"lifetime"},"Lifetime"),(0,o.kt)(a.Z,{mdxType:"CodeDescPanel"},(0,o.kt)("div",null,(0,o.kt)("p",null,"Just as other registrations, decorators also can have their lifetime. It means, in addition to the service's lifetime, all decorator's lifetime will be applied to the wrapped service."),(0,o.kt)("admonition",{type:"note"},(0,o.kt)("p",{parentName:"admonition"},"When you don't set a decorator's lifetime, it'll implicitly inherit the decorated service's lifetime."))),(0,o.kt)("div",null,(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register();\n// singleton decorator will change the transient \n// decorated service's lifetime to singleton.\ncontainer.RegisterDecorator(options => \n options.WithLifetime(Lifetimes.Singleton));\n// Singleton[new ValidatorProcessor()](Transien[new GeneralEventProcessor()]) \nvar processor = container.Resolve(); \n")))),(0,o.kt)("h2",{id:"wrappers"},"Wrappers"),(0,o.kt)(a.Z,{mdxType:"CodeDescPanel"},(0,o.kt)("div",null,(0,o.kt)("p",null,"Decorators are also applied to wrapped services. It means, in addition to the decoration, you can wrap your services in supported ",(0,o.kt)("a",{parentName:"p",href:"/docs/advanced/wrappers-resolvers#wrappers"},"wrappers"),".")),(0,o.kt)("div",null,(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register();\ncontainer.RegisterDecorator();\n// () => new ValidatorProcessor(new GeneralEventProcessor())\nvar processor = container.Resolve>();\n")))),(0,o.kt)("h2",{id:"interception"},"Interception"),(0,o.kt)("p",null,"From the combination of Stashbox's decorator support and ",(0,o.kt)("a",{parentName:"p",href:"http://www.castleproject.org/projects/dynamicproxy/"},"Castle DynamicProxy's")," proxy generator, we can take advantage of the ",(0,o.kt)("a",{parentName:"p",href:"https://en.wikipedia.org/wiki/Aspect-oriented_programming"},"Aspect-Oriented Programming's")," benefits. The following example defines a ",(0,o.kt)("inlineCode",{parentName:"p"},"LoggingInterceptor")," that will log additional messages related to the called service methods."),(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},'public class LoggingInterceptor : IInterceptor \n{\n private readonly ILogger logger;\n\n public LoggingInterceptor(ILogger logger)\n {\n this.logger = logger;\n }\n\n public void Intercept(IInvocation invocation)\n {\n var stopwatch = new Stopwatch();\n stopwatch.Start();\n\n // log before we invoke the intercepted method.\n this.logger.Log($"Method begin: {invocation.GetConcreteMethod().Name}");\n\n // call the intercepted method.\n invocation.Proceed();\n\n // log after we invoked the intercepted method and print how long it ran.\n this.logger.Log($"Method end: {invocation.GetConcreteMethod().Name}, execution duration: {stopwatch.ElapsedMiliseconds} ms");\n }\n}\n\n// create a DefaultProxyBuilder from the DynamicProxy library.\nvar proxyBuilder = new DefaultProxyBuilder();\n\n// build a proxy for the IEventProcessor interface.\nvar eventProcessorProxy = proxyBuilder.CreateInterfaceProxyTypeWithTargetInterface(\n typeof(IEventProcessor), \n new Type[0], \n ProxyGenerationOptions.Default);\n\n// register the logger for LoggingInterceptor.\ncontainer.Register();\n\n// register the service that we will intercept.\ncontainer.Register();\n\n// register the interceptor.\ncontainer.Register();\n\n// register the built proxy as a decorator.\ncontainer.RegisterDecorator(eventProcessorProxy);\n')))}h.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/851a7a9e.d73c6c03.js b/assets/js/851a7a9e.d73c6c03.js deleted file mode 100644 index 12a2e432..00000000 --- a/assets/js/851a7a9e.d73c6c03.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkwebsite=self.webpackChunkwebsite||[]).push([[153],{3905:(e,t,r)=>{r.d(t,{Zo:()=>d,kt:()=>h});var n=r(7294);function o(e,t,r){return t in e?Object.defineProperty(e,t,{value:r,enumerable:!0,configurable:!0,writable:!0}):e[t]=r,e}function a(e,t){var r=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);t&&(n=n.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),r.push.apply(r,n)}return r}function s(e){for(var t=1;t=0||(o[r]=e[r]);return o}(e,t);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(e,r)&&(o[r]=e[r])}return o}var c=n.createContext({}),l=function(e){var t=n.useContext(c),r=t;return e&&(r="function"==typeof e?e(t):s(s({},t),e)),r},d=function(e){var t=l(e.components);return n.createElement(c.Provider,{value:t},e.children)},p="mdxType",u={inlineCode:"code",wrapper:function(e){var t=e.children;return n.createElement(n.Fragment,{},t)}},v=n.forwardRef((function(e,t){var r=e.components,o=e.mdxType,a=e.originalType,c=e.parentName,d=i(e,["components","mdxType","originalType","parentName"]),p=l(r),v=o,h=p["".concat(c,".").concat(v)]||p[v]||u[v]||a;return r?n.createElement(h,s(s({ref:t},d),{},{components:r})):n.createElement(h,s({ref:t},d))}));function h(e,t){var r=arguments,o=t&&t.mdxType;if("string"==typeof e||o){var a=r.length,s=new Array(a);s[0]=v;var i={};for(var c in t)hasOwnProperty.call(t,c)&&(i[c]=t[c]);i.originalType=e,i[p]="string"==typeof e?e:o,s[1]=i;for(var l=2;l{r.d(t,{Z:()=>s});var n=r(7294),o=r(6010);const a="tabItem_Ymn6";function s(e){let{children:t,hidden:r,className:s}=e;return n.createElement("div",{role:"tabpanel",className:(0,o.Z)(a,s),hidden:r},t)}},4866:(e,t,r)=>{r.d(t,{Z:()=>w});var n=r(7462),o=r(7294),a=r(6010),s=r(2466),i=r(6550),c=r(1980),l=r(7392),d=r(12);function p(e){return function(e){return o.Children.map(e,(e=>{if(!e||(0,o.isValidElement)(e)&&function(e){const{props:t}=e;return!!t&&"object"==typeof t&&"value"in t}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}(e).map((e=>{let{props:{value:t,label:r,attributes:n,default:o}}=e;return{value:t,label:r,attributes:n,default:o}}))}function u(e){const{values:t,children:r}=e;return(0,o.useMemo)((()=>{const e=t??p(r);return function(e){const t=(0,l.l)(e,((e,t)=>e.value===t.value));if(t.length>0)throw new Error(`Docusaurus error: Duplicate values "${t.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[t,r])}function v(e){let{value:t,tabValues:r}=e;return r.some((e=>e.value===t))}function h(e){let{queryString:t=!1,groupId:r}=e;const n=(0,i.k6)(),a=function(e){let{queryString:t=!1,groupId:r}=e;if("string"==typeof t)return t;if(!1===t)return null;if(!0===t&&!r)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return r??null}({queryString:t,groupId:r});return[(0,c._X)(a),(0,o.useCallback)((e=>{if(!a)return;const t=new URLSearchParams(n.location.search);t.set(a,e),n.replace({...n.location,search:t.toString()})}),[a,n])]}function g(e){const{defaultValue:t,queryString:r=!1,groupId:n}=e,a=u(e),[s,i]=(0,o.useState)((()=>function(e){let{defaultValue:t,tabValues:r}=e;if(0===r.length)throw new Error("Docusaurus error: the component requires at least one children component");if(t){if(!v({value:t,tabValues:r}))throw new Error(`Docusaurus error: The has a defaultValue "${t}" but none of its children has the corresponding value. Available values are: ${r.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return t}const n=r.find((e=>e.default))??r[0];if(!n)throw new Error("Unexpected error: 0 tabValues");return n.value}({defaultValue:t,tabValues:a}))),[c,l]=h({queryString:r,groupId:n}),[p,g]=function(e){let{groupId:t}=e;const r=function(e){return e?`docusaurus.tab.${e}`:null}(t),[n,a]=(0,d.Nk)(r);return[n,(0,o.useCallback)((e=>{r&&a.set(e)}),[r,a])]}({groupId:n}),m=(()=>{const e=c??p;return v({value:e,tabValues:a})?e:null})();(0,o.useLayoutEffect)((()=>{m&&i(m)}),[m]);return{selectedValue:s,selectValue:(0,o.useCallback)((e=>{if(!v({value:e,tabValues:a}))throw new Error(`Can't select invalid tab value=${e}`);i(e),l(e),g(e)}),[l,g,a]),tabValues:a}}var m=r(2389);const f="tabList__CuJ",E="tabItem_LNqP";function P(e){let{className:t,block:r,selectedValue:i,selectValue:c,tabValues:l}=e;const d=[],{blockElementScrollPositionUntilNextRender:p}=(0,s.o5)(),u=e=>{const t=e.currentTarget,r=d.indexOf(t),n=l[r].value;n!==i&&(p(t),c(n))},v=e=>{let t=null;switch(e.key){case"Enter":u(e);break;case"ArrowRight":{const r=d.indexOf(e.currentTarget)+1;t=d[r]??d[0];break}case"ArrowLeft":{const r=d.indexOf(e.currentTarget)-1;t=d[r]??d[d.length-1];break}}t?.focus()};return o.createElement("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,a.Z)("tabs",{"tabs--block":r},t)},l.map((e=>{let{value:t,label:r,attributes:s}=e;return o.createElement("li",(0,n.Z)({role:"tab",tabIndex:i===t?0:-1,"aria-selected":i===t,key:t,ref:e=>d.push(e),onKeyDown:v,onClick:u},s,{className:(0,a.Z)("tabs__item",E,s?.className,{"tabs__item--active":i===t})}),r??t)})))}function b(e){let{lazy:t,children:r,selectedValue:n}=e;const a=(Array.isArray(r)?r:[r]).filter(Boolean);if(t){const e=a.find((e=>e.props.value===n));return e?(0,o.cloneElement)(e,{className:"margin-top--md"}):null}return o.createElement("div",{className:"margin-top--md"},a.map(((e,t)=>(0,o.cloneElement)(e,{key:t,hidden:e.props.value!==n}))))}function y(e){const t=g(e);return o.createElement("div",{className:(0,a.Z)("tabs-container",f)},o.createElement(P,(0,n.Z)({},e,t)),o.createElement(b,(0,n.Z)({},e,t)))}function w(e){const t=(0,m.Z)();return o.createElement(y,(0,n.Z)({key:String(t)},e))}},8846:(e,t,r)=>{r.d(t,{Z:()=>i});var n=r(7294);const o="codeDescContainer_ie8f",a="desc_jyqI",s="example_eYlF";function i(e){let{children:t}=e,r=n.Children.toArray(t).filter((e=>e));return n.createElement("div",{className:o},n.createElement("div",{className:a},r[0]),n.createElement("div",{className:s},r[1]))}},1212:(e,t,r)=>{r.r(t),r.d(t,{assets:()=>p,contentTitle:()=>l,default:()=>h,frontMatter:()=>c,metadata:()=>d,toc:()=>u});var n=r(7462),o=(r(7294),r(3905)),a=r(8846),s=r(4866),i=r(5162);const c={},l="Decorators",d={unversionedId:"advanced/decorators",id:"advanced/decorators",title:"Decorators",description:"Stashbox supports decorator service registration to take advantage of the Decorator pattern. This pattern is used to extend the functionality of a class without changing its implementation. This is also what the Open\u2013closed principle stands for; services should be open for extension but closed for modification.",source:"@site/docs/advanced/decorators.md",sourceDirName:"advanced",slug:"/advanced/decorators",permalink:"/stashbox/docs/advanced/decorators",draft:!1,editUrl:"https://github.com/z4kn4fein/stashbox/edit/master/docs/docs/advanced/decorators.md",tags:[],version:"current",lastUpdatedBy:"dependabot[bot]",lastUpdatedAt:1697644420,formattedLastUpdatedAt:"Oct 18, 2023",frontMatter:{},sidebar:"docs",previous:{title:"Generics",permalink:"/stashbox/docs/advanced/generics"},next:{title:"Wrappers & resolvers",permalink:"/stashbox/docs/advanced/wrappers-resolvers"}},p={},u=[{value:"Simple use-case",id:"simple-use-case",level:2},{value:"Multiple decorators",id:"multiple-decorators",level:2},{value:"Conditional decoration",id:"conditional-decoration",level:2},{value:"Generic decorators",id:"generic-decorators",level:2},{value:"Composite pattern",id:"composite-pattern",level:2},{value:"Decorating multiple services",id:"decorating-multiple-services",level:2},{value:"Lifetime",id:"lifetime",level:2},{value:"Wrappers",id:"wrappers",level:2},{value:"Interception",id:"interception",level:2}],v={toc:u};function h(e){let{components:t,...r}=e;return(0,o.kt)("wrapper",(0,n.Z)({},v,r,{components:t,mdxType:"MDXLayout"}),(0,o.kt)("h1",{id:"decorators"},"Decorators"),(0,o.kt)("p",null,"Stashbox supports decorator ",(0,o.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-registration--registered-service"},"service registration")," to take advantage of the ",(0,o.kt)("a",{parentName:"p",href:"https://en.wikipedia.org/wiki/Decorator_pattern"},"Decorator pattern"),". This pattern is used to extend the functionality of a class without changing its implementation. This is also what the ",(0,o.kt)("a",{parentName:"p",href:"https://en.wikipedia.org/wiki/Open%E2%80%93closed_principle"},"Open\u2013closed principle")," stands for; services should be open for extension but closed for modification."),(0,o.kt)("h2",{id:"simple-use-case"},"Simple use-case"),(0,o.kt)("p",null,"We define an ",(0,o.kt)("inlineCode",{parentName:"p"},"IEventProcessor")," service used to process ",(0,o.kt)("inlineCode",{parentName:"p"},"Event")," entities. Then we'll decorate this service with additional validation capabilities:"),(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"class Event { }\nclass UpdateEvent : Event { }\n\ninterface IEventProcessor\n{\n void ProcessEvent(Event @event);\n}\n\ninterface IEventValidator\n{\n bool IsValid(Event @event);\n}\n\nclass EventValidator : IEventValidator\n{\n public bool IsValid(Event @event) { /* do the actual validation. */ }\n}\n\nclass GeneralEventProcessor : IEventProcessor\n{\n public void ProcessEvent(Event @event)\n {\n // suppose this method is processing the given event.\n this.DoTheActualProcessing(@event);\n }\n}\n\nclass ValidatorProcessor : IEventProcessor\n{\n private readonly IEventProcessor nextProcessor;\n private readonly IEventValidator eventValidator;\n public ValidatorProcessor(IEventProcessor eventProcessor, IEventValidator eventValidator)\n {\n this.nextProcessor = eventProcessor;\n this.eventValidator = eventValidator;\n }\n\n public void ProcessEvent(Event @event)\n {\n // validate the event first.\n if (!this.eventValidator.IsValid(@event))\n throw new InvalidEventException();\n\n // if everything is ok, call the next processor.\n this.nextProcessor.ProcessEvent(@event);\n }\n}\n\nusing var container = new StashboxContainer();\n\ncontainer.Register();\ncontainer.Register();\ncontainer.RegisterDecorator();\n\n// new ValidatorProcessor(new GeneralEventProcessor(), new EventValidator())\nvar eventProcessor = container.Resolve();\n\n// process the event.\neventProcessor.ProcessEvent(new UpdateEvent());\n")),(0,o.kt)("p",null,"The ",(0,o.kt)("inlineCode",{parentName:"p"},"GeneralEventProcessor")," is an implementation of ",(0,o.kt)("inlineCode",{parentName:"p"},"IEventProcessor")," and does the actual event processing logic. It does not have any other responsibilities. Rather than putting the event validation's burden onto its shoulder, we create a different service for validation purposes. Instead of injecting the validator into the ",(0,o.kt)("inlineCode",{parentName:"p"},"GeneralEventProcessor")," directly, we let another ",(0,o.kt)("inlineCode",{parentName:"p"},"IEventProcessor")," decorate it like an ",(0,o.kt)("em",{parentName:"p"},"event processing pipeline")," that validates the event as a first step."),(0,o.kt)("h2",{id:"multiple-decorators"},"Multiple decorators"),(0,o.kt)(a.Z,{mdxType:"CodeDescPanel"},(0,o.kt)("div",null,(0,o.kt)("p",null,"You have the option to register multiple decorators for a service to extend its functionality."),(0,o.kt)("p",null,"The decoration order will be the same as the registration order of the decorators. The first registered decorator will decorate the service itself. Then, all the subsequent decorators will wrap the already decorated service.")),(0,o.kt)("div",null,(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register();\ncontainer.RegisterDecorator();\ncontainer.RegisterDecorator();\n\n// new ValidatorProcessor(new LoggerProcessor(new GeneralProcessor()));\nvar processor = container.Resolve(); \n")))),(0,o.kt)("h2",{id:"conditional-decoration"},"Conditional decoration"),(0,o.kt)("p",null,"With ",(0,o.kt)("a",{parentName:"p",href:"/docs/guides/service-resolution#conditional-resolution"},"conditional resolution")," you can control which decorator should be selected to decorate a given service."),(0,o.kt)(s.Z,{mdxType:"Tabs"},(0,o.kt)(i.Z,{value:"Decoretee",label:"Decoretee",mdxType:"TabItem"},(0,o.kt)("p",null,"You have the option to set which decorator should be selected for a given implementation. For a single type filter, you can use the ",(0,o.kt)("inlineCode",{parentName:"p"},".WhenDecoratedServiceIs()")," configuration option. To select more types, you can use the more generic ",(0,o.kt)("inlineCode",{parentName:"p"},".When()")," option. "),(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register();\ncontainer.Register();\n\ncontainer.RegisterDecorator(options => options\n // select when CustomProcessor or GeneralProcessor is resolved.\n .WhenDecoratedServiceIs()\n .WhenDecoratedServiceIs());\n\ncontainer.RegisterDecorator(options => options\n // select only when GeneralProcessor is resolved.\n .WhenDecoratedServiceIs());\n\n// [\n// new ValidatorProcessor(new LoggerProcessor(new GeneralProcessor())),\n// new LoggerProcessor(new CustomProcessor())\n// ]\nvar processors = container.ResolveAll();\n"))),(0,o.kt)(i.Z,{value:"Named",label:"Named",mdxType:"TabItem"},(0,o.kt)("p",null,"You can filter for service names to control the decorator selection."),(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register("General");\ncontainer.Register("Custom");\n\ncontainer.RegisterDecorator(options => options\n // select when CustomProcessor or GeneralProcessor is resolved.\n .WhenDecoratedServiceIs("General")\n .WhenDecoratedServiceIs("Custom"));\n\ncontainer.RegisterDecorator(options => options\n // select only when GeneralProcessor is resolved.\n .WhenDecoratedServiceIs("General"));\n\n// new ValidatorProcessor(new LoggerProcessor(new GeneralProcessor()))\nvar general = container.Resolve("General");\n\n// new LoggerProcessor(new CustomProcessor())\nvar custom = container.Resolve("Custom");\n'))),(0,o.kt)(i.Z,{value:"Attribute",label:"Attribute",mdxType:"TabItem"},(0,o.kt)("p",null,"You can use your custom attributes to control the decorator selection. With ",(0,o.kt)("strong",{parentName:"p"},"class attributes"),", you can mark your classes for decoration."),(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"class LogAttribute : Attribute { }\nclass ValidateAttribute : Attribute { }\n\n[Log, Validate]\nclass GeneralProcessor : IEventProcessor { }\n\n[Log]\nclass CustomProcessor : IEventProcessor { }\n\ncontainer.Register();\ncontainer.Register();\n\ncontainer.RegisterDecorator(options => options\n // select when the resolving class has 'LogAttribute'.\n .WhenDecoratedServiceHas());\n\ncontainer.RegisterDecorator(options => options\n // select when the resolving class has 'ValidateAttribute'.\n .WhenDecoratedServiceHas());\n\n// [\n// new ValidatorProcessor(new LoggerProcessor(new GeneralProcessor())),\n// new LoggerProcessor(new CustomProcessor())\n// ]\nvar processors = container.ResolveAll();\n")),(0,o.kt)("p",null,"You can also mark your dependencies for decoration with ",(0,o.kt)("strong",{parentName:"p"},"property / field / parameter attributes"),". "),(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"class LogAttribute : Attribute { }\nclass ValidateAttribute : Attribute { }\n\nclass ProcessorExecutor\n{\n public ProcessorExecutor([Log, Validate]IEventProcessor eventProcessor)\n { }\n}\n\ncontainer.Register();\ncontainer.Register();\n\ncontainer.RegisterDecorator(options => options\n // select when the resolving dependency has 'LogAttribute'.\n .WhenHas());\n\ncontainer.RegisterDecorator(options => options\n // select when the resolving dependency has 'ValidateAttribute'.\n .WhenHas());\n\n// new ProcessorExecutor(new ValidatorProcessor(new LoggerProcessor(new GeneralProcessor())))\nvar executor = container.ResolveAll();\n")))),(0,o.kt)("h2",{id:"generic-decorators"},"Generic decorators"),(0,o.kt)("p",null,"Stashbox supports the registration of open-generic decorators, which allows the extension of open-generic services.\nInspection of ",(0,o.kt)("a",{parentName:"p",href:"/docs/advanced/generics#generic-constraints"},"generic parameter constraints")," and ",(0,o.kt)("a",{parentName:"p",href:"/docs/advanced/generics#variance"},"variance handling")," is supported on generic decorators also."),(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"interface IEventProcessor\n{\n void ProcessEvent(TEvent @event);\n}\n\nclass GeneralEventProcessor : IEventProcessor\n{\n public void ProcessEvent(TEvent @event) { /* suppose this method is processing the given event.*/ }\n}\n\nclass ValidatorProcessor : IEventProcessor\n{\n private readonly IEventProcessor nextProcessor;\n\n public ValidatorProcessor(IEventProcessor eventProcessor)\n {\n this.nextProcessor = eventProcessor;\n }\n\n public void ProcessEvent(TEvent @event)\n {\n // validate the event first.\n if (!this.IsValid(@event))\n throw new InvalidEventException();\n\n // if everything is ok, call the next processor.\n this.nextProcessor.ProcessEvent(@event);\n }\n}\n\nusing var container = new StashboxContainer();\ncontainer.Register(typeof(IEventProcessor<>), typeof(GeneralEventProcessor<>));\ncontainer.RegisterDecorator(typeof(IEventProcessor<>), typeof(ValidatorProcessor<>));\n\n// new ValidatorProcessor(new GeneralEventProcessor())\nvar eventProcessor = container.Resolve>();\n\n// process the event.\neventProcessor.ProcessEvent(new UpdateEvent());\n")),(0,o.kt)("h2",{id:"composite-pattern"},"Composite pattern"),(0,o.kt)("p",null,"The ",(0,o.kt)("a",{parentName:"p",href:"https://en.wikipedia.org/wiki/Composite_pattern"},"Composite pattern")," allows a group of objects to be treated the same way as a single instance of the same type. It's useful when you want to use the functionality of multiple instances behind the same interface. You can achieve this by registering a decorator that takes a collection of the same service as a dependency. "),(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"public class CompositeValidator : IEventValidator\n{\n private readonly IEnumerable> validators;\n\n public CompositeValidator(IEnumerable> validators)\n {\n this.validators = validators;\n }\n\n public bool IsValid(TEvent @event)\n {\n return this.validators.All(validator => validator.IsValid(@event));\n }\n}\n\ncontainer.Register(typeof(IEventValidator<>), typeof(EventValidator<>));\ncontainer.Register(typeof(IEventValidator<>), typeof(AnotherEventValidator<>));\ncontainer.RegisterDecorator(typeof(IEventValidator<>), typeof(CompositeValidator<>));\n")),(0,o.kt)("h2",{id:"decorating-multiple-services"},"Decorating multiple services"),(0,o.kt)("p",null,"You have the option to organize similar decorating functionalities for different interfaces into the same decorator class.\nIn this example, we would like to validate a given ",(0,o.kt)("inlineCode",{parentName:"p"},"Event")," right before publishing and also before processing. "),(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"public class EventValidator : IEventProcessor, IEventPublisher\n{\n private readonly IEventProcessor processor;\n private readonly IEventPublisher publisher;\n private readonly IEventValidator validator;\n\n public CompositeValidator(IEventProcessor processor, \n IEventPublisher publisher, \n IEventValidator validator)\n {\n this.processor = processor;\n this.publisher = publisher;\n this.validator = validator;\n }\n\n public void ProcessEvent(TEvent @event)\n {\n // validate the event first.\n if (!this.validator.IsValid(@event))\n throw new InvalidEventException();\n\n // if everything is ok, call the processor.\n this.processor.ProcessEvent(@event);\n }\n\n public void PublishEvent(TEvent @event)\n {\n // validate the event first.\n if (!this.validator.IsValid(@event))\n throw new InvalidEventException();\n\n // if everything is ok, call the publisher.\n this.publisher.PublishEvent(@event);\n }\n}\n\ncontainer.Register(typeof(IEventProcessor<>), typeof(EventProcessor<>));\ncontainer.Register(typeof(IEventPublisher<>), typeof(EventPublisher<>));\ncontainer.Register(typeof(IEventValidator<>), typeof(EventValidator<>));\n\n// without specifying the interface type, the container binds this registration to all of its implemented types\ncontainer.RegisterDecorator(typeof(EventValidator<>));\n")),(0,o.kt)("admonition",{type:"info"},(0,o.kt)("p",{parentName:"admonition"},"You can also use the ",(0,o.kt)("a",{parentName:"p",href:"/docs/guides/advanced-registration#binding-to-multiple-services"},"Binding to multiple services")," options.")),(0,o.kt)("h2",{id:"lifetime"},"Lifetime"),(0,o.kt)(a.Z,{mdxType:"CodeDescPanel"},(0,o.kt)("div",null,(0,o.kt)("p",null,"Just as other registrations, decorators also can have their lifetime. It means, in addition to the service's lifetime, all decorator's lifetime will be applied to the wrapped service."),(0,o.kt)("admonition",{type:"note"},(0,o.kt)("p",{parentName:"admonition"},"When you don't set a decorator's lifetime, it'll implicitly inherit the decorated service's lifetime."))),(0,o.kt)("div",null,(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register();\n// singleton decorator will change the transient \n// decorated service's lifetime to singleton.\ncontainer.RegisterDecorator(options => \n options.WithLifetime(Lifetimes.Singleton));\n// Singleton[new ValidatorProcessor()](Transien[new GeneralEventProcessor()]) \nvar processor = container.Resolve(); \n")))),(0,o.kt)("h2",{id:"wrappers"},"Wrappers"),(0,o.kt)(a.Z,{mdxType:"CodeDescPanel"},(0,o.kt)("div",null,(0,o.kt)("p",null,"Decorators are also applied to wrapped services. It means, in addition to the decoration, you can wrap your services in supported ",(0,o.kt)("a",{parentName:"p",href:"/docs/advanced/wrappers-resolvers#wrappers"},"wrappers"),".")),(0,o.kt)("div",null,(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register();\ncontainer.RegisterDecorator();\n// () => new ValidatorProcessor(new GeneralEventProcessor())\nvar processor = container.Resolve>();\n")))),(0,o.kt)("h2",{id:"interception"},"Interception"),(0,o.kt)("p",null,"From the combination of Stashbox's decorator support and ",(0,o.kt)("a",{parentName:"p",href:"http://www.castleproject.org/projects/dynamicproxy/"},"Castle DynamicProxy's")," proxy generator, we can take advantage of the ",(0,o.kt)("a",{parentName:"p",href:"https://en.wikipedia.org/wiki/Aspect-oriented_programming"},"Aspect-Oriented Programming's")," benefits. The following example defines a ",(0,o.kt)("inlineCode",{parentName:"p"},"LoggingInterceptor")," that will log additional messages related to the called service methods."),(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-cs"},'public class LoggingInterceptor : IInterceptor \n{\n private readonly ILogger logger;\n\n public LoggingInterceptor(ILogger logger)\n {\n this.logger = logger;\n }\n\n public void Intercept(IInvocation invocation)\n {\n var stopwatch = new Stopwatch();\n stopwatch.Start();\n\n // log before we invoke the intercepted method.\n this.logger.Log($"Method begin: {invocation.GetConcreteMethod().Name}");\n\n // call the intercepted method.\n invocation.Proceed();\n\n // log after we invoked the intercepted method and print how long it ran.\n this.logger.Log($"Method end: {invocation.GetConcreteMethod().Name}, execution duration: {stopwatch.ElapsedMiliseconds} ms");\n }\n}\n\n// create a DefaultProxyBuilder from the DynamicProxy library.\nvar proxyBuilder = new DefaultProxyBuilder();\n\n// build a proxy for the IEventProcessor interface.\nvar eventProcessorProxy = proxyBuilder.CreateInterfaceProxyTypeWithTargetInterface(\n typeof(IEventProcessor), \n new Type[0], \n ProxyGenerationOptions.Default);\n\n// register the logger for LoggingInterceptor.\ncontainer.Register();\n\n// register the service that we will intercept.\ncontainer.Register();\n\n// register the interceptor.\ncontainer.Register();\n\n// register the built proxy as a decorator.\ncontainer.RegisterDecorator(eventProcessorProxy);\n')))}h.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/9ff4038f.5b6e8ac6.js b/assets/js/9ff4038f.5b6e8ac6.js deleted file mode 100644 index 02c54e1e..00000000 --- a/assets/js/9ff4038f.5b6e8ac6.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkwebsite=self.webpackChunkwebsite||[]).push([[353],{3905:(e,t,n)=>{n.d(t,{Zo:()=>u,kt:()=>m});var a=n(7294);function r(e,t,n){return t in e?Object.defineProperty(e,t,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[t]=n,e}function o(e,t){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);t&&(a=a.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),n.push.apply(n,a)}return n}function i(e){for(var t=1;t=0||(r[n]=e[n]);return r}(e,t);if(Object.getOwnPropertySymbols){var o=Object.getOwnPropertySymbols(e);for(a=0;a=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(r[n]=e[n])}return r}var l=a.createContext({}),c=function(e){var t=a.useContext(l),n=t;return e&&(n="function"==typeof e?e(t):i(i({},t),e)),n},u=function(e){var t=c(e.components);return a.createElement(l.Provider,{value:t},e.children)},p="mdxType",d={inlineCode:"code",wrapper:function(e){var t=e.children;return a.createElement(a.Fragment,{},t)}},g=a.forwardRef((function(e,t){var n=e.components,r=e.mdxType,o=e.originalType,l=e.parentName,u=s(e,["components","mdxType","originalType","parentName"]),p=c(n),g=r,m=p["".concat(l,".").concat(g)]||p[g]||d[g]||o;return n?a.createElement(m,i(i({ref:t},u),{},{components:n})):a.createElement(m,i({ref:t},u))}));function m(e,t){var n=arguments,r=t&&t.mdxType;if("string"==typeof e||r){var o=n.length,i=new Array(o);i[0]=g;var s={};for(var l in t)hasOwnProperty.call(t,l)&&(s[l]=t[l]);s.originalType=e,s[p]="string"==typeof e?e:r,i[1]=s;for(var c=2;c{n.d(t,{Z:()=>i});var a=n(7294),r=n(6010);const o="tabItem_Ymn6";function i(e){let{children:t,hidden:n,className:i}=e;return a.createElement("div",{role:"tabpanel",className:(0,r.Z)(o,i),hidden:n},t)}},4866:(e,t,n)=>{n.d(t,{Z:()=>N});var a=n(7462),r=n(7294),o=n(6010),i=n(2466),s=n(6550),l=n(1980),c=n(7392),u=n(12);function p(e){return function(e){return r.Children.map(e,(e=>{if(!e||(0,r.isValidElement)(e)&&function(e){const{props:t}=e;return!!t&&"object"==typeof t&&"value"in t}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}(e).map((e=>{let{props:{value:t,label:n,attributes:a,default:r}}=e;return{value:t,label:n,attributes:a,default:r}}))}function d(e){const{values:t,children:n}=e;return(0,r.useMemo)((()=>{const e=t??p(n);return function(e){const t=(0,c.l)(e,((e,t)=>e.value===t.value));if(t.length>0)throw new Error(`Docusaurus error: Duplicate values "${t.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[t,n])}function g(e){let{value:t,tabValues:n}=e;return n.some((e=>e.value===t))}function m(e){let{queryString:t=!1,groupId:n}=e;const a=(0,s.k6)(),o=function(e){let{queryString:t=!1,groupId:n}=e;if("string"==typeof t)return t;if(!1===t)return null;if(!0===t&&!n)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return n??null}({queryString:t,groupId:n});return[(0,l._X)(o),(0,r.useCallback)((e=>{if(!o)return;const t=new URLSearchParams(a.location.search);t.set(o,e),a.replace({...a.location,search:t.toString()})}),[o,a])]}function h(e){const{defaultValue:t,queryString:n=!1,groupId:a}=e,o=d(e),[i,s]=(0,r.useState)((()=>function(e){let{defaultValue:t,tabValues:n}=e;if(0===n.length)throw new Error("Docusaurus error: the component requires at least one children component");if(t){if(!g({value:t,tabValues:n}))throw new Error(`Docusaurus error: The has a defaultValue "${t}" but none of its children has the corresponding value. Available values are: ${n.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return t}const a=n.find((e=>e.default))??n[0];if(!a)throw new Error("Unexpected error: 0 tabValues");return a.value}({defaultValue:t,tabValues:o}))),[l,c]=m({queryString:n,groupId:a}),[p,h]=function(e){let{groupId:t}=e;const n=function(e){return e?`docusaurus.tab.${e}`:null}(t),[a,o]=(0,u.Nk)(n);return[a,(0,r.useCallback)((e=>{n&&o.set(e)}),[n,o])]}({groupId:a}),b=(()=>{const e=l??p;return g({value:e,tabValues:o})?e:null})();(0,r.useLayoutEffect)((()=>{b&&s(b)}),[b]);return{selectedValue:i,selectValue:(0,r.useCallback)((e=>{if(!g({value:e,tabValues:o}))throw new Error(`Can't select invalid tab value=${e}`);s(e),c(e),h(e)}),[c,h,o]),tabValues:o}}var b=n(2389);const f="tabList__CuJ",v="tabItem_LNqP";function k(e){let{className:t,block:n,selectedValue:s,selectValue:l,tabValues:c}=e;const u=[],{blockElementScrollPositionUntilNextRender:p}=(0,i.o5)(),d=e=>{const t=e.currentTarget,n=u.indexOf(t),a=c[n].value;a!==s&&(p(t),l(a))},g=e=>{let t=null;switch(e.key){case"Enter":d(e);break;case"ArrowRight":{const n=u.indexOf(e.currentTarget)+1;t=u[n]??u[0];break}case"ArrowLeft":{const n=u.indexOf(e.currentTarget)-1;t=u[n]??u[u.length-1];break}}t?.focus()};return r.createElement("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,o.Z)("tabs",{"tabs--block":n},t)},c.map((e=>{let{value:t,label:n,attributes:i}=e;return r.createElement("li",(0,a.Z)({role:"tab",tabIndex:s===t?0:-1,"aria-selected":s===t,key:t,ref:e=>u.push(e),onKeyDown:g,onClick:d},i,{className:(0,o.Z)("tabs__item",v,i?.className,{"tabs__item--active":s===t})}),n??t)})))}function y(e){let{lazy:t,children:n,selectedValue:a}=e;const o=(Array.isArray(n)?n:[n]).filter(Boolean);if(t){const e=o.find((e=>e.props.value===a));return e?(0,r.cloneElement)(e,{className:"margin-top--md"}):null}return r.createElement("div",{className:"margin-top--md"},o.map(((e,t)=>(0,r.cloneElement)(e,{key:t,hidden:e.props.value!==a}))))}function w(e){const t=h(e);return r.createElement("div",{className:(0,o.Z)("tabs-container",f)},r.createElement(k,(0,a.Z)({},e,t)),r.createElement(y,(0,a.Z)({},e,t)))}function N(e){const t=(0,b.Z)();return r.createElement(w,(0,a.Z)({key:String(t)},e))}},5007:(e,t,n)=>{n.r(t),n.d(t,{assets:()=>u,contentTitle:()=>l,default:()=>g,frontMatter:()=>s,metadata:()=>c,toc:()=>p});var a=n(7462),r=(n(7294),n(3905)),o=n(4866),i=n(5162);const s={title:"Introduction"},l=void 0,c={unversionedId:"getting-started/introduction",id:"getting-started/introduction",title:"Introduction",description:"Stashbox and its extensions are distributed via NuGet packages.",source:"@site/docs/getting-started/introduction.md",sourceDirName:"getting-started",slug:"/getting-started/introduction",permalink:"/stashbox/docs/getting-started/introduction",draft:!1,editUrl:"https://github.com/z4kn4fein/stashbox/edit/master/docs/docs/getting-started/introduction.md",tags:[],version:"current",lastUpdatedBy:"dependabot[bot]",lastUpdatedAt:1697644420,formattedLastUpdatedAt:"Oct 18, 2023",frontMatter:{title:"Introduction"},sidebar:"docs",previous:{title:"Overview",permalink:"/stashbox/docs/getting-started/overview"},next:{title:"Glossary",permalink:"/stashbox/docs/getting-started/glossary"}},u={},p=[{value:"Usage",id:"usage",level:2},{value:"How it works?",id:"how-it-works",level:2},{value:"Example",id:"example",level:2}],d={toc:p};function g(e){let{components:t,...n}=e;return(0,r.kt)("wrapper",(0,a.Z)({},d,n,{components:t,mdxType:"MDXLayout"}),(0,r.kt)("p",null,"Stashbox and its extensions are distributed via ",(0,r.kt)("a",{parentName:"p",href:"https://www.nuget.org/packages?q=stashbox"},"NuGet")," packages."),(0,r.kt)(o.Z,{mdxType:"Tabs"},(0,r.kt)(i.Z,{value:"Package Manager",label:"Package Manager",mdxType:"TabItem"},(0,r.kt)("p",null,"You can install the package by typing the following into the Package Manager Console:"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-powershell"},"Install-Package Stashbox -Version 5.12.2\n"))),(0,r.kt)(i.Z,{value:"dotnet CLI",label:"dotnet CLI",mdxType:"TabItem"},(0,r.kt)("p",null,"You can install the package by using the dotnet cli:"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-bash"},"dotnet add package Stashbox --version 5.12.2\n"))),(0,r.kt)(i.Z,{value:"PackageReference",label:"PackageReference",mdxType:"TabItem"},(0,r.kt)("p",null,"You can add the package into the package references of your ",(0,r.kt)("inlineCode",{parentName:"p"},".csproj"),":"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-xml"},'\n')))),(0,r.kt)("h2",{id:"usage"},"Usage"),(0,r.kt)("p",null,"The general idea behind using Stashbox is that you structure your code with loosely coupled components with the ",(0,r.kt)("a",{parentName:"p",href:"https://en.wikipedia.org/wiki/Dependency_inversion_principle"},"Dependency Inversion Principle"),", ",(0,r.kt)("a",{parentName:"p",href:"https://en.wikipedia.org/wiki/Inversion_of_control"},"Inversion Of Control")," and ",(0,r.kt)("a",{parentName:"p",href:"https://martinfowler.com/articles/injection.html"},"Dependency Injection")," in mind. "),(0,r.kt)("p",null,"Rather than letting the services instantiate their dependencies inside themselves, inject the dependencies on construction. Also, rather than creating the object hierarchy manually, you can use a Dependency Injection framework that does the work for you. That's why you are here, I suppose. \ud83d\ude42"),(0,r.kt)("p",null,"To achieve the most efficient usage of Stashbox, you should follow these steps:"),(0,r.kt)("ul",null,(0,r.kt)("li",{parentName:"ul"},"At the startup of your application, instantiate a ",(0,r.kt)("inlineCode",{parentName:"li"},"StashboxContainer"),"."),(0,r.kt)("li",{parentName:"ul"},"Register your services into the container."),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"/docs/diagnostics/validation"},"Validate")," the state of the container and the registrations with the ",(0,r.kt)("inlineCode",{parentName:"li"},".Validate()")," method. ",(0,r.kt)("em",{parentName:"li"},"(Optional)")),(0,r.kt)("li",{parentName:"ul"},"During the lifetime of the application, use the container to resolve your services."),(0,r.kt)("li",{parentName:"ul"},"Create ",(0,r.kt)("a",{parentName:"li",href:"/docs/guides/scopes"},"scopes")," and use them to resolve your services. ",(0,r.kt)("em",{parentName:"li"},"(Optional)")),(0,r.kt)("li",{parentName:"ul"},"On application exit, call the container's ",(0,r.kt)("inlineCode",{parentName:"li"},".Dispose()")," or ",(0,r.kt)("inlineCode",{parentName:"li"},".DisposeAsync()")," method to clean up the resources. ",(0,r.kt)("em",{parentName:"li"},"(Optional)"))),(0,r.kt)("admonition",{type:"caution"},(0,r.kt)("p",{parentName:"admonition"},"You should create only a single instance from ",(0,r.kt)("inlineCode",{parentName:"p"},"StashboxContainer")," (plus child containers if you use them) per application domain. ",(0,r.kt)("inlineCode",{parentName:"p"},"StashboxContainer")," instances are thread-safe. Do not create new container instances continuously, such action will bypass the container's internal delegate cache and could lead to performance degradation. ")),(0,r.kt)("h2",{id:"how-it-works"},"How it works?"),(0,r.kt)("p",null,"Stashbox builds and maintains a collection of ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-registration--registered-service"},"registered services"),". When a service is requested for resolution, Stashbox starts looking for a matching registration that has the same ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type")," as the type that was requested. If it finds one, the container initiates a scan on the ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"implementation type's")," available constructors and selects the one with the most arguments it knows how to resolve by matching argument types to other registrations."),(0,r.kt)("p",null,"When every constructor argument has a companion registration, Stashbox jumps to the first one and continues the same scanning operation. "),(0,r.kt)("p",null,"This process is repeated until every ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#injectable-dependency"},"injectable dependency")," has a matching registration in the ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#resolution-tree"},"resolution tree"),". At the end of the process, Stashbox will have each dependency node built-up in a hierarchical object structure to instantiate the initially requested service object."),(0,r.kt)("h2",{id:"example"},"Example"),(0,r.kt)("p",null,"Let's see a quick example. We have three services ",(0,r.kt)("inlineCode",{parentName:"p"},"DbBackup"),", ",(0,r.kt)("inlineCode",{parentName:"p"},"MessageBus")," and ",(0,r.kt)("inlineCode",{parentName:"p"},"ConsoleLogger"),". ",(0,r.kt)("inlineCode",{parentName:"p"},"DbBackup")," has a dependency on ",(0,r.kt)("inlineCode",{parentName:"p"},"IEventBroadcaster")," (implemented by ",(0,r.kt)("inlineCode",{parentName:"p"},"MessageBus"),") and ",(0,r.kt)("inlineCode",{parentName:"p"},"ILogger")," (implemented by ",(0,r.kt)("inlineCode",{parentName:"p"},"ConsoleLogger"),"), ",(0,r.kt)("inlineCode",{parentName:"p"},"MessageBus")," also depending on an ",(0,r.kt)("inlineCode",{parentName:"p"},"ILogger"),":"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},'public interface IJob { void DoTheJob(); }\n\npublic interface ILogger { void Log(string message); }\n\npublic interface IEventBroadcaster { void Broadcast(IEvent @event); }\n\n\npublic class ConsoleLogger : ILogger\n{\n public void Log(string message) => Console.WriteLine(message);\n}\n\npublic class MessageBus : IEventBroadcaster\n{\n private readonly ILogger logger;\n\n public MessageBus(ILogger logger)\n {\n this.logger = logger;\n }\n\n void Broadcast(IEvent @event) \n {\n this.logger.Log($"Sending event to bus: {@event.Name}");\n // Do the actual event broadcast.\n }\n}\n\npublic class DbBackup : IJob\n{\n private readonly ILogger logger;\n private readonly IEventBroadcaster eventBroadcaster;\n\n public DbBackup(ILogger logger, IEventBroadcaster eventBroadcaster)\n {\n this.logger = logger;\n this.eventBroadcaster = eventBroadcaster;\n }\n\n public void DoTheJob() \n {\n this.logger.Log("Backing up!");\n // Do the actual backup.\n this.eventBroadcaster.Broadcast(new DbBackupCompleted());\n } \n}\n')),(0,r.kt)("admonition",{type:"info"},(0,r.kt)("p",{parentName:"admonition"},"By depending only on interfaces, you decouple your services from concrete implementations. This gives you the flexibility of a more comfortable implementation replacement and also isolates your components from each other. For example, unit testing benefits a lot from the possibility of replacing real implementations with mocks.")),(0,r.kt)("p",null,"The example above configured with Stashbox in a Console Application:"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"using Stashbox;\nusing System;\n\nnamespace Example\n{\n public class Program\n {\n private static readonly IStashboxContainer container;\n\n static Program()\n {\n // 1. Create container\n container = new StashboxContainer();\n\n // 2. Register your services\n container.RegisterSingleton();\n container.Register();\n container.Register();\n\n // 3. Validate the configuration.\n container.Validate();\n }\n\n static void Main(string[] args)\n {\n // 4. Resolve and use your service\n var job = container.Resolve();\n job.DoTheJob();\n }\n }\n}\n")))}g.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/9ff4038f.abdc75cc.js b/assets/js/9ff4038f.abdc75cc.js new file mode 100644 index 00000000..b22ecc36 --- /dev/null +++ b/assets/js/9ff4038f.abdc75cc.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkwebsite=self.webpackChunkwebsite||[]).push([[353],{3905:(e,t,n)=>{n.d(t,{Zo:()=>u,kt:()=>m});var a=n(7294);function r(e,t,n){return t in e?Object.defineProperty(e,t,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[t]=n,e}function o(e,t){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);t&&(a=a.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),n.push.apply(n,a)}return n}function i(e){for(var t=1;t=0||(r[n]=e[n]);return r}(e,t);if(Object.getOwnPropertySymbols){var o=Object.getOwnPropertySymbols(e);for(a=0;a=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(r[n]=e[n])}return r}var l=a.createContext({}),c=function(e){var t=a.useContext(l),n=t;return e&&(n="function"==typeof e?e(t):i(i({},t),e)),n},u=function(e){var t=c(e.components);return a.createElement(l.Provider,{value:t},e.children)},p="mdxType",d={inlineCode:"code",wrapper:function(e){var t=e.children;return a.createElement(a.Fragment,{},t)}},g=a.forwardRef((function(e,t){var n=e.components,r=e.mdxType,o=e.originalType,l=e.parentName,u=s(e,["components","mdxType","originalType","parentName"]),p=c(n),g=r,m=p["".concat(l,".").concat(g)]||p[g]||d[g]||o;return n?a.createElement(m,i(i({ref:t},u),{},{components:n})):a.createElement(m,i({ref:t},u))}));function m(e,t){var n=arguments,r=t&&t.mdxType;if("string"==typeof e||r){var o=n.length,i=new Array(o);i[0]=g;var s={};for(var l in t)hasOwnProperty.call(t,l)&&(s[l]=t[l]);s.originalType=e,s[p]="string"==typeof e?e:r,i[1]=s;for(var c=2;c{n.d(t,{Z:()=>i});var a=n(7294),r=n(6010);const o="tabItem_Ymn6";function i(e){let{children:t,hidden:n,className:i}=e;return a.createElement("div",{role:"tabpanel",className:(0,r.Z)(o,i),hidden:n},t)}},4866:(e,t,n)=>{n.d(t,{Z:()=>N});var a=n(7462),r=n(7294),o=n(6010),i=n(2466),s=n(6550),l=n(1980),c=n(7392),u=n(12);function p(e){return function(e){return r.Children.map(e,(e=>{if(!e||(0,r.isValidElement)(e)&&function(e){const{props:t}=e;return!!t&&"object"==typeof t&&"value"in t}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}(e).map((e=>{let{props:{value:t,label:n,attributes:a,default:r}}=e;return{value:t,label:n,attributes:a,default:r}}))}function d(e){const{values:t,children:n}=e;return(0,r.useMemo)((()=>{const e=t??p(n);return function(e){const t=(0,c.l)(e,((e,t)=>e.value===t.value));if(t.length>0)throw new Error(`Docusaurus error: Duplicate values "${t.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[t,n])}function g(e){let{value:t,tabValues:n}=e;return n.some((e=>e.value===t))}function m(e){let{queryString:t=!1,groupId:n}=e;const a=(0,s.k6)(),o=function(e){let{queryString:t=!1,groupId:n}=e;if("string"==typeof t)return t;if(!1===t)return null;if(!0===t&&!n)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return n??null}({queryString:t,groupId:n});return[(0,l._X)(o),(0,r.useCallback)((e=>{if(!o)return;const t=new URLSearchParams(a.location.search);t.set(o,e),a.replace({...a.location,search:t.toString()})}),[o,a])]}function h(e){const{defaultValue:t,queryString:n=!1,groupId:a}=e,o=d(e),[i,s]=(0,r.useState)((()=>function(e){let{defaultValue:t,tabValues:n}=e;if(0===n.length)throw new Error("Docusaurus error: the component requires at least one children component");if(t){if(!g({value:t,tabValues:n}))throw new Error(`Docusaurus error: The has a defaultValue "${t}" but none of its children has the corresponding value. Available values are: ${n.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return t}const a=n.find((e=>e.default))??n[0];if(!a)throw new Error("Unexpected error: 0 tabValues");return a.value}({defaultValue:t,tabValues:o}))),[l,c]=m({queryString:n,groupId:a}),[p,h]=function(e){let{groupId:t}=e;const n=function(e){return e?`docusaurus.tab.${e}`:null}(t),[a,o]=(0,u.Nk)(n);return[a,(0,r.useCallback)((e=>{n&&o.set(e)}),[n,o])]}({groupId:a}),b=(()=>{const e=l??p;return g({value:e,tabValues:o})?e:null})();(0,r.useLayoutEffect)((()=>{b&&s(b)}),[b]);return{selectedValue:i,selectValue:(0,r.useCallback)((e=>{if(!g({value:e,tabValues:o}))throw new Error(`Can't select invalid tab value=${e}`);s(e),c(e),h(e)}),[c,h,o]),tabValues:o}}var b=n(2389);const f="tabList__CuJ",v="tabItem_LNqP";function k(e){let{className:t,block:n,selectedValue:s,selectValue:l,tabValues:c}=e;const u=[],{blockElementScrollPositionUntilNextRender:p}=(0,i.o5)(),d=e=>{const t=e.currentTarget,n=u.indexOf(t),a=c[n].value;a!==s&&(p(t),l(a))},g=e=>{let t=null;switch(e.key){case"Enter":d(e);break;case"ArrowRight":{const n=u.indexOf(e.currentTarget)+1;t=u[n]??u[0];break}case"ArrowLeft":{const n=u.indexOf(e.currentTarget)-1;t=u[n]??u[u.length-1];break}}t?.focus()};return r.createElement("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,o.Z)("tabs",{"tabs--block":n},t)},c.map((e=>{let{value:t,label:n,attributes:i}=e;return r.createElement("li",(0,a.Z)({role:"tab",tabIndex:s===t?0:-1,"aria-selected":s===t,key:t,ref:e=>u.push(e),onKeyDown:g,onClick:d},i,{className:(0,o.Z)("tabs__item",v,i?.className,{"tabs__item--active":s===t})}),n??t)})))}function y(e){let{lazy:t,children:n,selectedValue:a}=e;const o=(Array.isArray(n)?n:[n]).filter(Boolean);if(t){const e=o.find((e=>e.props.value===a));return e?(0,r.cloneElement)(e,{className:"margin-top--md"}):null}return r.createElement("div",{className:"margin-top--md"},o.map(((e,t)=>(0,r.cloneElement)(e,{key:t,hidden:e.props.value!==a}))))}function w(e){const t=h(e);return r.createElement("div",{className:(0,o.Z)("tabs-container",f)},r.createElement(k,(0,a.Z)({},e,t)),r.createElement(y,(0,a.Z)({},e,t)))}function N(e){const t=(0,b.Z)();return r.createElement(w,(0,a.Z)({key:String(t)},e))}},5007:(e,t,n)=>{n.r(t),n.d(t,{assets:()=>u,contentTitle:()=>l,default:()=>g,frontMatter:()=>s,metadata:()=>c,toc:()=>p});var a=n(7462),r=(n(7294),n(3905)),o=n(4866),i=n(5162);const s={title:"Introduction"},l=void 0,c={unversionedId:"getting-started/introduction",id:"getting-started/introduction",title:"Introduction",description:"Stashbox and its extensions are distributed via NuGet packages.",source:"@site/docs/getting-started/introduction.md",sourceDirName:"getting-started",slug:"/getting-started/introduction",permalink:"/stashbox/docs/getting-started/introduction",draft:!1,editUrl:"https://github.com/z4kn4fein/stashbox/edit/master/docs/docs/getting-started/introduction.md",tags:[],version:"current",lastUpdatedBy:"dependabot[bot]",lastUpdatedAt:1699538162,formattedLastUpdatedAt:"Nov 9, 2023",frontMatter:{title:"Introduction"},sidebar:"docs",previous:{title:"Overview",permalink:"/stashbox/docs/getting-started/overview"},next:{title:"Glossary",permalink:"/stashbox/docs/getting-started/glossary"}},u={},p=[{value:"Usage",id:"usage",level:2},{value:"How it works?",id:"how-it-works",level:2},{value:"Example",id:"example",level:2}],d={toc:p};function g(e){let{components:t,...n}=e;return(0,r.kt)("wrapper",(0,a.Z)({},d,n,{components:t,mdxType:"MDXLayout"}),(0,r.kt)("p",null,"Stashbox and its extensions are distributed via ",(0,r.kt)("a",{parentName:"p",href:"https://www.nuget.org/packages?q=stashbox"},"NuGet")," packages."),(0,r.kt)(o.Z,{mdxType:"Tabs"},(0,r.kt)(i.Z,{value:"Package Manager",label:"Package Manager",mdxType:"TabItem"},(0,r.kt)("p",null,"You can install the package by typing the following into the Package Manager Console:"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-powershell"},"Install-Package Stashbox -Version 5.12.2\n"))),(0,r.kt)(i.Z,{value:"dotnet CLI",label:"dotnet CLI",mdxType:"TabItem"},(0,r.kt)("p",null,"You can install the package by using the dotnet cli:"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-bash"},"dotnet add package Stashbox --version 5.12.2\n"))),(0,r.kt)(i.Z,{value:"PackageReference",label:"PackageReference",mdxType:"TabItem"},(0,r.kt)("p",null,"You can add the package into the package references of your ",(0,r.kt)("inlineCode",{parentName:"p"},".csproj"),":"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-xml"},'\n')))),(0,r.kt)("h2",{id:"usage"},"Usage"),(0,r.kt)("p",null,"The general idea behind using Stashbox is that you structure your code with loosely coupled components with the ",(0,r.kt)("a",{parentName:"p",href:"https://en.wikipedia.org/wiki/Dependency_inversion_principle"},"Dependency Inversion Principle"),", ",(0,r.kt)("a",{parentName:"p",href:"https://en.wikipedia.org/wiki/Inversion_of_control"},"Inversion Of Control")," and ",(0,r.kt)("a",{parentName:"p",href:"https://martinfowler.com/articles/injection.html"},"Dependency Injection")," in mind. "),(0,r.kt)("p",null,"Rather than letting the services instantiate their dependencies inside themselves, inject the dependencies on construction. Also, rather than creating the object hierarchy manually, you can use a Dependency Injection framework that does the work for you. That's why you are here, I suppose. \ud83d\ude42"),(0,r.kt)("p",null,"To achieve the most efficient usage of Stashbox, you should follow these steps:"),(0,r.kt)("ul",null,(0,r.kt)("li",{parentName:"ul"},"At the startup of your application, instantiate a ",(0,r.kt)("inlineCode",{parentName:"li"},"StashboxContainer"),"."),(0,r.kt)("li",{parentName:"ul"},"Register your services into the container."),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"/docs/diagnostics/validation"},"Validate")," the state of the container and the registrations with the ",(0,r.kt)("inlineCode",{parentName:"li"},".Validate()")," method. ",(0,r.kt)("em",{parentName:"li"},"(Optional)")),(0,r.kt)("li",{parentName:"ul"},"During the lifetime of the application, use the container to resolve your services."),(0,r.kt)("li",{parentName:"ul"},"Create ",(0,r.kt)("a",{parentName:"li",href:"/docs/guides/scopes"},"scopes")," and use them to resolve your services. ",(0,r.kt)("em",{parentName:"li"},"(Optional)")),(0,r.kt)("li",{parentName:"ul"},"On application exit, call the container's ",(0,r.kt)("inlineCode",{parentName:"li"},".Dispose()")," or ",(0,r.kt)("inlineCode",{parentName:"li"},".DisposeAsync()")," method to clean up the resources. ",(0,r.kt)("em",{parentName:"li"},"(Optional)"))),(0,r.kt)("admonition",{type:"caution"},(0,r.kt)("p",{parentName:"admonition"},"You should create only a single instance from ",(0,r.kt)("inlineCode",{parentName:"p"},"StashboxContainer")," (plus child containers if you use them) per application domain. ",(0,r.kt)("inlineCode",{parentName:"p"},"StashboxContainer")," instances are thread-safe. Do not create new container instances continuously, such action will bypass the container's internal delegate cache and could lead to performance degradation. ")),(0,r.kt)("h2",{id:"how-it-works"},"How it works?"),(0,r.kt)("p",null,"Stashbox builds and maintains a collection of ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-registration--registered-service"},"registered services"),". When a service is requested for resolution, Stashbox starts looking for a matching registration that has the same ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type")," as the type that was requested. If it finds one, the container initiates a scan on the ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"implementation type's")," available constructors and selects the one with the most arguments it knows how to resolve by matching argument types to other registrations."),(0,r.kt)("p",null,"When every constructor argument has a companion registration, Stashbox jumps to the first one and continues the same scanning operation. "),(0,r.kt)("p",null,"This process is repeated until every ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#injectable-dependency"},"injectable dependency")," has a matching registration in the ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#resolution-tree"},"resolution tree"),". At the end of the process, Stashbox will have each dependency node built-up in a hierarchical object structure to instantiate the initially requested service object."),(0,r.kt)("h2",{id:"example"},"Example"),(0,r.kt)("p",null,"Let's see a quick example. We have three services ",(0,r.kt)("inlineCode",{parentName:"p"},"DbBackup"),", ",(0,r.kt)("inlineCode",{parentName:"p"},"MessageBus")," and ",(0,r.kt)("inlineCode",{parentName:"p"},"ConsoleLogger"),". ",(0,r.kt)("inlineCode",{parentName:"p"},"DbBackup")," has a dependency on ",(0,r.kt)("inlineCode",{parentName:"p"},"IEventBroadcaster")," (implemented by ",(0,r.kt)("inlineCode",{parentName:"p"},"MessageBus"),") and ",(0,r.kt)("inlineCode",{parentName:"p"},"ILogger")," (implemented by ",(0,r.kt)("inlineCode",{parentName:"p"},"ConsoleLogger"),"), ",(0,r.kt)("inlineCode",{parentName:"p"},"MessageBus")," also depending on an ",(0,r.kt)("inlineCode",{parentName:"p"},"ILogger"),":"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},'public interface IJob { void DoTheJob(); }\n\npublic interface ILogger { void Log(string message); }\n\npublic interface IEventBroadcaster { void Broadcast(IEvent @event); }\n\n\npublic class ConsoleLogger : ILogger\n{\n public void Log(string message) => Console.WriteLine(message);\n}\n\npublic class MessageBus : IEventBroadcaster\n{\n private readonly ILogger logger;\n\n public MessageBus(ILogger logger)\n {\n this.logger = logger;\n }\n\n void Broadcast(IEvent @event) \n {\n this.logger.Log($"Sending event to bus: {@event.Name}");\n // Do the actual event broadcast.\n }\n}\n\npublic class DbBackup : IJob\n{\n private readonly ILogger logger;\n private readonly IEventBroadcaster eventBroadcaster;\n\n public DbBackup(ILogger logger, IEventBroadcaster eventBroadcaster)\n {\n this.logger = logger;\n this.eventBroadcaster = eventBroadcaster;\n }\n\n public void DoTheJob() \n {\n this.logger.Log("Backing up!");\n // Do the actual backup.\n this.eventBroadcaster.Broadcast(new DbBackupCompleted());\n } \n}\n')),(0,r.kt)("admonition",{type:"info"},(0,r.kt)("p",{parentName:"admonition"},"By depending only on interfaces, you decouple your services from concrete implementations. This gives you the flexibility of a more comfortable implementation replacement and also isolates your components from each other. For example, unit testing benefits a lot from the possibility of replacing real implementations with mocks.")),(0,r.kt)("p",null,"The example above configured with Stashbox in a Console Application:"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"using Stashbox;\nusing System;\n\nnamespace Example\n{\n public class Program\n {\n private static readonly IStashboxContainer container;\n\n static Program()\n {\n // 1. Create container\n container = new StashboxContainer();\n\n // 2. Register your services\n container.RegisterSingleton();\n container.Register();\n container.Register();\n\n // 3. Validate the configuration.\n container.Validate();\n }\n\n static void Main(string[] args)\n {\n // 4. Resolve and use your service\n var job = container.Resolve();\n job.DoTheJob();\n }\n }\n}\n")))}g.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/a1e0d343.bffda49f.js b/assets/js/a1e0d343.40bd786b.js similarity index 55% rename from assets/js/a1e0d343.bffda49f.js rename to assets/js/a1e0d343.40bd786b.js index afd85c35..b09664be 100644 --- a/assets/js/a1e0d343.bffda49f.js +++ b/assets/js/a1e0d343.40bd786b.js @@ -1 +1 @@ -"use strict";(self.webpackChunkwebsite=self.webpackChunkwebsite||[]).push([[524],{3905:(e,t,n)=>{n.d(t,{Zo:()=>c,kt:()=>g});var a=n(7294);function r(e,t,n){return t in e?Object.defineProperty(e,t,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[t]=n,e}function i(e,t){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);t&&(a=a.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),n.push.apply(n,a)}return n}function l(e){for(var t=1;t=0||(r[n]=e[n]);return r}(e,t);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(e);for(a=0;a=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(r[n]=e[n])}return r}var s=a.createContext({}),u=function(e){var t=a.useContext(s),n=t;return e&&(n="function"==typeof e?e(t):l(l({},t),e)),n},c=function(e){var t=u(e.components);return a.createElement(s.Provider,{value:t},e.children)},d="mdxType",p={inlineCode:"code",wrapper:function(e){var t=e.children;return a.createElement(a.Fragment,{},t)}},m=a.forwardRef((function(e,t){var n=e.components,r=e.mdxType,i=e.originalType,s=e.parentName,c=o(e,["components","mdxType","originalType","parentName"]),d=u(n),m=r,g=d["".concat(s,".").concat(m)]||d[m]||p[m]||i;return n?a.createElement(g,l(l({ref:t},c),{},{components:n})):a.createElement(g,l({ref:t},c))}));function g(e,t){var n=arguments,r=t&&t.mdxType;if("string"==typeof e||r){var i=n.length,l=new Array(i);l[0]=m;var o={};for(var s in t)hasOwnProperty.call(t,s)&&(o[s]=t[s]);o.originalType=e,o[d]="string"==typeof e?e:r,l[1]=o;for(var u=2;u{n.d(t,{Z:()=>l});var a=n(7294),r=n(6010);const i="tabItem_Ymn6";function l(e){let{children:t,hidden:n,className:l}=e;return a.createElement("div",{role:"tabpanel",className:(0,r.Z)(i,l),hidden:n},t)}},4866:(e,t,n)=>{n.d(t,{Z:()=>I});var a=n(7462),r=n(7294),i=n(6010),l=n(2466),o=n(6550),s=n(1980),u=n(7392),c=n(12);function d(e){return function(e){return r.Children.map(e,(e=>{if(!e||(0,r.isValidElement)(e)&&function(e){const{props:t}=e;return!!t&&"object"==typeof t&&"value"in t}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}(e).map((e=>{let{props:{value:t,label:n,attributes:a,default:r}}=e;return{value:t,label:n,attributes:a,default:r}}))}function p(e){const{values:t,children:n}=e;return(0,r.useMemo)((()=>{const e=t??d(n);return function(e){const t=(0,u.l)(e,((e,t)=>e.value===t.value));if(t.length>0)throw new Error(`Docusaurus error: Duplicate values "${t.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[t,n])}function m(e){let{value:t,tabValues:n}=e;return n.some((e=>e.value===t))}function g(e){let{queryString:t=!1,groupId:n}=e;const a=(0,o.k6)(),i=function(e){let{queryString:t=!1,groupId:n}=e;if("string"==typeof t)return t;if(!1===t)return null;if(!0===t&&!n)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return n??null}({queryString:t,groupId:n});return[(0,s._X)(i),(0,r.useCallback)((e=>{if(!i)return;const t=new URLSearchParams(a.location.search);t.set(i,e),a.replace({...a.location,search:t.toString()})}),[i,a])]}function b(e){const{defaultValue:t,queryString:n=!1,groupId:a}=e,i=p(e),[l,o]=(0,r.useState)((()=>function(e){let{defaultValue:t,tabValues:n}=e;if(0===n.length)throw new Error("Docusaurus error: the component requires at least one children component");if(t){if(!m({value:t,tabValues:n}))throw new Error(`Docusaurus error: The has a defaultValue "${t}" but none of its children has the corresponding value. Available values are: ${n.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return t}const a=n.find((e=>e.default))??n[0];if(!a)throw new Error("Unexpected error: 0 tabValues");return a.value}({defaultValue:t,tabValues:i}))),[s,u]=g({queryString:n,groupId:a}),[d,b]=function(e){let{groupId:t}=e;const n=function(e){return e?`docusaurus.tab.${e}`:null}(t),[a,i]=(0,c.Nk)(n);return[a,(0,r.useCallback)((e=>{n&&i.set(e)}),[n,i])]}({groupId:a}),f=(()=>{const e=s??d;return m({value:e,tabValues:i})?e:null})();(0,r.useLayoutEffect)((()=>{f&&o(f)}),[f]);return{selectedValue:l,selectValue:(0,r.useCallback)((e=>{if(!m({value:e,tabValues:i}))throw new Error(`Can't select invalid tab value=${e}`);o(e),u(e),b(e)}),[u,b,i]),tabValues:i}}var f=n(2389);const h="tabList__CuJ",v="tabItem_LNqP";function y(e){let{className:t,block:n,selectedValue:o,selectValue:s,tabValues:u}=e;const c=[],{blockElementScrollPositionUntilNextRender:d}=(0,l.o5)(),p=e=>{const t=e.currentTarget,n=c.indexOf(t),a=u[n].value;a!==o&&(d(t),s(a))},m=e=>{let t=null;switch(e.key){case"Enter":p(e);break;case"ArrowRight":{const n=c.indexOf(e.currentTarget)+1;t=c[n]??c[0];break}case"ArrowLeft":{const n=c.indexOf(e.currentTarget)-1;t=c[n]??c[c.length-1];break}}t?.focus()};return r.createElement("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,i.Z)("tabs",{"tabs--block":n},t)},u.map((e=>{let{value:t,label:n,attributes:l}=e;return r.createElement("li",(0,a.Z)({role:"tab",tabIndex:o===t?0:-1,"aria-selected":o===t,key:t,ref:e=>c.push(e),onKeyDown:m,onClick:p},l,{className:(0,i.Z)("tabs__item",v,l?.className,{"tabs__item--active":o===t})}),n??t)})))}function k(e){let{lazy:t,children:n,selectedValue:a}=e;const i=(Array.isArray(n)?n:[n]).filter(Boolean);if(t){const e=i.find((e=>e.props.value===a));return e?(0,r.cloneElement)(e,{className:"margin-top--md"}):null}return r.createElement("div",{className:"margin-top--md"},i.map(((e,t)=>(0,r.cloneElement)(e,{key:t,hidden:e.props.value!==a}))))}function N(e){const t=b(e);return r.createElement("div",{className:(0,i.Z)("tabs-container",h)},r.createElement(y,(0,a.Z)({},e,t)),r.createElement(k,(0,a.Z)({},e,t)))}function I(e){const t=(0,f.Z)();return r.createElement(N,(0,a.Z)({key:String(t)},e))}},8846:(e,t,n)=>{n.d(t,{Z:()=>o});var a=n(7294);const r="codeDescContainer_ie8f",i="desc_jyqI",l="example_eYlF";function o(e){let{children:t}=e,n=a.Children.toArray(t).filter((e=>e));return a.createElement("div",{className:r},a.createElement("div",{className:i},n[0]),a.createElement("div",{className:l},n[1]))}},9395:(e,t,n)=>{n.r(t),n.d(t,{assets:()=>d,contentTitle:()=>u,default:()=>g,frontMatter:()=>s,metadata:()=>c,toc:()=>p});var a=n(7462),r=(n(7294),n(3905)),i=n(8846),l=n(4866),o=n(5162);const s={},u="Utilities",c={unversionedId:"diagnostics/utilities",id:"diagnostics/utilities",title:"Utilities",description:"Is registered?",source:"@site/docs/diagnostics/utilities.md",sourceDirName:"diagnostics",slug:"/diagnostics/utilities",permalink:"/stashbox/docs/diagnostics/utilities",draft:!1,editUrl:"https://github.com/z4kn4fein/stashbox/edit/master/docs/docs/diagnostics/utilities.md",tags:[],version:"current",lastUpdatedBy:"dependabot[bot]",lastUpdatedAt:1697644420,formattedLastUpdatedAt:"Oct 18, 2023",frontMatter:{},sidebar:"docs",previous:{title:"Validation",permalink:"/stashbox/docs/diagnostics/validation"}},d={},p=[{value:"Is registered?",id:"is-registered",level:2},{value:"Named",id:"named",level:4},{value:"Can resolve?",id:"can-resolve",level:2},{value:"Get all mappings",id:"get-all-mappings",level:2},{value:"Registration diagnostics",id:"registration-diagnostics",level:2}],m={toc:p};function g(e){let{components:t,...n}=e;return(0,r.kt)("wrapper",(0,a.Z)({},m,n,{components:t,mdxType:"MDXLayout"}),(0,r.kt)("h1",{id:"utilities"},"Utilities"),(0,r.kt)("h2",{id:"is-registered"},"Is registered?"),(0,r.kt)(i.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("p",null,"With the ",(0,r.kt)("inlineCode",{parentName:"p"},"IsRegistered()")," function, you can find out whether a service is registered into the container or not."),(0,r.kt)("p",null,"It returns ",(0,r.kt)("inlineCode",{parentName:"p"},"true")," only when the container has a registration with the given type (and name). It only checks the actual container's registrations. For every cases, you should use the ",(0,r.kt)("inlineCode",{parentName:"p"},"CanResolve()")," method.")),(0,r.kt)("div",null,(0,r.kt)(l.Z,{groupId:"generic-runtime-apis",mdxType:"Tabs"},(0,r.kt)(o.Z,{value:"Generic API",label:"Generic API",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"bool isIJobRegistered = container.IsRegistered();\n"))),(0,r.kt)(o.Z,{value:"Runtime type API",label:"Runtime type API",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"bool isIJobRegistered = container.IsRegistered(typeof(IJob));\n"))),(0,r.kt)(o.Z,{value:"Named",label:"Named",mdxType:"TabItem"},(0,r.kt)("h4",{id:"named"},(0,r.kt)("strong",{parentName:"h4"},"Named")),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},'bool isIJobRegistered = container.IsRegistered("DbBackup");\n')))))),(0,r.kt)("h2",{id:"can-resolve"},"Can resolve?"),(0,r.kt)(i.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("p",null,"There might be cases when you are more interested in whether a service is resolvable from the container's actual state rather than finding out whether it's registered."),(0,r.kt)("p",null,(0,r.kt)("inlineCode",{parentName:"p"},"CanResolve()")," returns ",(0,r.kt)("inlineCode",{parentName:"p"},"true")," only when at least one of the following is true:"),(0,r.kt)("ul",null,(0,r.kt)("li",{parentName:"ul"},"The requested type is registered in the current or one of the parent containers."),(0,r.kt)("li",{parentName:"ul"},"The requested type is a closed generic type, and its open generic definition is registered."),(0,r.kt)("li",{parentName:"ul"},"The requested type is a wrapper (",(0,r.kt)("inlineCode",{parentName:"li"},"IEnumerable<>"),", ",(0,r.kt)("inlineCode",{parentName:"li"},"Lazy<>"),", ",(0,r.kt)("inlineCode",{parentName:"li"},"Func<>"),", ",(0,r.kt)("inlineCode",{parentName:"li"},"KeyValuePair<,>"),", ",(0,r.kt)("inlineCode",{parentName:"li"},"ReadOnlyKeyValue<,>"),", ",(0,r.kt)("inlineCode",{parentName:"li"},"Metadata<,>"),", ",(0,r.kt)("inlineCode",{parentName:"li"},"ValueTuple<,>"),", or ",(0,r.kt)("inlineCode",{parentName:"li"},"Tuple<,>"),"), and the underlying type is registered."),(0,r.kt)("li",{parentName:"ul"},"The requested type is not registered, but it's resolvable, and ",(0,r.kt)("a",{parentName:"li",href:"/docs/configuration/container-configuration#unknown-type-resolution"},"unknown type resolution")," is enabled."))),(0,r.kt)("div",null,(0,r.kt)(l.Z,{groupId:"generic-runtime-apis",mdxType:"Tabs"},(0,r.kt)(o.Z,{value:"Generic API",label:"Generic API",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"bool isIJobResolvable = container.CanResolve();\n"))),(0,r.kt)(o.Z,{value:"Runtime type API",label:"Runtime type API",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"bool isIJobResolvable = container.CanResolve(typeof(IJob));\n"))),(0,r.kt)(o.Z,{value:"Named",label:"Named",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},'bool isIJobResolvable = container.CanResolve("DbBackup");\n')))))),(0,r.kt)("h2",{id:"get-all-mappings"},"Get all mappings"),(0,r.kt)(i.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("p",null,"You can get all registrations in a key-value pair collection (where the key is the ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type")," and the value is the actual registration) by calling the ",(0,r.kt)("inlineCode",{parentName:"p"},".GetRegistrationMappings()")," method.")),(0,r.kt)("div",null,(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"IEnumerable> mappings = \n container.GetRegistrationMappings();\n")))),(0,r.kt)("h2",{id:"registration-diagnostics"},"Registration diagnostics"),(0,r.kt)(i.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("p",null,"You can get a much more readable version of the registration mappings by calling the ",(0,r.kt)("inlineCode",{parentName:"p"},".GetRegistrationDiagnostics()")," method."),(0,r.kt)("p",null,(0,r.kt)("inlineCode",{parentName:"p"},"RegistrationDiagnosticsInfo")," has an overridden ",(0,r.kt)("inlineCode",{parentName:"p"},".ToString()")," method that returns the mapping details formatted in a human-readable form.")),(0,r.kt)("div",null,(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register("DbBackupJob");\ncontainer.Register(typeof(IEventHandler<>), typeof(EventHandler<>));\n\nIEnumerable diagnostics = \n container.GetRegistrationDiagnostics();\n\ndiagnostics.ForEach(Console.WriteLine);\n// output:\n// IJob => DbBackup, name: DbBackupJob\n// IEventHandler<> => EventHandler<>, name: null\n')))))}g.isMDXComponent=!0}}]); \ No newline at end of file +"use strict";(self.webpackChunkwebsite=self.webpackChunkwebsite||[]).push([[524],{3905:(e,t,n)=>{n.d(t,{Zo:()=>c,kt:()=>g});var a=n(7294);function r(e,t,n){return t in e?Object.defineProperty(e,t,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[t]=n,e}function i(e,t){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);t&&(a=a.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),n.push.apply(n,a)}return n}function l(e){for(var t=1;t=0||(r[n]=e[n]);return r}(e,t);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(e);for(a=0;a=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(r[n]=e[n])}return r}var s=a.createContext({}),u=function(e){var t=a.useContext(s),n=t;return e&&(n="function"==typeof e?e(t):l(l({},t),e)),n},c=function(e){var t=u(e.components);return a.createElement(s.Provider,{value:t},e.children)},d="mdxType",p={inlineCode:"code",wrapper:function(e){var t=e.children;return a.createElement(a.Fragment,{},t)}},m=a.forwardRef((function(e,t){var n=e.components,r=e.mdxType,i=e.originalType,s=e.parentName,c=o(e,["components","mdxType","originalType","parentName"]),d=u(n),m=r,g=d["".concat(s,".").concat(m)]||d[m]||p[m]||i;return n?a.createElement(g,l(l({ref:t},c),{},{components:n})):a.createElement(g,l({ref:t},c))}));function g(e,t){var n=arguments,r=t&&t.mdxType;if("string"==typeof e||r){var i=n.length,l=new Array(i);l[0]=m;var o={};for(var s in t)hasOwnProperty.call(t,s)&&(o[s]=t[s]);o.originalType=e,o[d]="string"==typeof e?e:r,l[1]=o;for(var u=2;u{n.d(t,{Z:()=>l});var a=n(7294),r=n(6010);const i="tabItem_Ymn6";function l(e){let{children:t,hidden:n,className:l}=e;return a.createElement("div",{role:"tabpanel",className:(0,r.Z)(i,l),hidden:n},t)}},4866:(e,t,n)=>{n.d(t,{Z:()=>I});var a=n(7462),r=n(7294),i=n(6010),l=n(2466),o=n(6550),s=n(1980),u=n(7392),c=n(12);function d(e){return function(e){return r.Children.map(e,(e=>{if(!e||(0,r.isValidElement)(e)&&function(e){const{props:t}=e;return!!t&&"object"==typeof t&&"value"in t}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}(e).map((e=>{let{props:{value:t,label:n,attributes:a,default:r}}=e;return{value:t,label:n,attributes:a,default:r}}))}function p(e){const{values:t,children:n}=e;return(0,r.useMemo)((()=>{const e=t??d(n);return function(e){const t=(0,u.l)(e,((e,t)=>e.value===t.value));if(t.length>0)throw new Error(`Docusaurus error: Duplicate values "${t.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[t,n])}function m(e){let{value:t,tabValues:n}=e;return n.some((e=>e.value===t))}function g(e){let{queryString:t=!1,groupId:n}=e;const a=(0,o.k6)(),i=function(e){let{queryString:t=!1,groupId:n}=e;if("string"==typeof t)return t;if(!1===t)return null;if(!0===t&&!n)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return n??null}({queryString:t,groupId:n});return[(0,s._X)(i),(0,r.useCallback)((e=>{if(!i)return;const t=new URLSearchParams(a.location.search);t.set(i,e),a.replace({...a.location,search:t.toString()})}),[i,a])]}function b(e){const{defaultValue:t,queryString:n=!1,groupId:a}=e,i=p(e),[l,o]=(0,r.useState)((()=>function(e){let{defaultValue:t,tabValues:n}=e;if(0===n.length)throw new Error("Docusaurus error: the component requires at least one children component");if(t){if(!m({value:t,tabValues:n}))throw new Error(`Docusaurus error: The has a defaultValue "${t}" but none of its children has the corresponding value. Available values are: ${n.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return t}const a=n.find((e=>e.default))??n[0];if(!a)throw new Error("Unexpected error: 0 tabValues");return a.value}({defaultValue:t,tabValues:i}))),[s,u]=g({queryString:n,groupId:a}),[d,b]=function(e){let{groupId:t}=e;const n=function(e){return e?`docusaurus.tab.${e}`:null}(t),[a,i]=(0,c.Nk)(n);return[a,(0,r.useCallback)((e=>{n&&i.set(e)}),[n,i])]}({groupId:a}),f=(()=>{const e=s??d;return m({value:e,tabValues:i})?e:null})();(0,r.useLayoutEffect)((()=>{f&&o(f)}),[f]);return{selectedValue:l,selectValue:(0,r.useCallback)((e=>{if(!m({value:e,tabValues:i}))throw new Error(`Can't select invalid tab value=${e}`);o(e),u(e),b(e)}),[u,b,i]),tabValues:i}}var f=n(2389);const h="tabList__CuJ",v="tabItem_LNqP";function y(e){let{className:t,block:n,selectedValue:o,selectValue:s,tabValues:u}=e;const c=[],{blockElementScrollPositionUntilNextRender:d}=(0,l.o5)(),p=e=>{const t=e.currentTarget,n=c.indexOf(t),a=u[n].value;a!==o&&(d(t),s(a))},m=e=>{let t=null;switch(e.key){case"Enter":p(e);break;case"ArrowRight":{const n=c.indexOf(e.currentTarget)+1;t=c[n]??c[0];break}case"ArrowLeft":{const n=c.indexOf(e.currentTarget)-1;t=c[n]??c[c.length-1];break}}t?.focus()};return r.createElement("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,i.Z)("tabs",{"tabs--block":n},t)},u.map((e=>{let{value:t,label:n,attributes:l}=e;return r.createElement("li",(0,a.Z)({role:"tab",tabIndex:o===t?0:-1,"aria-selected":o===t,key:t,ref:e=>c.push(e),onKeyDown:m,onClick:p},l,{className:(0,i.Z)("tabs__item",v,l?.className,{"tabs__item--active":o===t})}),n??t)})))}function k(e){let{lazy:t,children:n,selectedValue:a}=e;const i=(Array.isArray(n)?n:[n]).filter(Boolean);if(t){const e=i.find((e=>e.props.value===a));return e?(0,r.cloneElement)(e,{className:"margin-top--md"}):null}return r.createElement("div",{className:"margin-top--md"},i.map(((e,t)=>(0,r.cloneElement)(e,{key:t,hidden:e.props.value!==a}))))}function N(e){const t=b(e);return r.createElement("div",{className:(0,i.Z)("tabs-container",h)},r.createElement(y,(0,a.Z)({},e,t)),r.createElement(k,(0,a.Z)({},e,t)))}function I(e){const t=(0,f.Z)();return r.createElement(N,(0,a.Z)({key:String(t)},e))}},8846:(e,t,n)=>{n.d(t,{Z:()=>o});var a=n(7294);const r="codeDescContainer_ie8f",i="desc_jyqI",l="example_eYlF";function o(e){let{children:t}=e,n=a.Children.toArray(t).filter((e=>e));return a.createElement("div",{className:r},a.createElement("div",{className:i},n[0]),a.createElement("div",{className:l},n[1]))}},9395:(e,t,n)=>{n.r(t),n.d(t,{assets:()=>d,contentTitle:()=>u,default:()=>g,frontMatter:()=>s,metadata:()=>c,toc:()=>p});var a=n(7462),r=(n(7294),n(3905)),i=n(8846),l=n(4866),o=n(5162);const s={},u="Utilities",c={unversionedId:"diagnostics/utilities",id:"diagnostics/utilities",title:"Utilities",description:"Is registered?",source:"@site/docs/diagnostics/utilities.md",sourceDirName:"diagnostics",slug:"/diagnostics/utilities",permalink:"/stashbox/docs/diagnostics/utilities",draft:!1,editUrl:"https://github.com/z4kn4fein/stashbox/edit/master/docs/docs/diagnostics/utilities.md",tags:[],version:"current",lastUpdatedBy:"dependabot[bot]",lastUpdatedAt:1699538162,formattedLastUpdatedAt:"Nov 9, 2023",frontMatter:{},sidebar:"docs",previous:{title:"Validation",permalink:"/stashbox/docs/diagnostics/validation"}},d={},p=[{value:"Is registered?",id:"is-registered",level:2},{value:"Named",id:"named",level:4},{value:"Can resolve?",id:"can-resolve",level:2},{value:"Get all mappings",id:"get-all-mappings",level:2},{value:"Registration diagnostics",id:"registration-diagnostics",level:2}],m={toc:p};function g(e){let{components:t,...n}=e;return(0,r.kt)("wrapper",(0,a.Z)({},m,n,{components:t,mdxType:"MDXLayout"}),(0,r.kt)("h1",{id:"utilities"},"Utilities"),(0,r.kt)("h2",{id:"is-registered"},"Is registered?"),(0,r.kt)(i.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("p",null,"With the ",(0,r.kt)("inlineCode",{parentName:"p"},"IsRegistered()")," function, you can find out whether a service is registered into the container or not."),(0,r.kt)("p",null,"It returns ",(0,r.kt)("inlineCode",{parentName:"p"},"true")," only when the container has a registration with the given type (and name). It only checks the actual container's registrations. For every cases, you should use the ",(0,r.kt)("inlineCode",{parentName:"p"},"CanResolve()")," method.")),(0,r.kt)("div",null,(0,r.kt)(l.Z,{groupId:"generic-runtime-apis",mdxType:"Tabs"},(0,r.kt)(o.Z,{value:"Generic API",label:"Generic API",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"bool isIJobRegistered = container.IsRegistered();\n"))),(0,r.kt)(o.Z,{value:"Runtime type API",label:"Runtime type API",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"bool isIJobRegistered = container.IsRegistered(typeof(IJob));\n"))),(0,r.kt)(o.Z,{value:"Named",label:"Named",mdxType:"TabItem"},(0,r.kt)("h4",{id:"named"},(0,r.kt)("strong",{parentName:"h4"},"Named")),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},'bool isIJobRegistered = container.IsRegistered("DbBackup");\n')))))),(0,r.kt)("h2",{id:"can-resolve"},"Can resolve?"),(0,r.kt)(i.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("p",null,"There might be cases when you are more interested in whether a service is resolvable from the container's actual state rather than finding out whether it's registered."),(0,r.kt)("p",null,(0,r.kt)("inlineCode",{parentName:"p"},"CanResolve()")," returns ",(0,r.kt)("inlineCode",{parentName:"p"},"true")," only when at least one of the following is true:"),(0,r.kt)("ul",null,(0,r.kt)("li",{parentName:"ul"},"The requested type is registered in the current or one of the parent containers."),(0,r.kt)("li",{parentName:"ul"},"The requested type is a closed generic type, and its open generic definition is registered."),(0,r.kt)("li",{parentName:"ul"},"The requested type is a wrapper (",(0,r.kt)("inlineCode",{parentName:"li"},"IEnumerable<>"),", ",(0,r.kt)("inlineCode",{parentName:"li"},"Lazy<>"),", ",(0,r.kt)("inlineCode",{parentName:"li"},"Func<>"),", ",(0,r.kt)("inlineCode",{parentName:"li"},"KeyValuePair<,>"),", ",(0,r.kt)("inlineCode",{parentName:"li"},"ReadOnlyKeyValue<,>"),", ",(0,r.kt)("inlineCode",{parentName:"li"},"Metadata<,>"),", ",(0,r.kt)("inlineCode",{parentName:"li"},"ValueTuple<,>"),", or ",(0,r.kt)("inlineCode",{parentName:"li"},"Tuple<,>"),"), and the underlying type is registered."),(0,r.kt)("li",{parentName:"ul"},"The requested type is not registered, but it's resolvable, and ",(0,r.kt)("a",{parentName:"li",href:"/docs/configuration/container-configuration#unknown-type-resolution"},"unknown type resolution")," is enabled."))),(0,r.kt)("div",null,(0,r.kt)(l.Z,{groupId:"generic-runtime-apis",mdxType:"Tabs"},(0,r.kt)(o.Z,{value:"Generic API",label:"Generic API",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"bool isIJobResolvable = container.CanResolve();\n"))),(0,r.kt)(o.Z,{value:"Runtime type API",label:"Runtime type API",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"bool isIJobResolvable = container.CanResolve(typeof(IJob));\n"))),(0,r.kt)(o.Z,{value:"Named",label:"Named",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},'bool isIJobResolvable = container.CanResolve("DbBackup");\n')))))),(0,r.kt)("h2",{id:"get-all-mappings"},"Get all mappings"),(0,r.kt)(i.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("p",null,"You can get all registrations in a key-value pair collection (where the key is the ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type")," and the value is the actual registration) by calling the ",(0,r.kt)("inlineCode",{parentName:"p"},".GetRegistrationMappings()")," method.")),(0,r.kt)("div",null,(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"IEnumerable> mappings = \n container.GetRegistrationMappings();\n")))),(0,r.kt)("h2",{id:"registration-diagnostics"},"Registration diagnostics"),(0,r.kt)(i.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("p",null,"You can get a much more readable version of the registration mappings by calling the ",(0,r.kt)("inlineCode",{parentName:"p"},".GetRegistrationDiagnostics()")," method."),(0,r.kt)("p",null,(0,r.kt)("inlineCode",{parentName:"p"},"RegistrationDiagnosticsInfo")," has an overridden ",(0,r.kt)("inlineCode",{parentName:"p"},".ToString()")," method that returns the mapping details formatted in a human-readable form.")),(0,r.kt)("div",null,(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register("DbBackupJob");\ncontainer.Register(typeof(IEventHandler<>), typeof(EventHandler<>));\n\nIEnumerable diagnostics = \n container.GetRegistrationDiagnostics();\n\ndiagnostics.ForEach(Console.WriteLine);\n// output:\n// IJob => DbBackup, name: DbBackupJob\n// IEventHandler<> => EventHandler<>, name: null\n')))))}g.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/b43f639d.7859f4e2.js b/assets/js/b43f639d.7859f4e2.js deleted file mode 100644 index 46bc6a72..00000000 --- a/assets/js/b43f639d.7859f4e2.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkwebsite=self.webpackChunkwebsite||[]).push([[965],{3905:(e,t,n)=>{n.d(t,{Zo:()=>c,kt:()=>h});var i=n(7294);function a(e,t,n){return t in e?Object.defineProperty(e,t,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[t]=n,e}function o(e,t){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(e);t&&(i=i.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),n.push.apply(n,i)}return n}function r(e){for(var t=1;t=0||(a[n]=e[n]);return a}(e,t);if(Object.getOwnPropertySymbols){var o=Object.getOwnPropertySymbols(e);for(i=0;i=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(a[n]=e[n])}return a}var s=i.createContext({}),u=function(e){var t=i.useContext(s),n=t;return e&&(n="function"==typeof e?e(t):r(r({},t),e)),n},c=function(e){var t=u(e.components);return i.createElement(s.Provider,{value:t},e.children)},p="mdxType",d={inlineCode:"code",wrapper:function(e){var t=e.children;return i.createElement(i.Fragment,{},t)}},m=i.forwardRef((function(e,t){var n=e.components,a=e.mdxType,o=e.originalType,s=e.parentName,c=l(e,["components","mdxType","originalType","parentName"]),p=u(n),m=a,h=p["".concat(s,".").concat(m)]||p[m]||d[m]||o;return n?i.createElement(h,r(r({ref:t},c),{},{components:n})):i.createElement(h,r({ref:t},c))}));function h(e,t){var n=arguments,a=t&&t.mdxType;if("string"==typeof e||a){var o=n.length,r=new Array(o);r[0]=m;var l={};for(var s in t)hasOwnProperty.call(t,s)&&(l[s]=t[s]);l.originalType=e,l[p]="string"==typeof e?e:a,r[1]=l;for(var u=2;u{n.d(t,{Z:()=>r});var i=n(7294),a=n(6010);const o="tabItem_Ymn6";function r(e){let{children:t,hidden:n,className:r}=e;return i.createElement("div",{role:"tabpanel",className:(0,a.Z)(o,r),hidden:n},t)}},4866:(e,t,n)=>{n.d(t,{Z:()=>C});var i=n(7462),a=n(7294),o=n(6010),r=n(2466),l=n(6550),s=n(1980),u=n(7392),c=n(12);function p(e){return function(e){return a.Children.map(e,(e=>{if(!e||(0,a.isValidElement)(e)&&function(e){const{props:t}=e;return!!t&&"object"==typeof t&&"value"in t}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}(e).map((e=>{let{props:{value:t,label:n,attributes:i,default:a}}=e;return{value:t,label:n,attributes:i,default:a}}))}function d(e){const{values:t,children:n}=e;return(0,a.useMemo)((()=>{const e=t??p(n);return function(e){const t=(0,u.l)(e,((e,t)=>e.value===t.value));if(t.length>0)throw new Error(`Docusaurus error: Duplicate values "${t.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[t,n])}function m(e){let{value:t,tabValues:n}=e;return n.some((e=>e.value===t))}function h(e){let{queryString:t=!1,groupId:n}=e;const i=(0,l.k6)(),o=function(e){let{queryString:t=!1,groupId:n}=e;if("string"==typeof t)return t;if(!1===t)return null;if(!0===t&&!n)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return n??null}({queryString:t,groupId:n});return[(0,s._X)(o),(0,a.useCallback)((e=>{if(!o)return;const t=new URLSearchParams(i.location.search);t.set(o,e),i.replace({...i.location,search:t.toString()})}),[o,i])]}function f(e){const{defaultValue:t,queryString:n=!1,groupId:i}=e,o=d(e),[r,l]=(0,a.useState)((()=>function(e){let{defaultValue:t,tabValues:n}=e;if(0===n.length)throw new Error("Docusaurus error: the component requires at least one children component");if(t){if(!m({value:t,tabValues:n}))throw new Error(`Docusaurus error: The has a defaultValue "${t}" but none of its children has the corresponding value. Available values are: ${n.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return t}const i=n.find((e=>e.default))??n[0];if(!i)throw new Error("Unexpected error: 0 tabValues");return i.value}({defaultValue:t,tabValues:o}))),[s,u]=h({queryString:n,groupId:i}),[p,f]=function(e){let{groupId:t}=e;const n=function(e){return e?`docusaurus.tab.${e}`:null}(t),[i,o]=(0,c.Nk)(n);return[i,(0,a.useCallback)((e=>{n&&o.set(e)}),[n,o])]}({groupId:i}),k=(()=>{const e=s??p;return m({value:e,tabValues:o})?e:null})();(0,a.useLayoutEffect)((()=>{k&&l(k)}),[k]);return{selectedValue:r,selectValue:(0,a.useCallback)((e=>{if(!m({value:e,tabValues:o}))throw new Error(`Can't select invalid tab value=${e}`);l(e),u(e),f(e)}),[u,f,o]),tabValues:o}}var k=n(2389);const g="tabList__CuJ",v="tabItem_LNqP";function b(e){let{className:t,block:n,selectedValue:l,selectValue:s,tabValues:u}=e;const c=[],{blockElementScrollPositionUntilNextRender:p}=(0,r.o5)(),d=e=>{const t=e.currentTarget,n=c.indexOf(t),i=u[n].value;i!==l&&(p(t),s(i))},m=e=>{let t=null;switch(e.key){case"Enter":d(e);break;case"ArrowRight":{const n=c.indexOf(e.currentTarget)+1;t=c[n]??c[0];break}case"ArrowLeft":{const n=c.indexOf(e.currentTarget)-1;t=c[n]??c[c.length-1];break}}t?.focus()};return a.createElement("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,o.Z)("tabs",{"tabs--block":n},t)},u.map((e=>{let{value:t,label:n,attributes:r}=e;return a.createElement("li",(0,i.Z)({role:"tab",tabIndex:l===t?0:-1,"aria-selected":l===t,key:t,ref:e=>c.push(e),onKeyDown:m,onClick:d},r,{className:(0,o.Z)("tabs__item",v,r?.className,{"tabs__item--active":l===t})}),n??t)})))}function y(e){let{lazy:t,children:n,selectedValue:i}=e;const o=(Array.isArray(n)?n:[n]).filter(Boolean);if(t){const e=o.find((e=>e.props.value===i));return e?(0,a.cloneElement)(e,{className:"margin-top--md"}):null}return a.createElement("div",{className:"margin-top--md"},o.map(((e,t)=>(0,a.cloneElement)(e,{key:t,hidden:e.props.value!==i}))))}function w(e){const t=f(e);return a.createElement("div",{className:(0,o.Z)("tabs-container",g)},a.createElement(b,(0,i.Z)({},e,t)),a.createElement(y,(0,i.Z)({},e,t)))}function C(e){const t=(0,k.Z)();return a.createElement(w,(0,i.Z)({key:String(t)},e))}},8846:(e,t,n)=>{n.d(t,{Z:()=>l});var i=n(7294);const a="codeDescContainer_ie8f",o="desc_jyqI",r="example_eYlF";function l(e){let{children:t}=e,n=i.Children.toArray(t).filter((e=>e));return i.createElement("div",{className:a},i.createElement("div",{className:o},n[0]),i.createElement("div",{className:r},n[1]))}},1336:(e,t,n)=>{n.r(t),n.d(t,{assets:()=>u,contentTitle:()=>l,default:()=>d,frontMatter:()=>r,metadata:()=>s,toc:()=>c});var i=n(7462),a=(n(7294),n(3905)),o=n(8846);n(4866),n(5162);const r={},l="Container configuration",s={unversionedId:"configuration/container-configuration",id:"configuration/container-configuration",title:"Container configuration",description:"The container's constructor has an Action parameter used to configure its behavior.",source:"@site/docs/configuration/container-configuration.md",sourceDirName:"configuration",slug:"/configuration/container-configuration",permalink:"/stashbox/docs/configuration/container-configuration",draft:!1,editUrl:"https://github.com/z4kn4fein/stashbox/edit/master/docs/docs/configuration/container-configuration.md",tags:[],version:"current",lastUpdatedBy:"dependabot[bot]",lastUpdatedAt:1697644420,formattedLastUpdatedAt:"Oct 18, 2023",frontMatter:{},sidebar:"docs",previous:{title:"Registration configuration",permalink:"/stashbox/docs/configuration/registration-configuration"},next:{title:"Generics",permalink:"/stashbox/docs/advanced/generics"}},u={},c=[{value:"Default configuration",id:"default-configuration",level:2},{value:"Tracking disposable transients",id:"tracking-disposable-transients",level:2},{value:"Auto member-injection",id:"auto-member-injection",level:2},{value:"PropertiesWithPublicSetter",id:"propertieswithpublicsetter",level:3},{value:"PropertiesWithLimitedAccess",id:"propertieswithlimitedaccess",level:3},{value:"PrivateFields",id:"privatefields",level:3},{value:"Combined rules",id:"combined-rules",level:3},{value:"Constructor selection",id:"constructor-selection",level:2},{value:"PreferMostParameters",id:"prefermostparameters",level:3},{value:"PreferLeastParameters",id:"preferleastparameters",level:3},{value:"Registration behavior",id:"registration-behavior",level:2},{value:"SkipDuplications",id:"skipduplications",level:3},{value:"ThrowException",id:"throwexception",level:3},{value:"ReplaceExisting",id:"replaceexisting",level:3},{value:"PreserveDuplications",id:"preserveduplications",level:3},{value:"Default lifetime",id:"default-lifetime",level:2},{value:"Lifetime validation",id:"lifetime-validation",level:2},{value:"Conventional resolution",id:"conventional-resolution",level:2},{value:"Using named service for un-named requests",id:"using-named-service-for-un-named-requests",level:2},{value:"Default value injection",id:"default-value-injection",level:2},{value:"Unknown type resolution",id:"unknown-type-resolution",level:2},{value:"Custom compiler",id:"custom-compiler",level:2},{value:"Re-build singletons in child containers",id:"re-build-singletons-in-child-containers",level:2}],p={toc:c};function d(e){let{components:t,...n}=e;return(0,a.kt)("wrapper",(0,i.Z)({},p,n,{components:t,mdxType:"MDXLayout"}),(0,a.kt)("h1",{id:"container-configuration"},"Container configuration"),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"The container's constructor has an ",(0,a.kt)("inlineCode",{parentName:"p"},"Action")," parameter used to configure its behavior."),(0,a.kt)("p",null,"The configuration API is fluent, which means you can chain the configuration methods after each other.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"var container = new StashboxContainer(options => options\n .WithDisposableTransientTracking()\n .WithConstructorSelectionRule(Rules.ConstructorSelection.PreferLeastParameters)\n .WithRegistrationBehavior(Rules.RegistrationBehavior.ThrowException));\n")))),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,(0,a.kt)("strong",{parentName:"p"},"Re-configuration")," of the container is also supported by calling its ",(0,a.kt)("inlineCode",{parentName:"p"},".Configure()")," method.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"var container = new StashboxContainer();\ncontainer.Configure(options => options.WithDisposableTransientTracking());\n")))),(0,a.kt)("h2",{id:"default-configuration"},"Default configuration"),(0,a.kt)("p",null,"These features are set by default:"),(0,a.kt)("ul",null,(0,a.kt)("li",{parentName:"ul"},(0,a.kt)("a",{parentName:"li",href:"/docs/configuration/container-configuration#constructor-selection"},"Constructor selection"),": ",(0,a.kt)("inlineCode",{parentName:"li"},"Rules.ConstructorSelection.PreferMostParameters")),(0,a.kt)("li",{parentName:"ul"},(0,a.kt)("a",{parentName:"li",href:"/docs/configuration/container-configuration#registration-behavior"},"Registration behavior"),": ",(0,a.kt)("inlineCode",{parentName:"li"},"Rules.RegistrationBehavior.SkipDuplications")),(0,a.kt)("li",{parentName:"ul"},(0,a.kt)("a",{parentName:"li",href:"/docs/configuration/container-configuration#default-lifetime"},"Default lifetime"),": ",(0,a.kt)("inlineCode",{parentName:"li"},"Lifetimes.Transient"))),(0,a.kt)("h2",{id:"tracking-disposable-transients"},"Tracking disposable transients"),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"With this option, you can enable or disable the tracking of disposable transient objects.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .WithDisposableTransientTracking());\n")))),(0,a.kt)("h2",{id:"auto-member-injection"},"Auto member-injection"),(0,a.kt)("p",null,"With this option, you can enable or disable the auto member-injection without ",(0,a.kt)("a",{parentName:"p",href:"/docs/guides/service-resolution#attributes"},"attributes"),"."),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("h3",{id:"propertieswithpublicsetter"},(0,a.kt)("inlineCode",{parentName:"h3"},"PropertiesWithPublicSetter")),(0,a.kt)("p",null,"With this flag, the container will perform auto-injection on properties with public setters.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .WithAutoMemberInjection(\n Rules.AutoMemberInjectionRules.PropertiesWithPublicSetter));\n")))),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("h3",{id:"propertieswithlimitedaccess"},(0,a.kt)("inlineCode",{parentName:"h3"},"PropertiesWithLimitedAccess")),(0,a.kt)("p",null,"With this flag, the container will perform auto-injection on properties even when they don't have a public setter.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .WithAutoMemberInjection(\n Rules.AutoMemberInjectionRules.PropertiesWithLimitedAccess));\n")))),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("h3",{id:"privatefields"},(0,a.kt)("inlineCode",{parentName:"h3"},"PrivateFields")),(0,a.kt)("p",null,"With this flag, the container will perform auto-injection on private fields too.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .WithAutoMemberInjection(\n Rules.AutoMemberInjectionRules.PrivateFields));\n")))),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("h3",{id:"combined-rules"},"Combined rules"),(0,a.kt)("p",null,"You can also combine these flags with bitwise logical operators to get a merged ruleset.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .WithAutoMemberInjection(Rules.AutoMemberInjectionRules.PrivateFields | \n Rules.AutoMemberInjectionRules.PropertiesWithPublicSetter));\n")))),(0,a.kt)("admonition",{type:"note"},(0,a.kt)("p",{parentName:"admonition"},"Member selection filter: ",(0,a.kt)("inlineCode",{parentName:"p"},"config.WithAutoMemberInjection(filter: member => member.Type != typeof(IJob))"))),(0,a.kt)("h2",{id:"constructor-selection"},"Constructor selection"),(0,a.kt)("p",null,"With this option, you can set the constructor selection rule used to determine which constructor the container should use for instantiation."),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("h3",{id:"prefermostparameters"},(0,a.kt)("inlineCode",{parentName:"h3"},"PreferMostParameters")),(0,a.kt)("p",null,"It prefers the constructor which has the most extended parameter list.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .WithConstructorSelectionRule(\n Rules.ConstructorSelection.PreferMostParameters));\n")))),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("h3",{id:"preferleastparameters"},(0,a.kt)("inlineCode",{parentName:"h3"},"PreferLeastParameters")),(0,a.kt)("p",null,"It prefers the constructor which has the shortest parameter list.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .WithConstructorSelectionRule(\n Rules.ConstructorSelection.PreferLeastParameters));\n")))),(0,a.kt)("h2",{id:"registration-behavior"},"Registration behavior"),(0,a.kt)("p",null,"With this option, you can set the actual behavior used when a new service is registered into the container. These options do not affect named registrations."),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("h3",{id:"skipduplications"},(0,a.kt)("inlineCode",{parentName:"h3"},"SkipDuplications")),(0,a.kt)("p",null,"The container will skip new registrations when the given ",(0,a.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"implementation type")," is already registered.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .WithRegistrationBehavior(\n Rules.RegistrationBehavior.SkipDuplications));\n")))),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("h3",{id:"throwexception"},(0,a.kt)("inlineCode",{parentName:"h3"},"ThrowException")),(0,a.kt)("p",null,"The container throws an ",(0,a.kt)("a",{parentName:"p",href:"/docs/diagnostics/validation#servicealreadyregisteredexception"},"exception")," when the given ",(0,a.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"implementation type")," is already registered.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .WithRegistrationBehavior(\n Rules.RegistrationBehavior.ThrowException));\n")))),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("h3",{id:"replaceexisting"},(0,a.kt)("inlineCode",{parentName:"h3"},"ReplaceExisting")),(0,a.kt)("p",null,"The container will replace the already ",(0,a.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-registration--registered-service"},"registered service")," with the given one when they have the same ",(0,a.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"implementation type"),".")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .WithRegistrationBehavior(\n Rules.RegistrationBehavior.ReplaceExisting));\n")))),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("h3",{id:"preserveduplications"},(0,a.kt)("inlineCode",{parentName:"h3"},"PreserveDuplications")),(0,a.kt)("p",null,"The container will keep registering the new services with the same ",(0,a.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"implementation type"),".")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .WithRegistrationBehavior(\n Rules.RegistrationBehavior.PreserveDuplications));\n")))),(0,a.kt)("h2",{id:"default-lifetime"},"Default lifetime"),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"With this option, you can set the default lifetime used when a service doesn't have a configured one.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .WithDefaultLifetime(Lifetimes.Scoped));\n")))),(0,a.kt)("h2",{id:"lifetime-validation"},"Lifetime validation"),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"With this option, you can enable or disable the life-span and ",(0,a.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#root-scope"},"root scope")," resolution ",(0,a.kt)("a",{parentName:"p",href:"/docs/diagnostics/validation#lifetime-validation"},"validation")," on the dependency tree.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .WithLifetimeValidation());\n")))),(0,a.kt)("h2",{id:"conventional-resolution"},"Conventional resolution"),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"With this option, you can enable or disable conventional resolution, which means the container treats the constructor/method parameter or member names as dependency names used by ",(0,a.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#named-resolution"},"named resolution"),".")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .TreatParameterAndMemberNameAsDependencyName());\n")))),(0,a.kt)("h2",{id:"using-named-service-for-un-named-requests"},"Using named service for un-named requests"),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"With this option, you can enable or disable the selection of named registrations when the resolution request is un-named but with the same type.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .WithNamedDependencyResolutionForUnNamedRequests());\n")))),(0,a.kt)("h2",{id:"default-value-injection"},"Default value injection"),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"With this option, you can enable or disable the default value injection.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .WithDefaultValueInjection());\n")))),(0,a.kt)("h2",{id:"unknown-type-resolution"},"Unknown type resolution"),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"With this option, you can enable or disable the resolution of unregistered types. You can also use a configurator delegate to configure the registrations the container will create from the unknown types.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .WithUnknownTypeResolution(config => config.AsImplementedTypes()));\n")))),(0,a.kt)("h2",{id:"custom-compiler"},"Custom compiler"),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"With this option, you can set an external expression tree compiler. It can be useful on platforms where the IL generator modules are not available; therefore, the expression compiler in Stashbox couldn't work.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .WithExpressionCompiler(\n Rules.ExpressionCompilers.MicrosoftExpressionCompiler));\n")))),(0,a.kt)("h2",{id:"re-build-singletons-in-child-containers"},"Re-build singletons in child containers"),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"With this option, you can enable or disable the re-building of singletons in child containers. It allows the child containers to override singleton dependencies in the parent.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .WithReBuildSingletonsInChildContainer());\n")))),(0,a.kt)("admonition",{type:"note"},(0,a.kt)("p",{parentName:"admonition"},"This feature is not affecting the already built singleton instances in the parent.")))}d.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/b43f639d.9e4b25d8.js b/assets/js/b43f639d.9e4b25d8.js new file mode 100644 index 00000000..2749eebc --- /dev/null +++ b/assets/js/b43f639d.9e4b25d8.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkwebsite=self.webpackChunkwebsite||[]).push([[965],{3905:(e,t,n)=>{n.d(t,{Zo:()=>c,kt:()=>h});var i=n(7294);function a(e,t,n){return t in e?Object.defineProperty(e,t,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[t]=n,e}function o(e,t){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(e);t&&(i=i.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),n.push.apply(n,i)}return n}function r(e){for(var t=1;t=0||(a[n]=e[n]);return a}(e,t);if(Object.getOwnPropertySymbols){var o=Object.getOwnPropertySymbols(e);for(i=0;i=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(a[n]=e[n])}return a}var s=i.createContext({}),u=function(e){var t=i.useContext(s),n=t;return e&&(n="function"==typeof e?e(t):r(r({},t),e)),n},c=function(e){var t=u(e.components);return i.createElement(s.Provider,{value:t},e.children)},p="mdxType",d={inlineCode:"code",wrapper:function(e){var t=e.children;return i.createElement(i.Fragment,{},t)}},m=i.forwardRef((function(e,t){var n=e.components,a=e.mdxType,o=e.originalType,s=e.parentName,c=l(e,["components","mdxType","originalType","parentName"]),p=u(n),m=a,h=p["".concat(s,".").concat(m)]||p[m]||d[m]||o;return n?i.createElement(h,r(r({ref:t},c),{},{components:n})):i.createElement(h,r({ref:t},c))}));function h(e,t){var n=arguments,a=t&&t.mdxType;if("string"==typeof e||a){var o=n.length,r=new Array(o);r[0]=m;var l={};for(var s in t)hasOwnProperty.call(t,s)&&(l[s]=t[s]);l.originalType=e,l[p]="string"==typeof e?e:a,r[1]=l;for(var u=2;u{n.d(t,{Z:()=>r});var i=n(7294),a=n(6010);const o="tabItem_Ymn6";function r(e){let{children:t,hidden:n,className:r}=e;return i.createElement("div",{role:"tabpanel",className:(0,a.Z)(o,r),hidden:n},t)}},4866:(e,t,n)=>{n.d(t,{Z:()=>N});var i=n(7462),a=n(7294),o=n(6010),r=n(2466),l=n(6550),s=n(1980),u=n(7392),c=n(12);function p(e){return function(e){return a.Children.map(e,(e=>{if(!e||(0,a.isValidElement)(e)&&function(e){const{props:t}=e;return!!t&&"object"==typeof t&&"value"in t}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}(e).map((e=>{let{props:{value:t,label:n,attributes:i,default:a}}=e;return{value:t,label:n,attributes:i,default:a}}))}function d(e){const{values:t,children:n}=e;return(0,a.useMemo)((()=>{const e=t??p(n);return function(e){const t=(0,u.l)(e,((e,t)=>e.value===t.value));if(t.length>0)throw new Error(`Docusaurus error: Duplicate values "${t.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[t,n])}function m(e){let{value:t,tabValues:n}=e;return n.some((e=>e.value===t))}function h(e){let{queryString:t=!1,groupId:n}=e;const i=(0,l.k6)(),o=function(e){let{queryString:t=!1,groupId:n}=e;if("string"==typeof t)return t;if(!1===t)return null;if(!0===t&&!n)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return n??null}({queryString:t,groupId:n});return[(0,s._X)(o),(0,a.useCallback)((e=>{if(!o)return;const t=new URLSearchParams(i.location.search);t.set(o,e),i.replace({...i.location,search:t.toString()})}),[o,i])]}function f(e){const{defaultValue:t,queryString:n=!1,groupId:i}=e,o=d(e),[r,l]=(0,a.useState)((()=>function(e){let{defaultValue:t,tabValues:n}=e;if(0===n.length)throw new Error("Docusaurus error: the component requires at least one children component");if(t){if(!m({value:t,tabValues:n}))throw new Error(`Docusaurus error: The has a defaultValue "${t}" but none of its children has the corresponding value. Available values are: ${n.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return t}const i=n.find((e=>e.default))??n[0];if(!i)throw new Error("Unexpected error: 0 tabValues");return i.value}({defaultValue:t,tabValues:o}))),[s,u]=h({queryString:n,groupId:i}),[p,f]=function(e){let{groupId:t}=e;const n=function(e){return e?`docusaurus.tab.${e}`:null}(t),[i,o]=(0,c.Nk)(n);return[i,(0,a.useCallback)((e=>{n&&o.set(e)}),[n,o])]}({groupId:i}),k=(()=>{const e=s??p;return m({value:e,tabValues:o})?e:null})();(0,a.useLayoutEffect)((()=>{k&&l(k)}),[k]);return{selectedValue:r,selectValue:(0,a.useCallback)((e=>{if(!m({value:e,tabValues:o}))throw new Error(`Can't select invalid tab value=${e}`);l(e),u(e),f(e)}),[u,f,o]),tabValues:o}}var k=n(2389);const g="tabList__CuJ",v="tabItem_LNqP";function b(e){let{className:t,block:n,selectedValue:l,selectValue:s,tabValues:u}=e;const c=[],{blockElementScrollPositionUntilNextRender:p}=(0,r.o5)(),d=e=>{const t=e.currentTarget,n=c.indexOf(t),i=u[n].value;i!==l&&(p(t),s(i))},m=e=>{let t=null;switch(e.key){case"Enter":d(e);break;case"ArrowRight":{const n=c.indexOf(e.currentTarget)+1;t=c[n]??c[0];break}case"ArrowLeft":{const n=c.indexOf(e.currentTarget)-1;t=c[n]??c[c.length-1];break}}t?.focus()};return a.createElement("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,o.Z)("tabs",{"tabs--block":n},t)},u.map((e=>{let{value:t,label:n,attributes:r}=e;return a.createElement("li",(0,i.Z)({role:"tab",tabIndex:l===t?0:-1,"aria-selected":l===t,key:t,ref:e=>c.push(e),onKeyDown:m,onClick:d},r,{className:(0,o.Z)("tabs__item",v,r?.className,{"tabs__item--active":l===t})}),n??t)})))}function y(e){let{lazy:t,children:n,selectedValue:i}=e;const o=(Array.isArray(n)?n:[n]).filter(Boolean);if(t){const e=o.find((e=>e.props.value===i));return e?(0,a.cloneElement)(e,{className:"margin-top--md"}):null}return a.createElement("div",{className:"margin-top--md"},o.map(((e,t)=>(0,a.cloneElement)(e,{key:t,hidden:e.props.value!==i}))))}function w(e){const t=f(e);return a.createElement("div",{className:(0,o.Z)("tabs-container",g)},a.createElement(b,(0,i.Z)({},e,t)),a.createElement(y,(0,i.Z)({},e,t)))}function N(e){const t=(0,k.Z)();return a.createElement(w,(0,i.Z)({key:String(t)},e))}},8846:(e,t,n)=>{n.d(t,{Z:()=>l});var i=n(7294);const a="codeDescContainer_ie8f",o="desc_jyqI",r="example_eYlF";function l(e){let{children:t}=e,n=i.Children.toArray(t).filter((e=>e));return i.createElement("div",{className:a},i.createElement("div",{className:o},n[0]),i.createElement("div",{className:r},n[1]))}},1336:(e,t,n)=>{n.r(t),n.d(t,{assets:()=>u,contentTitle:()=>l,default:()=>d,frontMatter:()=>r,metadata:()=>s,toc:()=>c});var i=n(7462),a=(n(7294),n(3905)),o=n(8846);n(4866),n(5162);const r={},l="Container configuration",s={unversionedId:"configuration/container-configuration",id:"configuration/container-configuration",title:"Container configuration",description:"The container's constructor has an Action parameter used to configure its behavior.",source:"@site/docs/configuration/container-configuration.md",sourceDirName:"configuration",slug:"/configuration/container-configuration",permalink:"/stashbox/docs/configuration/container-configuration",draft:!1,editUrl:"https://github.com/z4kn4fein/stashbox/edit/master/docs/docs/configuration/container-configuration.md",tags:[],version:"current",lastUpdatedBy:"dependabot[bot]",lastUpdatedAt:1699538162,formattedLastUpdatedAt:"Nov 9, 2023",frontMatter:{},sidebar:"docs",previous:{title:"Registration configuration",permalink:"/stashbox/docs/configuration/registration-configuration"},next:{title:"Generics",permalink:"/stashbox/docs/advanced/generics"}},u={},c=[{value:"Default configuration",id:"default-configuration",level:2},{value:"Tracking disposable transients",id:"tracking-disposable-transients",level:2},{value:"Auto member-injection",id:"auto-member-injection",level:2},{value:"PropertiesWithPublicSetter",id:"propertieswithpublicsetter",level:3},{value:"PropertiesWithLimitedAccess",id:"propertieswithlimitedaccess",level:3},{value:"PrivateFields",id:"privatefields",level:3},{value:"Combined rules",id:"combined-rules",level:3},{value:"Constructor selection",id:"constructor-selection",level:2},{value:"PreferMostParameters",id:"prefermostparameters",level:3},{value:"PreferLeastParameters",id:"preferleastparameters",level:3},{value:"Registration behavior",id:"registration-behavior",level:2},{value:"SkipDuplications",id:"skipduplications",level:3},{value:"ThrowException",id:"throwexception",level:3},{value:"ReplaceExisting",id:"replaceexisting",level:3},{value:"PreserveDuplications",id:"preserveduplications",level:3},{value:"Default lifetime",id:"default-lifetime",level:2},{value:"Lifetime validation",id:"lifetime-validation",level:2},{value:"Conventional resolution",id:"conventional-resolution",level:2},{value:"Using named service for un-named requests",id:"using-named-service-for-un-named-requests",level:2},{value:"Default value injection",id:"default-value-injection",level:2},{value:"Unknown type resolution",id:"unknown-type-resolution",level:2},{value:"Custom compiler",id:"custom-compiler",level:2},{value:"Re-build singletons in child containers",id:"re-build-singletons-in-child-containers",level:2}],p={toc:c};function d(e){let{components:t,...n}=e;return(0,a.kt)("wrapper",(0,i.Z)({},p,n,{components:t,mdxType:"MDXLayout"}),(0,a.kt)("h1",{id:"container-configuration"},"Container configuration"),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"The container's constructor has an ",(0,a.kt)("inlineCode",{parentName:"p"},"Action")," parameter used to configure its behavior."),(0,a.kt)("p",null,"The configuration API is fluent, which means you can chain the configuration methods after each other.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"var container = new StashboxContainer(options => options\n .WithDisposableTransientTracking()\n .WithConstructorSelectionRule(Rules.ConstructorSelection.PreferLeastParameters)\n .WithRegistrationBehavior(Rules.RegistrationBehavior.ThrowException));\n")))),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,(0,a.kt)("strong",{parentName:"p"},"Re-configuration")," of the container is also supported by calling its ",(0,a.kt)("inlineCode",{parentName:"p"},".Configure()")," method.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"var container = new StashboxContainer();\ncontainer.Configure(options => options.WithDisposableTransientTracking());\n")))),(0,a.kt)("h2",{id:"default-configuration"},"Default configuration"),(0,a.kt)("p",null,"These features are set by default:"),(0,a.kt)("ul",null,(0,a.kt)("li",{parentName:"ul"},(0,a.kt)("a",{parentName:"li",href:"/docs/configuration/container-configuration#constructor-selection"},"Constructor selection"),": ",(0,a.kt)("inlineCode",{parentName:"li"},"Rules.ConstructorSelection.PreferMostParameters")),(0,a.kt)("li",{parentName:"ul"},(0,a.kt)("a",{parentName:"li",href:"/docs/configuration/container-configuration#registration-behavior"},"Registration behavior"),": ",(0,a.kt)("inlineCode",{parentName:"li"},"Rules.RegistrationBehavior.SkipDuplications")),(0,a.kt)("li",{parentName:"ul"},(0,a.kt)("a",{parentName:"li",href:"/docs/configuration/container-configuration#default-lifetime"},"Default lifetime"),": ",(0,a.kt)("inlineCode",{parentName:"li"},"Lifetimes.Transient"))),(0,a.kt)("h2",{id:"tracking-disposable-transients"},"Tracking disposable transients"),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"With this option, you can enable or disable the tracking of disposable transient objects.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .WithDisposableTransientTracking());\n")))),(0,a.kt)("h2",{id:"auto-member-injection"},"Auto member-injection"),(0,a.kt)("p",null,"With this option, you can enable or disable the auto member-injection without ",(0,a.kt)("a",{parentName:"p",href:"/docs/guides/service-resolution#attributes"},"attributes"),"."),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("h3",{id:"propertieswithpublicsetter"},(0,a.kt)("inlineCode",{parentName:"h3"},"PropertiesWithPublicSetter")),(0,a.kt)("p",null,"With this flag, the container will perform auto-injection on properties with public setters.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .WithAutoMemberInjection(\n Rules.AutoMemberInjectionRules.PropertiesWithPublicSetter));\n")))),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("h3",{id:"propertieswithlimitedaccess"},(0,a.kt)("inlineCode",{parentName:"h3"},"PropertiesWithLimitedAccess")),(0,a.kt)("p",null,"With this flag, the container will perform auto-injection on properties even when they don't have a public setter.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .WithAutoMemberInjection(\n Rules.AutoMemberInjectionRules.PropertiesWithLimitedAccess));\n")))),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("h3",{id:"privatefields"},(0,a.kt)("inlineCode",{parentName:"h3"},"PrivateFields")),(0,a.kt)("p",null,"With this flag, the container will perform auto-injection on private fields too.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .WithAutoMemberInjection(\n Rules.AutoMemberInjectionRules.PrivateFields));\n")))),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("h3",{id:"combined-rules"},"Combined rules"),(0,a.kt)("p",null,"You can also combine these flags with bitwise logical operators to get a merged ruleset.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .WithAutoMemberInjection(Rules.AutoMemberInjectionRules.PrivateFields | \n Rules.AutoMemberInjectionRules.PropertiesWithPublicSetter));\n")))),(0,a.kt)("admonition",{type:"note"},(0,a.kt)("p",{parentName:"admonition"},"Member selection filter: ",(0,a.kt)("inlineCode",{parentName:"p"},"config.WithAutoMemberInjection(filter: member => member.Type != typeof(IJob))"))),(0,a.kt)("h2",{id:"constructor-selection"},"Constructor selection"),(0,a.kt)("p",null,"With this option, you can set the constructor selection rule used to determine which constructor the container should use for instantiation."),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("h3",{id:"prefermostparameters"},(0,a.kt)("inlineCode",{parentName:"h3"},"PreferMostParameters")),(0,a.kt)("p",null,"It prefers the constructor which has the most extended parameter list.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .WithConstructorSelectionRule(\n Rules.ConstructorSelection.PreferMostParameters));\n")))),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("h3",{id:"preferleastparameters"},(0,a.kt)("inlineCode",{parentName:"h3"},"PreferLeastParameters")),(0,a.kt)("p",null,"It prefers the constructor which has the shortest parameter list.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .WithConstructorSelectionRule(\n Rules.ConstructorSelection.PreferLeastParameters));\n")))),(0,a.kt)("h2",{id:"registration-behavior"},"Registration behavior"),(0,a.kt)("p",null,"With this option, you can set the actual behavior used when a new service is registered into the container. These options do not affect named registrations."),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("h3",{id:"skipduplications"},(0,a.kt)("inlineCode",{parentName:"h3"},"SkipDuplications")),(0,a.kt)("p",null,"The container will skip new registrations when the given ",(0,a.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"implementation type")," is already registered.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .WithRegistrationBehavior(\n Rules.RegistrationBehavior.SkipDuplications));\n")))),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("h3",{id:"throwexception"},(0,a.kt)("inlineCode",{parentName:"h3"},"ThrowException")),(0,a.kt)("p",null,"The container throws an ",(0,a.kt)("a",{parentName:"p",href:"/docs/diagnostics/validation#servicealreadyregisteredexception"},"exception")," when the given ",(0,a.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"implementation type")," is already registered.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .WithRegistrationBehavior(\n Rules.RegistrationBehavior.ThrowException));\n")))),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("h3",{id:"replaceexisting"},(0,a.kt)("inlineCode",{parentName:"h3"},"ReplaceExisting")),(0,a.kt)("p",null,"The container will replace the already ",(0,a.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-registration--registered-service"},"registered service")," with the given one when they have the same ",(0,a.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"implementation type"),".")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .WithRegistrationBehavior(\n Rules.RegistrationBehavior.ReplaceExisting));\n")))),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("h3",{id:"preserveduplications"},(0,a.kt)("inlineCode",{parentName:"h3"},"PreserveDuplications")),(0,a.kt)("p",null,"The container will keep registering the new services with the same ",(0,a.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"implementation type"),".")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .WithRegistrationBehavior(\n Rules.RegistrationBehavior.PreserveDuplications));\n")))),(0,a.kt)("h2",{id:"default-lifetime"},"Default lifetime"),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"With this option, you can set the default lifetime used when a service doesn't have a configured one.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .WithDefaultLifetime(Lifetimes.Scoped));\n")))),(0,a.kt)("h2",{id:"lifetime-validation"},"Lifetime validation"),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"With this option, you can enable or disable the life-span and ",(0,a.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#root-scope"},"root scope")," resolution ",(0,a.kt)("a",{parentName:"p",href:"/docs/diagnostics/validation#lifetime-validation"},"validation")," on the dependency tree.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .WithLifetimeValidation());\n")))),(0,a.kt)("h2",{id:"conventional-resolution"},"Conventional resolution"),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"With this option, you can enable or disable conventional resolution, which means the container treats the constructor/method parameter or member names as dependency names used by ",(0,a.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#named-resolution"},"named resolution"),".")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .TreatParameterAndMemberNameAsDependencyName());\n")))),(0,a.kt)("h2",{id:"using-named-service-for-un-named-requests"},"Using named service for un-named requests"),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"With this option, you can enable or disable the selection of named registrations when the resolution request is un-named but with the same type.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .WithNamedDependencyResolutionForUnNamedRequests());\n")))),(0,a.kt)("h2",{id:"default-value-injection"},"Default value injection"),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"With this option, you can enable or disable the default value injection.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .WithDefaultValueInjection());\n")))),(0,a.kt)("h2",{id:"unknown-type-resolution"},"Unknown type resolution"),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"With this option, you can enable or disable the resolution of unregistered types. You can also use a configurator delegate to configure the registrations the container will create from the unknown types.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .WithUnknownTypeResolution(config => config.AsImplementedTypes()));\n")))),(0,a.kt)("h2",{id:"custom-compiler"},"Custom compiler"),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"With this option, you can set an external expression tree compiler. It can be useful on platforms where the IL generator modules are not available; therefore, the expression compiler in Stashbox couldn't work.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .WithExpressionCompiler(\n Rules.ExpressionCompilers.MicrosoftExpressionCompiler));\n")))),(0,a.kt)("h2",{id:"re-build-singletons-in-child-containers"},"Re-build singletons in child containers"),(0,a.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"With this option, you can enable or disable the re-building of singletons in child containers. It allows the child containers to override singleton dependencies in the parent.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"new StashboxContainer(options => options\n .WithReBuildSingletonsInChildContainer());\n")))),(0,a.kt)("admonition",{type:"note"},(0,a.kt)("p",{parentName:"admonition"},"This feature is not affecting the already built singleton instances in the parent.")))}d.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/b53e7bc5.8724ef22.js b/assets/js/b53e7bc5.b0225f24.js similarity index 62% rename from assets/js/b53e7bc5.8724ef22.js rename to assets/js/b53e7bc5.b0225f24.js index 3fedcbd4..955311a9 100644 --- a/assets/js/b53e7bc5.8724ef22.js +++ b/assets/js/b53e7bc5.b0225f24.js @@ -1 +1 @@ -"use strict";(self.webpackChunkwebsite=self.webpackChunkwebsite||[]).push([[343],{3905:(e,n,t)=>{t.d(n,{Zo:()=>u,kt:()=>m});var a=t(7294);function r(e,n,t){return n in e?Object.defineProperty(e,n,{value:t,enumerable:!0,configurable:!0,writable:!0}):e[n]=t,e}function o(e,n){var t=Object.keys(e);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);n&&(a=a.filter((function(n){return Object.getOwnPropertyDescriptor(e,n).enumerable}))),t.push.apply(t,a)}return t}function i(e){for(var n=1;n=0||(r[t]=e[t]);return r}(e,n);if(Object.getOwnPropertySymbols){var o=Object.getOwnPropertySymbols(e);for(a=0;a=0||Object.prototype.propertyIsEnumerable.call(e,t)&&(r[t]=e[t])}return r}var s=a.createContext({}),c=function(e){var n=a.useContext(s),t=n;return e&&(t="function"==typeof e?e(n):i(i({},n),e)),t},u=function(e){var n=c(e.components);return a.createElement(s.Provider,{value:n},e.children)},p="mdxType",d={inlineCode:"code",wrapper:function(e){var n=e.children;return a.createElement(a.Fragment,{},n)}},f=a.forwardRef((function(e,n){var t=e.components,r=e.mdxType,o=e.originalType,s=e.parentName,u=l(e,["components","mdxType","originalType","parentName"]),p=c(t),f=r,m=p["".concat(s,".").concat(f)]||p[f]||d[f]||o;return t?a.createElement(m,i(i({ref:n},u),{},{components:t})):a.createElement(m,i({ref:n},u))}));function m(e,n){var t=arguments,r=n&&n.mdxType;if("string"==typeof e||r){var o=t.length,i=new Array(o);i[0]=f;var l={};for(var s in n)hasOwnProperty.call(n,s)&&(l[s]=n[s]);l.originalType=e,l[p]="string"==typeof e?e:r,i[1]=l;for(var c=2;c{t.d(n,{Z:()=>i});var a=t(7294),r=t(6010);const o="tabItem_Ymn6";function i(e){let{children:n,hidden:t,className:i}=e;return a.createElement("div",{role:"tabpanel",className:(0,r.Z)(o,i),hidden:t},n)}},4866:(e,n,t)=>{t.d(n,{Z:()=>S});var a=t(7462),r=t(7294),o=t(6010),i=t(2466),l=t(6550),s=t(1980),c=t(7392),u=t(12);function p(e){return function(e){return r.Children.map(e,(e=>{if(!e||(0,r.isValidElement)(e)&&function(e){const{props:n}=e;return!!n&&"object"==typeof n&&"value"in n}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}(e).map((e=>{let{props:{value:n,label:t,attributes:a,default:r}}=e;return{value:n,label:t,attributes:a,default:r}}))}function d(e){const{values:n,children:t}=e;return(0,r.useMemo)((()=>{const e=n??p(t);return function(e){const n=(0,c.l)(e,((e,n)=>e.value===n.value));if(n.length>0)throw new Error(`Docusaurus error: Duplicate values "${n.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[n,t])}function f(e){let{value:n,tabValues:t}=e;return t.some((e=>e.value===n))}function m(e){let{queryString:n=!1,groupId:t}=e;const a=(0,l.k6)(),o=function(e){let{queryString:n=!1,groupId:t}=e;if("string"==typeof n)return n;if(!1===n)return null;if(!0===n&&!t)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return t??null}({queryString:n,groupId:t});return[(0,s._X)(o),(0,r.useCallback)((e=>{if(!o)return;const n=new URLSearchParams(a.location.search);n.set(o,e),a.replace({...a.location,search:n.toString()})}),[o,a])]}function v(e){const{defaultValue:n,queryString:t=!1,groupId:a}=e,o=d(e),[i,l]=(0,r.useState)((()=>function(e){let{defaultValue:n,tabValues:t}=e;if(0===t.length)throw new Error("Docusaurus error: the component requires at least one children component");if(n){if(!f({value:n,tabValues:t}))throw new Error(`Docusaurus error: The has a defaultValue "${n}" but none of its children has the corresponding value. Available values are: ${t.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return n}const a=t.find((e=>e.default))??t[0];if(!a)throw new Error("Unexpected error: 0 tabValues");return a.value}({defaultValue:n,tabValues:o}))),[s,c]=m({queryString:t,groupId:a}),[p,v]=function(e){let{groupId:n}=e;const t=function(e){return e?`docusaurus.tab.${e}`:null}(n),[a,o]=(0,u.Nk)(t);return[a,(0,r.useCallback)((e=>{t&&o.set(e)}),[t,o])]}({groupId:a}),h=(()=>{const e=s??p;return f({value:e,tabValues:o})?e:null})();(0,r.useLayoutEffect)((()=>{h&&l(h)}),[h]);return{selectedValue:i,selectValue:(0,r.useCallback)((e=>{if(!f({value:e,tabValues:o}))throw new Error(`Can't select invalid tab value=${e}`);l(e),c(e),v(e)}),[c,v,o]),tabValues:o}}var h=t(2389);const b="tabList__CuJ",g="tabItem_LNqP";function y(e){let{className:n,block:t,selectedValue:l,selectValue:s,tabValues:c}=e;const u=[],{blockElementScrollPositionUntilNextRender:p}=(0,i.o5)(),d=e=>{const n=e.currentTarget,t=u.indexOf(n),a=c[t].value;a!==l&&(p(n),s(a))},f=e=>{let n=null;switch(e.key){case"Enter":d(e);break;case"ArrowRight":{const t=u.indexOf(e.currentTarget)+1;n=u[t]??u[0];break}case"ArrowLeft":{const t=u.indexOf(e.currentTarget)-1;n=u[t]??u[u.length-1];break}}n?.focus()};return r.createElement("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,o.Z)("tabs",{"tabs--block":t},n)},c.map((e=>{let{value:n,label:t,attributes:i}=e;return r.createElement("li",(0,a.Z)({role:"tab",tabIndex:l===n?0:-1,"aria-selected":l===n,key:n,ref:e=>u.push(e),onKeyDown:f,onClick:d},i,{className:(0,o.Z)("tabs__item",g,i?.className,{"tabs__item--active":l===n})}),t??n)})))}function k(e){let{lazy:n,children:t,selectedValue:a}=e;const o=(Array.isArray(t)?t:[t]).filter(Boolean);if(n){const e=o.find((e=>e.props.value===a));return e?(0,r.cloneElement)(e,{className:"margin-top--md"}):null}return r.createElement("div",{className:"margin-top--md"},o.map(((e,n)=>(0,r.cloneElement)(e,{key:n,hidden:e.props.value!==a}))))}function w(e){const n=v(e);return r.createElement("div",{className:(0,o.Z)("tabs-container",b)},r.createElement(y,(0,a.Z)({},e,n)),r.createElement(k,(0,a.Z)({},e,n)))}function S(e){const n=(0,h.Z)();return r.createElement(w,(0,a.Z)({key:String(n)},e))}},6184:(e,n,t)=>{t.r(n),t.d(n,{assets:()=>u,contentTitle:()=>s,default:()=>f,frontMatter:()=>l,metadata:()=>c,toc:()=>p});var a=t(7462),r=(t(7294),t(3905)),o=t(4866),i=t(5162);const l={},s="Special resolution cases",c={unversionedId:"advanced/special-resolution-cases",id:"advanced/special-resolution-cases",title:"Special resolution cases",description:"Unknown type resolution",source:"@site/docs/advanced/special-resolution-cases.md",sourceDirName:"advanced",slug:"/advanced/special-resolution-cases",permalink:"/stashbox/docs/advanced/special-resolution-cases",draft:!1,editUrl:"https://github.com/z4kn4fein/stashbox/edit/master/docs/docs/advanced/special-resolution-cases.md",tags:[],version:"current",lastUpdatedBy:"dependabot[bot]",lastUpdatedAt:1697644420,formattedLastUpdatedAt:"Oct 18, 2023",frontMatter:{},sidebar:"docs",previous:{title:"Child containers",permalink:"/stashbox/docs/advanced/child-containers"},next:{title:"Validation",permalink:"/stashbox/docs/diagnostics/validation"}},u={},p=[{value:"Unknown type resolution",id:"unknown-type-resolution",level:2},{value:"Default value injection",id:"default-value-injection",level:2},{value:"Optional value injection",id:"optional-value-injection",level:2}],d={toc:p};function f(e){let{components:n,...t}=e;return(0,r.kt)("wrapper",(0,a.Z)({},d,t,{components:n,mdxType:"MDXLayout"}),(0,r.kt)("h1",{id:"special-resolution-cases"},"Special resolution cases"),(0,r.kt)("h2",{id:"unknown-type-resolution"},"Unknown type resolution"),(0,r.kt)("p",null,"When this ",(0,r.kt)("a",{parentName:"p",href:"/docs/configuration/container-configuration#unknown-type-resolution"},"feature")," is enabled, the container will try to resolve unregistered types by registering them using a pre-defined configuration delegate."),(0,r.kt)(o.Z,{mdxType:"Tabs"},(0,r.kt)(i.Z,{value:"Default",label:"Default",mdxType:"TabItem"},(0,r.kt)("p",null,"Without a registration configuration, the container can resolve only non-interface and non-abstract unknown types. In the following example,\nthe container creates an implicit registration for ",(0,r.kt)("inlineCode",{parentName:"p"},"Dependency")," and injects its instance into ",(0,r.kt)("inlineCode",{parentName:"p"},"Service"),"."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"class Dependency { }\n\nclass Service \n{\n public Service(Dependency dependency)\n { } \n}\n\nvar container = new StashboxContainer(config => config\n .WithUnknownTypeResolution());\n\ncontainer.Register();\n\nvar service = container.Resolve();\n"))),(0,r.kt)(i.Z,{value:"With registration configuration",label:"With registration configuration",mdxType:"TabItem"},(0,r.kt)("p",null,"With a registration configuration, you can control how an unknown type's individual registration should behave. You can also react to a service resolution request. In the following example, we tell the container that if it finds an unregistered ",(0,r.kt)("inlineCode",{parentName:"p"},"IDependency")," for the first time, that should be mapped to ",(0,r.kt)("inlineCode",{parentName:"p"},"Dependency")," and have a singleton lifetime. Next time, when the container comes across this service, it will use the registration created at the first request."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"interface IDependency { }\n\nclass Dependency : IDependency { }\n\nclass Service \n{\n public Service(IDependency dependency)\n { } \n}\n\nvar container = new StashboxContainer(config => config\n .WithUnknownTypeResolution(options => \n {\n if(options.ServiceType == typeof(IDependency))\n {\n options.SetImplementationType(typeof(Dependency))\n .WithLifetime(Lifetimes.Singleton);\n }\n }));\n\ncontainer.Register();\n\nvar service = container.Resolve();\n")))),(0,r.kt)("h2",{id:"default-value-injection"},"Default value injection"),(0,r.kt)("p",null,"When this ",(0,r.kt)("a",{parentName:"p",href:"/docs/configuration/container-configuration#default-value-injection"},"feature")," is enabled, the container will resolve unknown primitive dependencies with their default value."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"class Person \n{\n public Person(string name, int age) { }\n}\n\nvar container = new StashboxContainer(config => config\n .WithDefaultValueInjection());\n// the name parameter will be null and the age will be 0.\nvar person = container.Resolve();\n")),(0,r.kt)("admonition",{type:"note"},(0,r.kt)("p",{parentName:"admonition"},"Unknown reference types are resolved to ",(0,r.kt)("inlineCode",{parentName:"p"},"null")," only in properties and fields.")),(0,r.kt)("h2",{id:"optional-value-injection"},"Optional value injection"),(0,r.kt)("p",null,"Stashbox respects the optional value of each constructor and method argument."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"class Person \n{\n public Person(string name = null, int age = 54, IContact contact = null) { }\n}\n\n// the name will be null \n// the age will be 54.\n// the contact will be null.\nvar person = container.Resolve();\n")))}f.isMDXComponent=!0}}]); \ No newline at end of file +"use strict";(self.webpackChunkwebsite=self.webpackChunkwebsite||[]).push([[343],{3905:(e,n,t)=>{t.d(n,{Zo:()=>u,kt:()=>m});var a=t(7294);function r(e,n,t){return n in e?Object.defineProperty(e,n,{value:t,enumerable:!0,configurable:!0,writable:!0}):e[n]=t,e}function o(e,n){var t=Object.keys(e);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);n&&(a=a.filter((function(n){return Object.getOwnPropertyDescriptor(e,n).enumerable}))),t.push.apply(t,a)}return t}function i(e){for(var n=1;n=0||(r[t]=e[t]);return r}(e,n);if(Object.getOwnPropertySymbols){var o=Object.getOwnPropertySymbols(e);for(a=0;a=0||Object.prototype.propertyIsEnumerable.call(e,t)&&(r[t]=e[t])}return r}var s=a.createContext({}),c=function(e){var n=a.useContext(s),t=n;return e&&(t="function"==typeof e?e(n):i(i({},n),e)),t},u=function(e){var n=c(e.components);return a.createElement(s.Provider,{value:n},e.children)},p="mdxType",d={inlineCode:"code",wrapper:function(e){var n=e.children;return a.createElement(a.Fragment,{},n)}},f=a.forwardRef((function(e,n){var t=e.components,r=e.mdxType,o=e.originalType,s=e.parentName,u=l(e,["components","mdxType","originalType","parentName"]),p=c(t),f=r,m=p["".concat(s,".").concat(f)]||p[f]||d[f]||o;return t?a.createElement(m,i(i({ref:n},u),{},{components:t})):a.createElement(m,i({ref:n},u))}));function m(e,n){var t=arguments,r=n&&n.mdxType;if("string"==typeof e||r){var o=t.length,i=new Array(o);i[0]=f;var l={};for(var s in n)hasOwnProperty.call(n,s)&&(l[s]=n[s]);l.originalType=e,l[p]="string"==typeof e?e:r,i[1]=l;for(var c=2;c{t.d(n,{Z:()=>i});var a=t(7294),r=t(6010);const o="tabItem_Ymn6";function i(e){let{children:n,hidden:t,className:i}=e;return a.createElement("div",{role:"tabpanel",className:(0,r.Z)(o,i),hidden:t},n)}},4866:(e,n,t)=>{t.d(n,{Z:()=>S});var a=t(7462),r=t(7294),o=t(6010),i=t(2466),l=t(6550),s=t(1980),c=t(7392),u=t(12);function p(e){return function(e){return r.Children.map(e,(e=>{if(!e||(0,r.isValidElement)(e)&&function(e){const{props:n}=e;return!!n&&"object"==typeof n&&"value"in n}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}(e).map((e=>{let{props:{value:n,label:t,attributes:a,default:r}}=e;return{value:n,label:t,attributes:a,default:r}}))}function d(e){const{values:n,children:t}=e;return(0,r.useMemo)((()=>{const e=n??p(t);return function(e){const n=(0,c.l)(e,((e,n)=>e.value===n.value));if(n.length>0)throw new Error(`Docusaurus error: Duplicate values "${n.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[n,t])}function f(e){let{value:n,tabValues:t}=e;return t.some((e=>e.value===n))}function m(e){let{queryString:n=!1,groupId:t}=e;const a=(0,l.k6)(),o=function(e){let{queryString:n=!1,groupId:t}=e;if("string"==typeof n)return n;if(!1===n)return null;if(!0===n&&!t)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return t??null}({queryString:n,groupId:t});return[(0,s._X)(o),(0,r.useCallback)((e=>{if(!o)return;const n=new URLSearchParams(a.location.search);n.set(o,e),a.replace({...a.location,search:n.toString()})}),[o,a])]}function v(e){const{defaultValue:n,queryString:t=!1,groupId:a}=e,o=d(e),[i,l]=(0,r.useState)((()=>function(e){let{defaultValue:n,tabValues:t}=e;if(0===t.length)throw new Error("Docusaurus error: the component requires at least one children component");if(n){if(!f({value:n,tabValues:t}))throw new Error(`Docusaurus error: The has a defaultValue "${n}" but none of its children has the corresponding value. Available values are: ${t.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return n}const a=t.find((e=>e.default))??t[0];if(!a)throw new Error("Unexpected error: 0 tabValues");return a.value}({defaultValue:n,tabValues:o}))),[s,c]=m({queryString:t,groupId:a}),[p,v]=function(e){let{groupId:n}=e;const t=function(e){return e?`docusaurus.tab.${e}`:null}(n),[a,o]=(0,u.Nk)(t);return[a,(0,r.useCallback)((e=>{t&&o.set(e)}),[t,o])]}({groupId:a}),h=(()=>{const e=s??p;return f({value:e,tabValues:o})?e:null})();(0,r.useLayoutEffect)((()=>{h&&l(h)}),[h]);return{selectedValue:i,selectValue:(0,r.useCallback)((e=>{if(!f({value:e,tabValues:o}))throw new Error(`Can't select invalid tab value=${e}`);l(e),c(e),v(e)}),[c,v,o]),tabValues:o}}var h=t(2389);const b="tabList__CuJ",g="tabItem_LNqP";function y(e){let{className:n,block:t,selectedValue:l,selectValue:s,tabValues:c}=e;const u=[],{blockElementScrollPositionUntilNextRender:p}=(0,i.o5)(),d=e=>{const n=e.currentTarget,t=u.indexOf(n),a=c[t].value;a!==l&&(p(n),s(a))},f=e=>{let n=null;switch(e.key){case"Enter":d(e);break;case"ArrowRight":{const t=u.indexOf(e.currentTarget)+1;n=u[t]??u[0];break}case"ArrowLeft":{const t=u.indexOf(e.currentTarget)-1;n=u[t]??u[u.length-1];break}}n?.focus()};return r.createElement("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,o.Z)("tabs",{"tabs--block":t},n)},c.map((e=>{let{value:n,label:t,attributes:i}=e;return r.createElement("li",(0,a.Z)({role:"tab",tabIndex:l===n?0:-1,"aria-selected":l===n,key:n,ref:e=>u.push(e),onKeyDown:f,onClick:d},i,{className:(0,o.Z)("tabs__item",g,i?.className,{"tabs__item--active":l===n})}),t??n)})))}function k(e){let{lazy:n,children:t,selectedValue:a}=e;const o=(Array.isArray(t)?t:[t]).filter(Boolean);if(n){const e=o.find((e=>e.props.value===a));return e?(0,r.cloneElement)(e,{className:"margin-top--md"}):null}return r.createElement("div",{className:"margin-top--md"},o.map(((e,n)=>(0,r.cloneElement)(e,{key:n,hidden:e.props.value!==a}))))}function w(e){const n=v(e);return r.createElement("div",{className:(0,o.Z)("tabs-container",b)},r.createElement(y,(0,a.Z)({},e,n)),r.createElement(k,(0,a.Z)({},e,n)))}function S(e){const n=(0,h.Z)();return r.createElement(w,(0,a.Z)({key:String(n)},e))}},6184:(e,n,t)=>{t.r(n),t.d(n,{assets:()=>u,contentTitle:()=>s,default:()=>f,frontMatter:()=>l,metadata:()=>c,toc:()=>p});var a=t(7462),r=(t(7294),t(3905)),o=t(4866),i=t(5162);const l={},s="Special resolution cases",c={unversionedId:"advanced/special-resolution-cases",id:"advanced/special-resolution-cases",title:"Special resolution cases",description:"Unknown type resolution",source:"@site/docs/advanced/special-resolution-cases.md",sourceDirName:"advanced",slug:"/advanced/special-resolution-cases",permalink:"/stashbox/docs/advanced/special-resolution-cases",draft:!1,editUrl:"https://github.com/z4kn4fein/stashbox/edit/master/docs/docs/advanced/special-resolution-cases.md",tags:[],version:"current",lastUpdatedBy:"dependabot[bot]",lastUpdatedAt:1699538162,formattedLastUpdatedAt:"Nov 9, 2023",frontMatter:{},sidebar:"docs",previous:{title:"Child containers",permalink:"/stashbox/docs/advanced/child-containers"},next:{title:"Validation",permalink:"/stashbox/docs/diagnostics/validation"}},u={},p=[{value:"Unknown type resolution",id:"unknown-type-resolution",level:2},{value:"Default value injection",id:"default-value-injection",level:2},{value:"Optional value injection",id:"optional-value-injection",level:2}],d={toc:p};function f(e){let{components:n,...t}=e;return(0,r.kt)("wrapper",(0,a.Z)({},d,t,{components:n,mdxType:"MDXLayout"}),(0,r.kt)("h1",{id:"special-resolution-cases"},"Special resolution cases"),(0,r.kt)("h2",{id:"unknown-type-resolution"},"Unknown type resolution"),(0,r.kt)("p",null,"When this ",(0,r.kt)("a",{parentName:"p",href:"/docs/configuration/container-configuration#unknown-type-resolution"},"feature")," is enabled, the container will try to resolve unregistered types by registering them using a pre-defined configuration delegate."),(0,r.kt)(o.Z,{mdxType:"Tabs"},(0,r.kt)(i.Z,{value:"Default",label:"Default",mdxType:"TabItem"},(0,r.kt)("p",null,"Without a registration configuration, the container can resolve only non-interface and non-abstract unknown types. In the following example,\nthe container creates an implicit registration for ",(0,r.kt)("inlineCode",{parentName:"p"},"Dependency")," and injects its instance into ",(0,r.kt)("inlineCode",{parentName:"p"},"Service"),"."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"class Dependency { }\n\nclass Service \n{\n public Service(Dependency dependency)\n { } \n}\n\nvar container = new StashboxContainer(config => config\n .WithUnknownTypeResolution());\n\ncontainer.Register();\n\nvar service = container.Resolve();\n"))),(0,r.kt)(i.Z,{value:"With registration configuration",label:"With registration configuration",mdxType:"TabItem"},(0,r.kt)("p",null,"With a registration configuration, you can control how an unknown type's individual registration should behave. You can also react to a service resolution request. In the following example, we tell the container that if it finds an unregistered ",(0,r.kt)("inlineCode",{parentName:"p"},"IDependency")," for the first time, that should be mapped to ",(0,r.kt)("inlineCode",{parentName:"p"},"Dependency")," and have a singleton lifetime. Next time, when the container comes across this service, it will use the registration created at the first request."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"interface IDependency { }\n\nclass Dependency : IDependency { }\n\nclass Service \n{\n public Service(IDependency dependency)\n { } \n}\n\nvar container = new StashboxContainer(config => config\n .WithUnknownTypeResolution(options => \n {\n if(options.ServiceType == typeof(IDependency))\n {\n options.SetImplementationType(typeof(Dependency))\n .WithLifetime(Lifetimes.Singleton);\n }\n }));\n\ncontainer.Register();\n\nvar service = container.Resolve();\n")))),(0,r.kt)("h2",{id:"default-value-injection"},"Default value injection"),(0,r.kt)("p",null,"When this ",(0,r.kt)("a",{parentName:"p",href:"/docs/configuration/container-configuration#default-value-injection"},"feature")," is enabled, the container will resolve unknown primitive dependencies with their default value."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"class Person \n{\n public Person(string name, int age) { }\n}\n\nvar container = new StashboxContainer(config => config\n .WithDefaultValueInjection());\n// the name parameter will be null and the age will be 0.\nvar person = container.Resolve();\n")),(0,r.kt)("admonition",{type:"note"},(0,r.kt)("p",{parentName:"admonition"},"Unknown reference types are resolved to ",(0,r.kt)("inlineCode",{parentName:"p"},"null")," only in properties and fields.")),(0,r.kt)("h2",{id:"optional-value-injection"},"Optional value injection"),(0,r.kt)("p",null,"Stashbox respects the optional value of each constructor and method argument."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"class Person \n{\n public Person(string name = null, int age = 54, IContact contact = null) { }\n}\n\n// the name will be null \n// the age will be 54.\n// the contact will be null.\nvar person = container.Resolve();\n")))}f.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/dd45a7f1.4a7501f3.js b/assets/js/dd45a7f1.4a7501f3.js deleted file mode 100644 index 0ca4454b..00000000 --- a/assets/js/dd45a7f1.4a7501f3.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkwebsite=self.webpackChunkwebsite||[]).push([[652],{3905:(e,t,n)=>{n.d(t,{Zo:()=>c,kt:()=>h});var r=n(7294);function a(e,t,n){return t in e?Object.defineProperty(e,t,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[t]=n,e}function i(e,t){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);t&&(r=r.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),n.push.apply(n,r)}return n}function s(e){for(var t=1;t=0||(a[n]=e[n]);return a}(e,t);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(e);for(r=0;r=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(a[n]=e[n])}return a}var l=r.createContext({}),p=function(e){var t=r.useContext(l),n=t;return e&&(n="function"==typeof e?e(t):s(s({},t),e)),n},c=function(e){var t=p(e.components);return r.createElement(l.Provider,{value:t},e.children)},d="mdxType",m={inlineCode:"code",wrapper:function(e){var t=e.children;return r.createElement(r.Fragment,{},t)}},u=r.forwardRef((function(e,t){var n=e.components,a=e.mdxType,i=e.originalType,l=e.parentName,c=o(e,["components","mdxType","originalType","parentName"]),d=p(n),u=a,h=d["".concat(l,".").concat(u)]||d[u]||m[u]||i;return n?r.createElement(h,s(s({ref:t},c),{},{components:n})):r.createElement(h,s({ref:t},c))}));function h(e,t){var n=arguments,a=t&&t.mdxType;if("string"==typeof e||a){var i=n.length,s=new Array(i);s[0]=u;var o={};for(var l in t)hasOwnProperty.call(t,l)&&(o[l]=t[l]);o.originalType=e,o[d]="string"==typeof e?e:a,s[1]=o;for(var p=2;p{n.d(t,{Z:()=>o});var r=n(7294);const a="codeDescContainer_ie8f",i="desc_jyqI",s="example_eYlF";function o(e){let{children:t}=e,n=r.Children.toArray(t).filter((e=>e));return r.createElement("div",{className:a},r.createElement("div",{className:i},n[0]),r.createElement("div",{className:s},n[1]))}},969:(e,t,n)=>{n.r(t),n.d(t,{assets:()=>p,contentTitle:()=>o,default:()=>m,frontMatter:()=>s,metadata:()=>l,toc:()=>c});var r=n(7462),a=(n(7294),n(3905)),i=n(8846);const s={},o="Glossary",l={unversionedId:"getting-started/glossary",id:"getting-started/glossary",title:"Glossary",description:"The following terms and definitions are used in this documentation.",source:"@site/docs/getting-started/glossary.md",sourceDirName:"getting-started",slug:"/getting-started/glossary",permalink:"/stashbox/docs/getting-started/glossary",draft:!1,editUrl:"https://github.com/z4kn4fein/stashbox/edit/master/docs/docs/getting-started/glossary.md",tags:[],version:"current",lastUpdatedBy:"dependabot[bot]",lastUpdatedAt:1697644420,formattedLastUpdatedAt:"Oct 18, 2023",frontMatter:{},sidebar:"docs",previous:{title:"Introduction",permalink:"/stashbox/docs/getting-started/introduction"},next:{title:"Basic usage",permalink:"/stashbox/docs/guides/basics"}},p={},c=[{value:"Service type | Implementation type",id:"service-type--implementation-type",level:2},{value:"Service registration | Registered service",id:"service-registration--registered-service",level:2},{value:"Injectable dependency",id:"injectable-dependency",level:2},{value:"Resolution tree",id:"resolution-tree",level:2},{value:"Dependency resolver",id:"dependency-resolver",level:2},{value:"Root scope",id:"root-scope",level:2},{value:"Named resolution",id:"named-resolution",level:2},{value:"Self registration",id:"self-registration",level:2}],d={toc:c};function m(e){let{components:t,...n}=e;return(0,a.kt)("wrapper",(0,r.Z)({},d,n,{components:t,mdxType:"MDXLayout"}),(0,a.kt)("h1",{id:"glossary"},"Glossary"),(0,a.kt)("p",null,"The following terms and definitions are used in this documentation."),(0,a.kt)("h2",{id:"service-type--implementation-type"},"Service type | Implementation type"),(0,a.kt)("p",null,"The ",(0,a.kt)("em",{parentName:"p"},"Service type")," is usually an interface or an abstract class type used for service resolution or dependency injection. The ",(0,a.kt)("em",{parentName:"p"},"Implementation type")," is the actual type registered to the ",(0,a.kt)("em",{parentName:"p"},"Service type"),". A registration maps the ",(0,a.kt)("em",{parentName:"p"},"Service type")," to an ",(0,a.kt)("em",{parentName:"p"},"Implementation type"),". The ",(0,a.kt)("em",{parentName:"p"},"Implementation type")," must implement or extend the ",(0,a.kt)("em",{parentName:"p"},"Service type"),". "),(0,a.kt)(i.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"Example where a ",(0,a.kt)("em",{parentName:"p"},"Service type")," is mapped to an ",(0,a.kt)("em",{parentName:"p"},"Implementation type"),":")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register();\n")))),(0,a.kt)(i.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"The ",(0,a.kt)("em",{parentName:"p"},"Service type")," used for requesting a service from the container:")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"container.Resolve(); // returns Implementation\n")))),(0,a.kt)("h2",{id:"service-registration--registered-service"},"Service registration | Registered service"),(0,a.kt)("p",null,"It's an entity created by Stashbox when a service is registered. The service registration stores required information about how to instantiate the service, e.g., reflected type information, name, lifetime, conditions, and more."),(0,a.kt)(i.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"In this example, we are registering a named service. The container will create a service registration entity to store the type mapping and the name. During resolution, the container will find the registration by checking for the ",(0,a.kt)("em",{parentName:"p"},"Service type")," and the ",(0,a.kt)("em",{parentName:"p"},"name"),".")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},'// the registration entity will look like: \n// IService => Implementation, name: Example\ncontainer.Register("Example");\nvar service = container.Resolve("Example");\n')))),(0,a.kt)("h2",{id:"injectable-dependency"},"Injectable dependency"),(0,a.kt)(i.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"It's a constructor/method argument or a property/field of a registered ",(0,a.kt)("em",{parentName:"p"},"Implementation type")," that gets evaluated (",(0,a.kt)("em",{parentName:"p"},"injected"),") by Stashbox during the service's construction."),(0,a.kt)("p",null,"In this example, ",(0,a.kt)("inlineCode",{parentName:"p"},"Implementation")," has an ",(0,a.kt)("inlineCode",{parentName:"p"},"IDependency")," ",(0,a.kt)("em",{parentName:"p"},"injectable dependency")," in its constructor.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"class Implementation : IService\n{\n public Implementation(IDependency dependency) \n { }\n}\n")))),(0,a.kt)("h2",{id:"resolution-tree"},"Resolution tree"),(0,a.kt)("p",null,"It's the structural representation of a service's resolution process. It describes the instantiation order of the dependencies required to resolve the desired type."),(0,a.kt)("p",null,"Let's see through an example:"),(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"class A\n{\n public A(B b, C c) { }\n}\n\nclass B\n{\n public B(C c, D d) { }\n}\n\nclass C { }\nclass D { }\n")),(0,a.kt)("p",null,"When we request the service ",(0,a.kt)("inlineCode",{parentName:"p"},"A"),", the container constructs the following resolution tree based on the dependencies and sub-dependencies."),(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre"}," A\n / \\\n B C\n / \\\n C D\n")),(0,a.kt)("p",null,"The container instantiates those services first that don't have any dependencies. ",(0,a.kt)("inlineCode",{parentName:"p"},"C")," and ",(0,a.kt)("inlineCode",{parentName:"p"},"D")," will be injected into ",(0,a.kt)("inlineCode",{parentName:"p"},"B"),". Then, a new ",(0,a.kt)("inlineCode",{parentName:"p"},"C")," is instantiated (if it's ",(0,a.kt)("a",{parentName:"p",href:"/docs/guides/lifetimes#transient-lifetime"},"transient"),") and injected into ",(0,a.kt)("inlineCode",{parentName:"p"},"A")," along with the previously created ",(0,a.kt)("inlineCode",{parentName:"p"},"B"),"."),(0,a.kt)("h2",{id:"dependency-resolver"},"Dependency resolver"),(0,a.kt)("p",null,"It's the container itself or the ",(0,a.kt)("a",{parentName:"p",href:"/docs/guides/scopes"},"current scope"),", depending on which was asked to resolve a particular service. They are both implementing Stashbox's ",(0,a.kt)("inlineCode",{parentName:"p"},"IDependencyResolver")," and the .NET framework's ",(0,a.kt)("inlineCode",{parentName:"p"},"IServiceProvider")," interface and can be used for service resolution."),(0,a.kt)("admonition",{type:"info"},(0,a.kt)("p",{parentName:"admonition"},"Stashbox implicitly injects the ",(0,a.kt)("a",{parentName:"p",href:"/docs/guides/scopes"},"current scope")," wherever ",(0,a.kt)("inlineCode",{parentName:"p"},"IDependencyResolver")," or ",(0,a.kt)("inlineCode",{parentName:"p"},"IServiceProvider")," is requested.")),(0,a.kt)("h2",{id:"root-scope"},"Root scope"),(0,a.kt)("p",null,"It's the ",(0,a.kt)("a",{parentName:"p",href:"/docs/guides/scopes"},"main scope")," created inside every container instance. It stores and handles the lifetime of all singletons. It's the base of each subsequent scope created by the container with the ",(0,a.kt)("inlineCode",{parentName:"p"},".BeginScope()")," method."),(0,a.kt)("admonition",{type:"caution"},(0,a.kt)("p",{parentName:"admonition"},(0,a.kt)("a",{parentName:"p",href:"/docs/guides/lifetimes#scoped-lifetime"},"Scoped services")," requested from the container (and not from a ",(0,a.kt)("a",{parentName:"p",href:"/docs/guides/scopes"},"scope"),") are managed by the root scope. This can lead to issues because their lifetime will effectively switch to singleton. Always be sure that you don't resolve scoped services directly from the container, only from a ",(0,a.kt)("a",{parentName:"p",href:"/docs/guides/scopes"},"scope"),". This case is monitored by the ",(0,a.kt)("a",{parentName:"p",href:"/docs/diagnostics/validation#lifetime-validation"},"lifetime")," validation rule when it's ",(0,a.kt)("a",{parentName:"p",href:"/docs/configuration/container-configuration#lifetime-validation"},"enabled"),". ")),(0,a.kt)("h2",{id:"named-resolution"},"Named resolution"),(0,a.kt)(i.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"It's a resolution request for a named service. The same applies, when the container sees a dependency in the resolution tree with a name (set by ",(0,a.kt)("a",{parentName:"p",href:"/docs/guides/service-resolution#attributes"},"attributes")," or ",(0,a.kt)("a",{parentName:"p",href:"/docs/guides/service-resolution#dependency-binding"},"bindings"),"); it will search for a matching ",(0,a.kt)("a",{parentName:"p",href:"/docs/guides/basics#named-registration"},"Named registration")," to inject.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register("Example");\n// the named request.\nvar service = container.Resolve("Example");\n')))),(0,a.kt)("h2",{id:"self-registration"},"Self registration"),(0,a.kt)(i.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"It's a service registration that's mapped to itself. This means its service and implementation type is the same.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"// equivalent to container.Register();\ncontainer.Register();\n")))))}m.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/dd45a7f1.ea398d3a.js b/assets/js/dd45a7f1.ea398d3a.js new file mode 100644 index 00000000..ea4280fd --- /dev/null +++ b/assets/js/dd45a7f1.ea398d3a.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkwebsite=self.webpackChunkwebsite||[]).push([[652],{3905:(e,t,n)=>{n.d(t,{Zo:()=>c,kt:()=>h});var r=n(7294);function a(e,t,n){return t in e?Object.defineProperty(e,t,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[t]=n,e}function i(e,t){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);t&&(r=r.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),n.push.apply(n,r)}return n}function s(e){for(var t=1;t=0||(a[n]=e[n]);return a}(e,t);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(e);for(r=0;r=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(a[n]=e[n])}return a}var l=r.createContext({}),p=function(e){var t=r.useContext(l),n=t;return e&&(n="function"==typeof e?e(t):s(s({},t),e)),n},c=function(e){var t=p(e.components);return r.createElement(l.Provider,{value:t},e.children)},d="mdxType",m={inlineCode:"code",wrapper:function(e){var t=e.children;return r.createElement(r.Fragment,{},t)}},u=r.forwardRef((function(e,t){var n=e.components,a=e.mdxType,i=e.originalType,l=e.parentName,c=o(e,["components","mdxType","originalType","parentName"]),d=p(n),u=a,h=d["".concat(l,".").concat(u)]||d[u]||m[u]||i;return n?r.createElement(h,s(s({ref:t},c),{},{components:n})):r.createElement(h,s({ref:t},c))}));function h(e,t){var n=arguments,a=t&&t.mdxType;if("string"==typeof e||a){var i=n.length,s=new Array(i);s[0]=u;var o={};for(var l in t)hasOwnProperty.call(t,l)&&(o[l]=t[l]);o.originalType=e,o[d]="string"==typeof e?e:a,s[1]=o;for(var p=2;p{n.d(t,{Z:()=>o});var r=n(7294);const a="codeDescContainer_ie8f",i="desc_jyqI",s="example_eYlF";function o(e){let{children:t}=e,n=r.Children.toArray(t).filter((e=>e));return r.createElement("div",{className:a},r.createElement("div",{className:i},n[0]),r.createElement("div",{className:s},n[1]))}},969:(e,t,n)=>{n.r(t),n.d(t,{assets:()=>p,contentTitle:()=>o,default:()=>m,frontMatter:()=>s,metadata:()=>l,toc:()=>c});var r=n(7462),a=(n(7294),n(3905)),i=n(8846);const s={},o="Glossary",l={unversionedId:"getting-started/glossary",id:"getting-started/glossary",title:"Glossary",description:"The following terms and definitions are used in this documentation.",source:"@site/docs/getting-started/glossary.md",sourceDirName:"getting-started",slug:"/getting-started/glossary",permalink:"/stashbox/docs/getting-started/glossary",draft:!1,editUrl:"https://github.com/z4kn4fein/stashbox/edit/master/docs/docs/getting-started/glossary.md",tags:[],version:"current",lastUpdatedBy:"dependabot[bot]",lastUpdatedAt:1699538162,formattedLastUpdatedAt:"Nov 9, 2023",frontMatter:{},sidebar:"docs",previous:{title:"Introduction",permalink:"/stashbox/docs/getting-started/introduction"},next:{title:"Basic usage",permalink:"/stashbox/docs/guides/basics"}},p={},c=[{value:"Service type | Implementation type",id:"service-type--implementation-type",level:2},{value:"Service registration | Registered service",id:"service-registration--registered-service",level:2},{value:"Injectable dependency",id:"injectable-dependency",level:2},{value:"Resolution tree",id:"resolution-tree",level:2},{value:"Dependency resolver",id:"dependency-resolver",level:2},{value:"Root scope",id:"root-scope",level:2},{value:"Named resolution",id:"named-resolution",level:2},{value:"Self registration",id:"self-registration",level:2}],d={toc:c};function m(e){let{components:t,...n}=e;return(0,a.kt)("wrapper",(0,r.Z)({},d,n,{components:t,mdxType:"MDXLayout"}),(0,a.kt)("h1",{id:"glossary"},"Glossary"),(0,a.kt)("p",null,"The following terms and definitions are used in this documentation."),(0,a.kt)("h2",{id:"service-type--implementation-type"},"Service type | Implementation type"),(0,a.kt)("p",null,"The ",(0,a.kt)("em",{parentName:"p"},"Service type")," is usually an interface or an abstract class type used for service resolution or dependency injection. The ",(0,a.kt)("em",{parentName:"p"},"Implementation type")," is the actual type registered to the ",(0,a.kt)("em",{parentName:"p"},"Service type"),". A registration maps the ",(0,a.kt)("em",{parentName:"p"},"Service type")," to an ",(0,a.kt)("em",{parentName:"p"},"Implementation type"),". The ",(0,a.kt)("em",{parentName:"p"},"Implementation type")," must implement or extend the ",(0,a.kt)("em",{parentName:"p"},"Service type"),". "),(0,a.kt)(i.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"Example where a ",(0,a.kt)("em",{parentName:"p"},"Service type")," is mapped to an ",(0,a.kt)("em",{parentName:"p"},"Implementation type"),":")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register();\n")))),(0,a.kt)(i.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"The ",(0,a.kt)("em",{parentName:"p"},"Service type")," used for requesting a service from the container:")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"container.Resolve(); // returns Implementation\n")))),(0,a.kt)("h2",{id:"service-registration--registered-service"},"Service registration | Registered service"),(0,a.kt)("p",null,"It's an entity created by Stashbox when a service is registered. The service registration stores required information about how to instantiate the service, e.g., reflected type information, name, lifetime, conditions, and more."),(0,a.kt)(i.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"In this example, we are registering a named service. The container will create a service registration entity to store the type mapping and the name. During resolution, the container will find the registration by checking for the ",(0,a.kt)("em",{parentName:"p"},"Service type")," and the ",(0,a.kt)("em",{parentName:"p"},"name"),".")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},'// the registration entity will look like: \n// IService => Implementation, name: Example\ncontainer.Register("Example");\nvar service = container.Resolve("Example");\n')))),(0,a.kt)("h2",{id:"injectable-dependency"},"Injectable dependency"),(0,a.kt)(i.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"It's a constructor/method argument or a property/field of a registered ",(0,a.kt)("em",{parentName:"p"},"Implementation type")," that gets evaluated (",(0,a.kt)("em",{parentName:"p"},"injected"),") by Stashbox during the service's construction."),(0,a.kt)("p",null,"In this example, ",(0,a.kt)("inlineCode",{parentName:"p"},"Implementation")," has an ",(0,a.kt)("inlineCode",{parentName:"p"},"IDependency")," ",(0,a.kt)("em",{parentName:"p"},"injectable dependency")," in its constructor.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"class Implementation : IService\n{\n public Implementation(IDependency dependency) \n { }\n}\n")))),(0,a.kt)("h2",{id:"resolution-tree"},"Resolution tree"),(0,a.kt)("p",null,"It's the structural representation of a service's resolution process. It describes the instantiation order of the dependencies required to resolve the desired type."),(0,a.kt)("p",null,"Let's see through an example:"),(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"class A\n{\n public A(B b, C c) { }\n}\n\nclass B\n{\n public B(C c, D d) { }\n}\n\nclass C { }\nclass D { }\n")),(0,a.kt)("p",null,"When we request the service ",(0,a.kt)("inlineCode",{parentName:"p"},"A"),", the container constructs the following resolution tree based on the dependencies and sub-dependencies."),(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre"}," A\n / \\\n B C\n / \\\n C D\n")),(0,a.kt)("p",null,"The container instantiates those services first that don't have any dependencies. ",(0,a.kt)("inlineCode",{parentName:"p"},"C")," and ",(0,a.kt)("inlineCode",{parentName:"p"},"D")," will be injected into ",(0,a.kt)("inlineCode",{parentName:"p"},"B"),". Then, a new ",(0,a.kt)("inlineCode",{parentName:"p"},"C")," is instantiated (if it's ",(0,a.kt)("a",{parentName:"p",href:"/docs/guides/lifetimes#transient-lifetime"},"transient"),") and injected into ",(0,a.kt)("inlineCode",{parentName:"p"},"A")," along with the previously created ",(0,a.kt)("inlineCode",{parentName:"p"},"B"),"."),(0,a.kt)("h2",{id:"dependency-resolver"},"Dependency resolver"),(0,a.kt)("p",null,"It's the container itself or the ",(0,a.kt)("a",{parentName:"p",href:"/docs/guides/scopes"},"current scope"),", depending on which was asked to resolve a particular service. They are both implementing Stashbox's ",(0,a.kt)("inlineCode",{parentName:"p"},"IDependencyResolver")," and the .NET framework's ",(0,a.kt)("inlineCode",{parentName:"p"},"IServiceProvider")," interface and can be used for service resolution."),(0,a.kt)("admonition",{type:"info"},(0,a.kt)("p",{parentName:"admonition"},"Stashbox implicitly injects the ",(0,a.kt)("a",{parentName:"p",href:"/docs/guides/scopes"},"current scope")," wherever ",(0,a.kt)("inlineCode",{parentName:"p"},"IDependencyResolver")," or ",(0,a.kt)("inlineCode",{parentName:"p"},"IServiceProvider")," is requested.")),(0,a.kt)("h2",{id:"root-scope"},"Root scope"),(0,a.kt)("p",null,"It's the ",(0,a.kt)("a",{parentName:"p",href:"/docs/guides/scopes"},"main scope")," created inside every container instance. It stores and handles the lifetime of all singletons. It's the base of each subsequent scope created by the container with the ",(0,a.kt)("inlineCode",{parentName:"p"},".BeginScope()")," method."),(0,a.kt)("admonition",{type:"caution"},(0,a.kt)("p",{parentName:"admonition"},(0,a.kt)("a",{parentName:"p",href:"/docs/guides/lifetimes#scoped-lifetime"},"Scoped services")," requested from the container (and not from a ",(0,a.kt)("a",{parentName:"p",href:"/docs/guides/scopes"},"scope"),") are managed by the root scope. This can lead to issues because their lifetime will effectively switch to singleton. Always be sure that you don't resolve scoped services directly from the container, only from a ",(0,a.kt)("a",{parentName:"p",href:"/docs/guides/scopes"},"scope"),". This case is monitored by the ",(0,a.kt)("a",{parentName:"p",href:"/docs/diagnostics/validation#lifetime-validation"},"lifetime")," validation rule when it's ",(0,a.kt)("a",{parentName:"p",href:"/docs/configuration/container-configuration#lifetime-validation"},"enabled"),". ")),(0,a.kt)("h2",{id:"named-resolution"},"Named resolution"),(0,a.kt)(i.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"It's a resolution request for a named service. The same applies, when the container sees a dependency in the resolution tree with a name (set by ",(0,a.kt)("a",{parentName:"p",href:"/docs/guides/service-resolution#attributes"},"attributes")," or ",(0,a.kt)("a",{parentName:"p",href:"/docs/guides/service-resolution#dependency-binding"},"bindings"),"); it will search for a matching ",(0,a.kt)("a",{parentName:"p",href:"/docs/guides/basics#named-registration"},"Named registration")," to inject.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register("Example");\n// the named request.\nvar service = container.Resolve("Example");\n')))),(0,a.kt)("h2",{id:"self-registration"},"Self registration"),(0,a.kt)(i.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"It's a service registration that's mapped to itself. This means its service and implementation type is the same.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"// equivalent to container.Register();\ncontainer.Register();\n")))))}m.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/fd379919.bc47ef9d.js b/assets/js/fd379919.bc47ef9d.js deleted file mode 100644 index 29bc99fe..00000000 --- a/assets/js/fd379919.bc47ef9d.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkwebsite=self.webpackChunkwebsite||[]).push([[995],{3905:(e,t,n)=>{n.d(t,{Zo:()=>p,kt:()=>b});var a=n(7294);function r(e,t,n){return t in e?Object.defineProperty(e,t,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[t]=n,e}function o(e,t){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);t&&(a=a.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),n.push.apply(n,a)}return n}function i(e){for(var t=1;t=0||(r[n]=e[n]);return r}(e,t);if(Object.getOwnPropertySymbols){var o=Object.getOwnPropertySymbols(e);for(a=0;a=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(r[n]=e[n])}return r}var l=a.createContext({}),c=function(e){var t=a.useContext(l),n=t;return e&&(n="function"==typeof e?e(t):i(i({},t),e)),n},p=function(e){var t=c(e.components);return a.createElement(l.Provider,{value:t},e.children)},u="mdxType",d={inlineCode:"code",wrapper:function(e){var t=e.children;return a.createElement(a.Fragment,{},t)}},m=a.forwardRef((function(e,t){var n=e.components,r=e.mdxType,o=e.originalType,l=e.parentName,p=s(e,["components","mdxType","originalType","parentName"]),u=c(n),m=r,b=u["".concat(l,".").concat(m)]||u[m]||d[m]||o;return n?a.createElement(b,i(i({ref:t},p),{},{components:n})):a.createElement(b,i({ref:t},p))}));function b(e,t){var n=arguments,r=t&&t.mdxType;if("string"==typeof e||r){var o=n.length,i=new Array(o);i[0]=m;var s={};for(var l in t)hasOwnProperty.call(t,l)&&(s[l]=t[l]);s.originalType=e,s[u]="string"==typeof e?e:r,i[1]=s;for(var c=2;c{n.d(t,{Z:()=>i});var a=n(7294),r=n(6010);const o="tabItem_Ymn6";function i(e){let{children:t,hidden:n,className:i}=e;return a.createElement("div",{role:"tabpanel",className:(0,r.Z)(o,i),hidden:n},t)}},4866:(e,t,n)=>{n.d(t,{Z:()=>N});var a=n(7462),r=n(7294),o=n(6010),i=n(2466),s=n(6550),l=n(1980),c=n(7392),p=n(12);function u(e){return function(e){return r.Children.map(e,(e=>{if(!e||(0,r.isValidElement)(e)&&function(e){const{props:t}=e;return!!t&&"object"==typeof t&&"value"in t}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}(e).map((e=>{let{props:{value:t,label:n,attributes:a,default:r}}=e;return{value:t,label:n,attributes:a,default:r}}))}function d(e){const{values:t,children:n}=e;return(0,r.useMemo)((()=>{const e=t??u(n);return function(e){const t=(0,c.l)(e,((e,t)=>e.value===t.value));if(t.length>0)throw new Error(`Docusaurus error: Duplicate values "${t.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[t,n])}function m(e){let{value:t,tabValues:n}=e;return n.some((e=>e.value===t))}function b(e){let{queryString:t=!1,groupId:n}=e;const a=(0,s.k6)(),o=function(e){let{queryString:t=!1,groupId:n}=e;if("string"==typeof t)return t;if(!1===t)return null;if(!0===t&&!n)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return n??null}({queryString:t,groupId:n});return[(0,l._X)(o),(0,r.useCallback)((e=>{if(!o)return;const t=new URLSearchParams(a.location.search);t.set(o,e),a.replace({...a.location,search:t.toString()})}),[o,a])]}function g(e){const{defaultValue:t,queryString:n=!1,groupId:a}=e,o=d(e),[i,s]=(0,r.useState)((()=>function(e){let{defaultValue:t,tabValues:n}=e;if(0===n.length)throw new Error("Docusaurus error: the component requires at least one children component");if(t){if(!m({value:t,tabValues:n}))throw new Error(`Docusaurus error: The has a defaultValue "${t}" but none of its children has the corresponding value. Available values are: ${n.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return t}const a=n.find((e=>e.default))??n[0];if(!a)throw new Error("Unexpected error: 0 tabValues");return a.value}({defaultValue:t,tabValues:o}))),[l,c]=b({queryString:n,groupId:a}),[u,g]=function(e){let{groupId:t}=e;const n=function(e){return e?`docusaurus.tab.${e}`:null}(t),[a,o]=(0,p.Nk)(n);return[a,(0,r.useCallback)((e=>{n&&o.set(e)}),[n,o])]}({groupId:a}),k=(()=>{const e=l??u;return m({value:e,tabValues:o})?e:null})();(0,r.useLayoutEffect)((()=>{k&&s(k)}),[k]);return{selectedValue:i,selectValue:(0,r.useCallback)((e=>{if(!m({value:e,tabValues:o}))throw new Error(`Can't select invalid tab value=${e}`);s(e),c(e),g(e)}),[c,g,o]),tabValues:o}}var k=n(2389);const h="tabList__CuJ",f="tabItem_LNqP";function v(e){let{className:t,block:n,selectedValue:s,selectValue:l,tabValues:c}=e;const p=[],{blockElementScrollPositionUntilNextRender:u}=(0,i.o5)(),d=e=>{const t=e.currentTarget,n=p.indexOf(t),a=c[n].value;a!==s&&(u(t),l(a))},m=e=>{let t=null;switch(e.key){case"Enter":d(e);break;case"ArrowRight":{const n=p.indexOf(e.currentTarget)+1;t=p[n]??p[0];break}case"ArrowLeft":{const n=p.indexOf(e.currentTarget)-1;t=p[n]??p[p.length-1];break}}t?.focus()};return r.createElement("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,o.Z)("tabs",{"tabs--block":n},t)},c.map((e=>{let{value:t,label:n,attributes:i}=e;return r.createElement("li",(0,a.Z)({role:"tab",tabIndex:s===t?0:-1,"aria-selected":s===t,key:t,ref:e=>p.push(e),onKeyDown:m,onClick:d},i,{className:(0,o.Z)("tabs__item",f,i?.className,{"tabs__item--active":s===t})}),n??t)})))}function y(e){let{lazy:t,children:n,selectedValue:a}=e;const o=(Array.isArray(n)?n:[n]).filter(Boolean);if(t){const e=o.find((e=>e.props.value===a));return e?(0,r.cloneElement)(e,{className:"margin-top--md"}):null}return r.createElement("div",{className:"margin-top--md"},o.map(((e,t)=>(0,r.cloneElement)(e,{key:t,hidden:e.props.value!==a}))))}function I(e){const t=g(e);return r.createElement("div",{className:(0,o.Z)("tabs-container",h)},r.createElement(v,(0,a.Z)({},e,t)),r.createElement(y,(0,a.Z)({},e,t)))}function N(e){const t=(0,k.Z)();return r.createElement(I,(0,a.Z)({key:String(t)},e))}},8846:(e,t,n)=>{n.d(t,{Z:()=>s});var a=n(7294);const r="codeDescContainer_ie8f",o="desc_jyqI",i="example_eYlF";function s(e){let{children:t}=e,n=a.Children.toArray(t).filter((e=>e));return a.createElement("div",{className:r},a.createElement("div",{className:o},n[0]),a.createElement("div",{className:i},n[1]))}},9144:(e,t,n)=>{n.r(t),n.d(t,{assets:()=>u,contentTitle:()=>c,default:()=>b,frontMatter:()=>l,metadata:()=>p,toc:()=>d});var a=n(7462),r=(n(7294),n(3905)),o=n(8846),i=n(4866),s=n(5162);const l={},c="Basic usage",p={unversionedId:"guides/basics",id:"guides/basics",title:"Basic usage",description:"This section is about the basics of Stashbox's API. It will give you a good starting point for more advanced topics described in the following sections.",source:"@site/docs/guides/basics.md",sourceDirName:"guides",slug:"/guides/basics",permalink:"/stashbox/docs/guides/basics",draft:!1,editUrl:"https://github.com/z4kn4fein/stashbox/edit/master/docs/docs/guides/basics.md",tags:[],version:"current",lastUpdatedBy:"dependabot[bot]",lastUpdatedAt:1697644420,formattedLastUpdatedAt:"Oct 18, 2023",frontMatter:{},sidebar:"docs",previous:{title:"Glossary",permalink:"/stashbox/docs/getting-started/glossary"},next:{title:"Advanced registration",permalink:"/stashbox/docs/guides/advanced-registration"}},u={},d=[{value:"Default registration",id:"default-registration",level:2},{value:"Named registration",id:"named-registration",level:2},{value:"Instance registration",id:"instance-registration",level:2},{value:"Re-mapping",id:"re-mapping",level:2},{value:"Wiring up",id:"wiring-up",level:2},{value:"Lifetime shortcuts",id:"lifetime-shortcuts",level:2}],m={toc:d};function b(e){let{components:t,...n}=e;return(0,r.kt)("wrapper",(0,a.Z)({},m,n,{components:t,mdxType:"MDXLayout"}),(0,r.kt)("h1",{id:"basic-usage"},"Basic usage"),(0,r.kt)("p",null,"This section is about the basics of Stashbox's API. It will give you a good starting point for more advanced topics described in the following sections.\nStashbox provides several methods that enable registering services, and we'll go through the most common scenarios with code examples."),(0,r.kt)("h2",{id:"default-registration"},"Default registration"),(0,r.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("p",null,"Stashbox allows registration operations via the ",(0,r.kt)("inlineCode",{parentName:"p"},"Register()")," methods. "),(0,r.kt)("p",null,"During registration, the container checks whether the ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type")," is assignable from the ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"implementation type")," and if not, the container throws an ",(0,r.kt)("a",{parentName:"p",href:"/docs/diagnostics/validation#registration-validation"},"exception"),". "),(0,r.kt)("p",null,"Also, when the implementation is not resolvable, the container throws the same ",(0,r.kt)("a",{parentName:"p",href:"/docs/diagnostics/validation#registration-validation"},"exception"),"."),(0,r.kt)("p",null,"The example registers ",(0,r.kt)("inlineCode",{parentName:"p"},"DbBackup")," to be returned when ",(0,r.kt)("inlineCode",{parentName:"p"},"IJob")," is requested.")),(0,r.kt)("div",null,(0,r.kt)(i.Z,{groupId:"generic-runtime-apis",mdxType:"Tabs"},(0,r.kt)(s.Z,{value:"Generic API",label:"Generic API",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register();\nIJob job = container.Resolve();\n// throws an exception because ConsoleLogger doesn't implement IJob.\ncontainer.Register();\n// throws an exception because IJob is not a valid implementation.\ncontainer.Register();\n"))),(0,r.kt)(s.Z,{value:"Runtime type API",label:"Runtime type API",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(typeof(IJob), typeof(DbBackup));\nobject job = container.Resolve(typeof(IJob));\n// throws an exception because ConsoleLogger doesn't implement IJob.\ncontainer.Register(typeof(IJob), typeof(ConsoleLogger));\n// throws an exception because IJob is not a valid implementation.\ncontainer.Register(typeof(IJob), typeof(IJob));\n")))))),(0,r.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("p",null,"You can register a service to itself without specifying a ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type"),", only the implementation (",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#self-registration"},"self registration"),"). "),(0,r.kt)("p",null,"In this case, the given implementation is considered the ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type")," and must be used to request the service (",(0,r.kt)("inlineCode",{parentName:"p"},"DbBackup")," in the example).")),(0,r.kt)("div",null,(0,r.kt)(i.Z,{groupId:"generic-runtime-apis",mdxType:"Tabs"},(0,r.kt)(s.Z,{value:"Generic API",label:"Generic API",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register();\nDbBackup backup = container.Resolve();\n"))),(0,r.kt)(s.Z,{value:"Runtime type API",label:"Runtime type API",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(typeof(DbBackup));\nobject backup = container.Resolve(typeof(DbBackup));\n")))))),(0,r.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("p",null,"The container's API is fluent, which means you can chain the calls on its methods after each other.")),(0,r.kt)("div",null,(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"var job = container.Register()\n .Register()\n .Resolve();\n")))),(0,r.kt)("h2",{id:"named-registration"},"Named registration"),(0,r.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("p",null,"The example shows how you can bind more implementations to a ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type")," using names for identification. "),(0,r.kt)("p",null,"The same name must be used to resolve the named service."),(0,r.kt)("admonition",{type:"note"},(0,r.kt)("p",{parentName:"admonition"},"The name is an ",(0,r.kt)("inlineCode",{parentName:"p"},"object")," type."))),(0,r.kt)("div",null,(0,r.kt)(i.Z,{groupId:"generic-runtime-apis",mdxType:"Tabs"},(0,r.kt)(s.Z,{value:"Generic API",label:"Generic API",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register("DbBackup");\ncontainer.Register("StorageCleanup");\nIJob cleanup = container.Resolve("StorageCleanup");\n'))),(0,r.kt)(s.Z,{value:"Runtime type API",label:"Runtime type API",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register(typeof(IJob), typeof(DbBackup), "DbBackup");\ncontainer.Register(typeof(IJob), typeof(StorageCleanup), "StorageCleanup");\nobject cleanup = container.Resolve(typeof(IJob), "StorageCleanup");\n')))))),(0,r.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("p",null,"You can also get each service that share the same name by requesting an ",(0,r.kt)("inlineCode",{parentName:"p"},"IEnumerable<>")," or using the ",(0,r.kt)("inlineCode",{parentName:"p"},"ResolveAll()")," method with the ",(0,r.kt)("inlineCode",{parentName:"p"},"name")," parameter.")),(0,r.kt)("div",null,(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register("StorageJobs");\ncontainer.Register("StorageJobs");\ncontainer.Register();\n// jobs will be [DbBackup, StorageCleanup].\nIEnumerable jobs = container.Resolve>("StorageJobs");\n')))),(0,r.kt)("h2",{id:"instance-registration"},"Instance registration"),(0,r.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("p",null,"With instance registration, you can provide an already created external instance to use when the given ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type")," is requested."),(0,r.kt)("p",null,"Stashbox automatically handles the ",(0,r.kt)("a",{parentName:"p",href:"/docs/guides/scopes#disposal"},"disposal")," of the registered instances, but you can turn this feature off with the ",(0,r.kt)("inlineCode",{parentName:"p"},"withoutDisposalTracking")," parameter."),(0,r.kt)("p",null,"When an ",(0,r.kt)("inlineCode",{parentName:"p"},"IJob")," is requested, the container will always return the external instance.")),(0,r.kt)("div",null,(0,r.kt)(i.Z,{groupId:"generic-runtime-apis",mdxType:"Tabs"},(0,r.kt)(s.Z,{value:"Generic API",label:"Generic API",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"var job = new DbBackup();\ncontainer.RegisterInstance(job);\n\n// resolvedJob and job are the same.\nIJob resolvedJob = container.Resolve();\n"))),(0,r.kt)(s.Z,{value:"Runtime type API",label:"Runtime type API",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"var job = new DbBackup();\ncontainer.RegisterInstance(job, typeof(IJob));\n\n// resolvedJob and job are the same.\nobject resolvedJob = container.Resolve(typeof(IJob));\n"))),(0,r.kt)(s.Z,{value:"Named",label:"Named",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},'var job = new DbBackup();\ncontainer.RegisterInstance(job, "DbBackup");\n\n// resolvedJob and job are the same.\nIJob resolvedJob = container.Resolve("DbBackup");\n'))),(0,r.kt)(s.Z,{value:"No dispose",label:"No dispose",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"var job = new DbBackup();\ncontainer.RegisterInstance(job, withoutDisposalTracking: true);\n\n// resolvedJob and job are the same.\nIJob resolvedJob = container.Resolve();\n")))))),(0,r.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("p",null,"The instance registration API allows the batched registration of different instances.")),(0,r.kt)("div",null,(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterInstances(new DbBackup(), new StorageCleanup());\nIEnumerable jobs = container.ResolveAll();\n")))),(0,r.kt)("h2",{id:"re-mapping"},"Re-mapping"),(0,r.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("p",null,"With re-map, you can bind new implementations to a ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type")," and delete old registrations in one action. "),(0,r.kt)("admonition",{type:"caution"},(0,r.kt)("p",{parentName:"admonition"},"When there are multiple registrations mapped to a ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type"),", ",(0,r.kt)("inlineCode",{parentName:"p"},".ReMap()")," will replace all of them with the given ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"implementation type"),". If you want to replace only one specific service, use the ",(0,r.kt)("inlineCode",{parentName:"p"},".ReplaceExisting()")," ",(0,r.kt)("a",{parentName:"p",href:"/docs/configuration/registration-configuration#replace"},"configuration option"),"."))),(0,r.kt)("div",null,(0,r.kt)(i.Z,{groupId:"generic-runtime-apis",mdxType:"Tabs"},(0,r.kt)(s.Z,{value:"Generic API",label:"Generic API",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register();\ncontainer.ReMap();\n// jobs contain all two jobs\nIEnumerable jobs = container.ResolveAll();\n\ncontainer.ReMap();\n// jobs contains only the SlackMessageSender\njobs = container.ResolveAll();\n"))),(0,r.kt)(s.Z,{value:"Runtime type API",label:"Runtime type API",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(typeof(IJob), typeof(DbBackup));\ncontainer.Register(typeof(IJob), typeof(StorageCleanup));\n// jobs contain all two jobs\nIEnumerable jobs = container.ResolveAll(typeof(IJob));\n\ncontainer.ReMap(typeof(IJob), typeof(SlackMessageSender));\n// jobs contains only the SlackMessageSender\njobs = container.ResolveAll(typeof(IJob));\n")))))),(0,r.kt)("h2",{id:"wiring-up"},"Wiring up"),(0,r.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("p",null,"Wiring up is similar to ",(0,r.kt)("a",{parentName:"p",href:"#instance-registration"},"Instance registration")," except that the container will perform property/field injection (if configured so and applicable) on the registered instance during resolution.")),(0,r.kt)("div",null,(0,r.kt)(i.Z,{groupId:"generic-runtime-apis",mdxType:"Tabs"},(0,r.kt)(s.Z,{value:"Generic API",label:"Generic API",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.WireUp(new DbBackup());\nIJob job = container.Resolve();\n"))),(0,r.kt)(s.Z,{value:"Runtime type API",label:"Runtime type API",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.WireUp(new DbBackup(), typeof(IJob));\nobject job = container.Resolve(typeof(IJob));\n")))))),(0,r.kt)("h2",{id:"lifetime-shortcuts"},"Lifetime shortcuts"),(0,r.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,"A service's lifetime indicates how long its instance will live and which re-using policy should be applied when it gets injected.",(0,r.kt)("p",null,"This example shows how you can use the registration API's shortcuts for lifetimes. These are just sugars, and there are more ways explained in the ",(0,r.kt)("a",{parentName:"p",href:"/docs/guides/lifetimes"},"lifetimes")," section."),(0,r.kt)("admonition",{type:"info"},(0,r.kt)("p",{parentName:"admonition"},"The ",(0,r.kt)("inlineCode",{parentName:"p"},"DefaultLifetime")," is ",(0,r.kt)("a",{parentName:"p",href:"/docs/guides/lifetimes#default-lifetime"},"configurable"),"."))),(0,r.kt)("div",null,(0,r.kt)(i.Z,{groupId:"generic-runtime-apis",mdxType:"Tabs"},(0,r.kt)(s.Z,{value:"Default",label:"Default",mdxType:"TabItem"},(0,r.kt)("p",null,"When no lifetime is specified, the service will use the container's ",(0,r.kt)("inlineCode",{parentName:"p"},"DefaultLifetime"),", which is ",(0,r.kt)("inlineCode",{parentName:"p"},"Transient")," by default."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register();\nIJob job = container.Resolve();\n"))),(0,r.kt)(s.Z,{value:"Singleton",label:"Singleton",mdxType:"TabItem"},(0,r.kt)("p",null,"A service with ",(0,r.kt)("inlineCode",{parentName:"p"},"Singleton")," lifetime will be instantiated once and reused during the container's lifetime."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterSingleton();\nIJob job = container.Resolve();\n"))),(0,r.kt)(s.Z,{value:"Scoped",label:"Scoped",mdxType:"TabItem"},(0,r.kt)("p",null,"The ",(0,r.kt)("inlineCode",{parentName:"p"},"Scoped")," lifetime behaves like a ",(0,r.kt)("inlineCode",{parentName:"p"},"Singleton")," within a ",(0,r.kt)("a",{parentName:"p",href:"/docs/guides/scopes"},"scope"),".\nA scoped service is instantiated once and reused during the scope's whole lifetime."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterScoped();\nIJob job = container.Resolve();\n")))))))}b.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/fd379919.bd8013b7.js b/assets/js/fd379919.bd8013b7.js new file mode 100644 index 00000000..84c5ed2c --- /dev/null +++ b/assets/js/fd379919.bd8013b7.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkwebsite=self.webpackChunkwebsite||[]).push([[995],{3905:(e,t,n)=>{n.d(t,{Zo:()=>p,kt:()=>b});var a=n(7294);function r(e,t,n){return t in e?Object.defineProperty(e,t,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[t]=n,e}function o(e,t){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);t&&(a=a.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),n.push.apply(n,a)}return n}function i(e){for(var t=1;t=0||(r[n]=e[n]);return r}(e,t);if(Object.getOwnPropertySymbols){var o=Object.getOwnPropertySymbols(e);for(a=0;a=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(r[n]=e[n])}return r}var l=a.createContext({}),c=function(e){var t=a.useContext(l),n=t;return e&&(n="function"==typeof e?e(t):i(i({},t),e)),n},p=function(e){var t=c(e.components);return a.createElement(l.Provider,{value:t},e.children)},u="mdxType",d={inlineCode:"code",wrapper:function(e){var t=e.children;return a.createElement(a.Fragment,{},t)}},m=a.forwardRef((function(e,t){var n=e.components,r=e.mdxType,o=e.originalType,l=e.parentName,p=s(e,["components","mdxType","originalType","parentName"]),u=c(n),m=r,b=u["".concat(l,".").concat(m)]||u[m]||d[m]||o;return n?a.createElement(b,i(i({ref:t},p),{},{components:n})):a.createElement(b,i({ref:t},p))}));function b(e,t){var n=arguments,r=t&&t.mdxType;if("string"==typeof e||r){var o=n.length,i=new Array(o);i[0]=m;var s={};for(var l in t)hasOwnProperty.call(t,l)&&(s[l]=t[l]);s.originalType=e,s[u]="string"==typeof e?e:r,i[1]=s;for(var c=2;c{n.d(t,{Z:()=>i});var a=n(7294),r=n(6010);const o="tabItem_Ymn6";function i(e){let{children:t,hidden:n,className:i}=e;return a.createElement("div",{role:"tabpanel",className:(0,r.Z)(o,i),hidden:n},t)}},4866:(e,t,n)=>{n.d(t,{Z:()=>N});var a=n(7462),r=n(7294),o=n(6010),i=n(2466),s=n(6550),l=n(1980),c=n(7392),p=n(12);function u(e){return function(e){return r.Children.map(e,(e=>{if(!e||(0,r.isValidElement)(e)&&function(e){const{props:t}=e;return!!t&&"object"==typeof t&&"value"in t}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}(e).map((e=>{let{props:{value:t,label:n,attributes:a,default:r}}=e;return{value:t,label:n,attributes:a,default:r}}))}function d(e){const{values:t,children:n}=e;return(0,r.useMemo)((()=>{const e=t??u(n);return function(e){const t=(0,c.l)(e,((e,t)=>e.value===t.value));if(t.length>0)throw new Error(`Docusaurus error: Duplicate values "${t.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[t,n])}function m(e){let{value:t,tabValues:n}=e;return n.some((e=>e.value===t))}function b(e){let{queryString:t=!1,groupId:n}=e;const a=(0,s.k6)(),o=function(e){let{queryString:t=!1,groupId:n}=e;if("string"==typeof t)return t;if(!1===t)return null;if(!0===t&&!n)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return n??null}({queryString:t,groupId:n});return[(0,l._X)(o),(0,r.useCallback)((e=>{if(!o)return;const t=new URLSearchParams(a.location.search);t.set(o,e),a.replace({...a.location,search:t.toString()})}),[o,a])]}function g(e){const{defaultValue:t,queryString:n=!1,groupId:a}=e,o=d(e),[i,s]=(0,r.useState)((()=>function(e){let{defaultValue:t,tabValues:n}=e;if(0===n.length)throw new Error("Docusaurus error: the component requires at least one children component");if(t){if(!m({value:t,tabValues:n}))throw new Error(`Docusaurus error: The has a defaultValue "${t}" but none of its children has the corresponding value. Available values are: ${n.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return t}const a=n.find((e=>e.default))??n[0];if(!a)throw new Error("Unexpected error: 0 tabValues");return a.value}({defaultValue:t,tabValues:o}))),[l,c]=b({queryString:n,groupId:a}),[u,g]=function(e){let{groupId:t}=e;const n=function(e){return e?`docusaurus.tab.${e}`:null}(t),[a,o]=(0,p.Nk)(n);return[a,(0,r.useCallback)((e=>{n&&o.set(e)}),[n,o])]}({groupId:a}),k=(()=>{const e=l??u;return m({value:e,tabValues:o})?e:null})();(0,r.useLayoutEffect)((()=>{k&&s(k)}),[k]);return{selectedValue:i,selectValue:(0,r.useCallback)((e=>{if(!m({value:e,tabValues:o}))throw new Error(`Can't select invalid tab value=${e}`);s(e),c(e),g(e)}),[c,g,o]),tabValues:o}}var k=n(2389);const h="tabList__CuJ",f="tabItem_LNqP";function v(e){let{className:t,block:n,selectedValue:s,selectValue:l,tabValues:c}=e;const p=[],{blockElementScrollPositionUntilNextRender:u}=(0,i.o5)(),d=e=>{const t=e.currentTarget,n=p.indexOf(t),a=c[n].value;a!==s&&(u(t),l(a))},m=e=>{let t=null;switch(e.key){case"Enter":d(e);break;case"ArrowRight":{const n=p.indexOf(e.currentTarget)+1;t=p[n]??p[0];break}case"ArrowLeft":{const n=p.indexOf(e.currentTarget)-1;t=p[n]??p[p.length-1];break}}t?.focus()};return r.createElement("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,o.Z)("tabs",{"tabs--block":n},t)},c.map((e=>{let{value:t,label:n,attributes:i}=e;return r.createElement("li",(0,a.Z)({role:"tab",tabIndex:s===t?0:-1,"aria-selected":s===t,key:t,ref:e=>p.push(e),onKeyDown:m,onClick:d},i,{className:(0,o.Z)("tabs__item",f,i?.className,{"tabs__item--active":s===t})}),n??t)})))}function y(e){let{lazy:t,children:n,selectedValue:a}=e;const o=(Array.isArray(n)?n:[n]).filter(Boolean);if(t){const e=o.find((e=>e.props.value===a));return e?(0,r.cloneElement)(e,{className:"margin-top--md"}):null}return r.createElement("div",{className:"margin-top--md"},o.map(((e,t)=>(0,r.cloneElement)(e,{key:t,hidden:e.props.value!==a}))))}function I(e){const t=g(e);return r.createElement("div",{className:(0,o.Z)("tabs-container",h)},r.createElement(v,(0,a.Z)({},e,t)),r.createElement(y,(0,a.Z)({},e,t)))}function N(e){const t=(0,k.Z)();return r.createElement(I,(0,a.Z)({key:String(t)},e))}},8846:(e,t,n)=>{n.d(t,{Z:()=>s});var a=n(7294);const r="codeDescContainer_ie8f",o="desc_jyqI",i="example_eYlF";function s(e){let{children:t}=e,n=a.Children.toArray(t).filter((e=>e));return a.createElement("div",{className:r},a.createElement("div",{className:o},n[0]),a.createElement("div",{className:i},n[1]))}},9144:(e,t,n)=>{n.r(t),n.d(t,{assets:()=>u,contentTitle:()=>c,default:()=>b,frontMatter:()=>l,metadata:()=>p,toc:()=>d});var a=n(7462),r=(n(7294),n(3905)),o=n(8846),i=n(4866),s=n(5162);const l={},c="Basic usage",p={unversionedId:"guides/basics",id:"guides/basics",title:"Basic usage",description:"This section is about the basics of Stashbox's API. It will give you a good starting point for more advanced topics described in the following sections.",source:"@site/docs/guides/basics.md",sourceDirName:"guides",slug:"/guides/basics",permalink:"/stashbox/docs/guides/basics",draft:!1,editUrl:"https://github.com/z4kn4fein/stashbox/edit/master/docs/docs/guides/basics.md",tags:[],version:"current",lastUpdatedBy:"dependabot[bot]",lastUpdatedAt:1699538162,formattedLastUpdatedAt:"Nov 9, 2023",frontMatter:{},sidebar:"docs",previous:{title:"Glossary",permalink:"/stashbox/docs/getting-started/glossary"},next:{title:"Advanced registration",permalink:"/stashbox/docs/guides/advanced-registration"}},u={},d=[{value:"Default registration",id:"default-registration",level:2},{value:"Named registration",id:"named-registration",level:2},{value:"Instance registration",id:"instance-registration",level:2},{value:"Re-mapping",id:"re-mapping",level:2},{value:"Wiring up",id:"wiring-up",level:2},{value:"Lifetime shortcuts",id:"lifetime-shortcuts",level:2}],m={toc:d};function b(e){let{components:t,...n}=e;return(0,r.kt)("wrapper",(0,a.Z)({},m,n,{components:t,mdxType:"MDXLayout"}),(0,r.kt)("h1",{id:"basic-usage"},"Basic usage"),(0,r.kt)("p",null,"This section is about the basics of Stashbox's API. It will give you a good starting point for more advanced topics described in the following sections.\nStashbox provides several methods that enable registering services, and we'll go through the most common scenarios with code examples."),(0,r.kt)("h2",{id:"default-registration"},"Default registration"),(0,r.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("p",null,"Stashbox allows registration operations via the ",(0,r.kt)("inlineCode",{parentName:"p"},"Register()")," methods. "),(0,r.kt)("p",null,"During registration, the container checks whether the ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type")," is assignable from the ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"implementation type")," and if not, the container throws an ",(0,r.kt)("a",{parentName:"p",href:"/docs/diagnostics/validation#registration-validation"},"exception"),". "),(0,r.kt)("p",null,"Also, when the implementation is not resolvable, the container throws the same ",(0,r.kt)("a",{parentName:"p",href:"/docs/diagnostics/validation#registration-validation"},"exception"),"."),(0,r.kt)("p",null,"The example registers ",(0,r.kt)("inlineCode",{parentName:"p"},"DbBackup")," to be returned when ",(0,r.kt)("inlineCode",{parentName:"p"},"IJob")," is requested.")),(0,r.kt)("div",null,(0,r.kt)(i.Z,{groupId:"generic-runtime-apis",mdxType:"Tabs"},(0,r.kt)(s.Z,{value:"Generic API",label:"Generic API",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register();\nIJob job = container.Resolve();\n// throws an exception because ConsoleLogger doesn't implement IJob.\ncontainer.Register();\n// throws an exception because IJob is not a valid implementation.\ncontainer.Register();\n"))),(0,r.kt)(s.Z,{value:"Runtime type API",label:"Runtime type API",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(typeof(IJob), typeof(DbBackup));\nobject job = container.Resolve(typeof(IJob));\n// throws an exception because ConsoleLogger doesn't implement IJob.\ncontainer.Register(typeof(IJob), typeof(ConsoleLogger));\n// throws an exception because IJob is not a valid implementation.\ncontainer.Register(typeof(IJob), typeof(IJob));\n")))))),(0,r.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("p",null,"You can register a service to itself without specifying a ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type"),", only the implementation (",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#self-registration"},"self registration"),"). "),(0,r.kt)("p",null,"In this case, the given implementation is considered the ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type")," and must be used to request the service (",(0,r.kt)("inlineCode",{parentName:"p"},"DbBackup")," in the example).")),(0,r.kt)("div",null,(0,r.kt)(i.Z,{groupId:"generic-runtime-apis",mdxType:"Tabs"},(0,r.kt)(s.Z,{value:"Generic API",label:"Generic API",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register();\nDbBackup backup = container.Resolve();\n"))),(0,r.kt)(s.Z,{value:"Runtime type API",label:"Runtime type API",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(typeof(DbBackup));\nobject backup = container.Resolve(typeof(DbBackup));\n")))))),(0,r.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("p",null,"The container's API is fluent, which means you can chain the calls on its methods after each other.")),(0,r.kt)("div",null,(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"var job = container.Register()\n .Register()\n .Resolve();\n")))),(0,r.kt)("h2",{id:"named-registration"},"Named registration"),(0,r.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("p",null,"The example shows how you can bind more implementations to a ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type")," using names for identification. "),(0,r.kt)("p",null,"The same name must be used to resolve the named service."),(0,r.kt)("admonition",{type:"note"},(0,r.kt)("p",{parentName:"admonition"},"The name is an ",(0,r.kt)("inlineCode",{parentName:"p"},"object")," type."))),(0,r.kt)("div",null,(0,r.kt)(i.Z,{groupId:"generic-runtime-apis",mdxType:"Tabs"},(0,r.kt)(s.Z,{value:"Generic API",label:"Generic API",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register("DbBackup");\ncontainer.Register("StorageCleanup");\nIJob cleanup = container.Resolve("StorageCleanup");\n'))),(0,r.kt)(s.Z,{value:"Runtime type API",label:"Runtime type API",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register(typeof(IJob), typeof(DbBackup), "DbBackup");\ncontainer.Register(typeof(IJob), typeof(StorageCleanup), "StorageCleanup");\nobject cleanup = container.Resolve(typeof(IJob), "StorageCleanup");\n')))))),(0,r.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("p",null,"You can also get each service that share the same name by requesting an ",(0,r.kt)("inlineCode",{parentName:"p"},"IEnumerable<>")," or using the ",(0,r.kt)("inlineCode",{parentName:"p"},"ResolveAll()")," method with the ",(0,r.kt)("inlineCode",{parentName:"p"},"name")," parameter.")),(0,r.kt)("div",null,(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register("StorageJobs");\ncontainer.Register("StorageJobs");\ncontainer.Register();\n// jobs will be [DbBackup, StorageCleanup].\nIEnumerable jobs = container.Resolve>("StorageJobs");\n')))),(0,r.kt)("h2",{id:"instance-registration"},"Instance registration"),(0,r.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("p",null,"With instance registration, you can provide an already created external instance to use when the given ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type")," is requested."),(0,r.kt)("p",null,"Stashbox automatically handles the ",(0,r.kt)("a",{parentName:"p",href:"/docs/guides/scopes#disposal"},"disposal")," of the registered instances, but you can turn this feature off with the ",(0,r.kt)("inlineCode",{parentName:"p"},"withoutDisposalTracking")," parameter."),(0,r.kt)("p",null,"When an ",(0,r.kt)("inlineCode",{parentName:"p"},"IJob")," is requested, the container will always return the external instance.")),(0,r.kt)("div",null,(0,r.kt)(i.Z,{groupId:"generic-runtime-apis",mdxType:"Tabs"},(0,r.kt)(s.Z,{value:"Generic API",label:"Generic API",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"var job = new DbBackup();\ncontainer.RegisterInstance(job);\n\n// resolvedJob and job are the same.\nIJob resolvedJob = container.Resolve();\n"))),(0,r.kt)(s.Z,{value:"Runtime type API",label:"Runtime type API",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"var job = new DbBackup();\ncontainer.RegisterInstance(job, typeof(IJob));\n\n// resolvedJob and job are the same.\nobject resolvedJob = container.Resolve(typeof(IJob));\n"))),(0,r.kt)(s.Z,{value:"Named",label:"Named",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},'var job = new DbBackup();\ncontainer.RegisterInstance(job, "DbBackup");\n\n// resolvedJob and job are the same.\nIJob resolvedJob = container.Resolve("DbBackup");\n'))),(0,r.kt)(s.Z,{value:"No dispose",label:"No dispose",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"var job = new DbBackup();\ncontainer.RegisterInstance(job, withoutDisposalTracking: true);\n\n// resolvedJob and job are the same.\nIJob resolvedJob = container.Resolve();\n")))))),(0,r.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("p",null,"The instance registration API allows the batched registration of different instances.")),(0,r.kt)("div",null,(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterInstances(new DbBackup(), new StorageCleanup());\nIEnumerable jobs = container.ResolveAll();\n")))),(0,r.kt)("h2",{id:"re-mapping"},"Re-mapping"),(0,r.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("p",null,"With re-map, you can bind new implementations to a ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type")," and delete old registrations in one action. "),(0,r.kt)("admonition",{type:"caution"},(0,r.kt)("p",{parentName:"admonition"},"When there are multiple registrations mapped to a ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"service type"),", ",(0,r.kt)("inlineCode",{parentName:"p"},".ReMap()")," will replace all of them with the given ",(0,r.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#service-type--implementation-type"},"implementation type"),". If you want to replace only one specific service, use the ",(0,r.kt)("inlineCode",{parentName:"p"},".ReplaceExisting()")," ",(0,r.kt)("a",{parentName:"p",href:"/docs/configuration/registration-configuration#replace"},"configuration option"),"."))),(0,r.kt)("div",null,(0,r.kt)(i.Z,{groupId:"generic-runtime-apis",mdxType:"Tabs"},(0,r.kt)(s.Z,{value:"Generic API",label:"Generic API",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register();\ncontainer.ReMap();\n// jobs contain all two jobs\nIEnumerable jobs = container.ResolveAll();\n\ncontainer.ReMap();\n// jobs contains only the SlackMessageSender\njobs = container.ResolveAll();\n"))),(0,r.kt)(s.Z,{value:"Runtime type API",label:"Runtime type API",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(typeof(IJob), typeof(DbBackup));\ncontainer.Register(typeof(IJob), typeof(StorageCleanup));\n// jobs contain all two jobs\nIEnumerable jobs = container.ResolveAll(typeof(IJob));\n\ncontainer.ReMap(typeof(IJob), typeof(SlackMessageSender));\n// jobs contains only the SlackMessageSender\njobs = container.ResolveAll(typeof(IJob));\n")))))),(0,r.kt)("h2",{id:"wiring-up"},"Wiring up"),(0,r.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,(0,r.kt)("p",null,"Wiring up is similar to ",(0,r.kt)("a",{parentName:"p",href:"#instance-registration"},"Instance registration")," except that the container will perform property/field injection (if configured so and applicable) on the registered instance during resolution.")),(0,r.kt)("div",null,(0,r.kt)(i.Z,{groupId:"generic-runtime-apis",mdxType:"Tabs"},(0,r.kt)(s.Z,{value:"Generic API",label:"Generic API",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.WireUp(new DbBackup());\nIJob job = container.Resolve();\n"))),(0,r.kt)(s.Z,{value:"Runtime type API",label:"Runtime type API",mdxType:"TabItem"},(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.WireUp(new DbBackup(), typeof(IJob));\nobject job = container.Resolve(typeof(IJob));\n")))))),(0,r.kt)("h2",{id:"lifetime-shortcuts"},"Lifetime shortcuts"),(0,r.kt)(o.Z,{mdxType:"CodeDescPanel"},(0,r.kt)("div",null,"A service's lifetime indicates how long its instance will live and which re-using policy should be applied when it gets injected.",(0,r.kt)("p",null,"This example shows how you can use the registration API's shortcuts for lifetimes. These are just sugars, and there are more ways explained in the ",(0,r.kt)("a",{parentName:"p",href:"/docs/guides/lifetimes"},"lifetimes")," section."),(0,r.kt)("admonition",{type:"info"},(0,r.kt)("p",{parentName:"admonition"},"The ",(0,r.kt)("inlineCode",{parentName:"p"},"DefaultLifetime")," is ",(0,r.kt)("a",{parentName:"p",href:"/docs/guides/lifetimes#default-lifetime"},"configurable"),"."))),(0,r.kt)("div",null,(0,r.kt)(i.Z,{groupId:"generic-runtime-apis",mdxType:"Tabs"},(0,r.kt)(s.Z,{value:"Default",label:"Default",mdxType:"TabItem"},(0,r.kt)("p",null,"When no lifetime is specified, the service will use the container's ",(0,r.kt)("inlineCode",{parentName:"p"},"DefaultLifetime"),", which is ",(0,r.kt)("inlineCode",{parentName:"p"},"Transient")," by default."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register();\nIJob job = container.Resolve();\n"))),(0,r.kt)(s.Z,{value:"Singleton",label:"Singleton",mdxType:"TabItem"},(0,r.kt)("p",null,"A service with ",(0,r.kt)("inlineCode",{parentName:"p"},"Singleton")," lifetime will be instantiated once and reused during the container's lifetime."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterSingleton();\nIJob job = container.Resolve();\n"))),(0,r.kt)(s.Z,{value:"Scoped",label:"Scoped",mdxType:"TabItem"},(0,r.kt)("p",null,"The ",(0,r.kt)("inlineCode",{parentName:"p"},"Scoped")," lifetime behaves like a ",(0,r.kt)("inlineCode",{parentName:"p"},"Singleton")," within a ",(0,r.kt)("a",{parentName:"p",href:"/docs/guides/scopes"},"scope"),".\nA scoped service is instantiated once and reused during the scope's whole lifetime."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterScoped();\nIJob job = container.Resolve();\n")))))))}b.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/ff10094c.4eb59afd.js b/assets/js/ff10094c.4eb59afd.js deleted file mode 100644 index 4fb4f746..00000000 --- a/assets/js/ff10094c.4eb59afd.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkwebsite=self.webpackChunkwebsite||[]).push([[544],{3905:(e,t,n)=>{n.d(t,{Zo:()=>u,kt:()=>f});var i=n(7294);function a(e,t,n){return t in e?Object.defineProperty(e,t,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[t]=n,e}function r(e,t){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(e);t&&(i=i.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),n.push.apply(n,i)}return n}function o(e){for(var t=1;t=0||(a[n]=e[n]);return a}(e,t);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);for(i=0;i=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(a[n]=e[n])}return a}var l=i.createContext({}),c=function(e){var t=i.useContext(l),n=t;return e&&(n="function"==typeof e?e(t):o(o({},t),e)),n},u=function(e){var t=c(e.components);return i.createElement(l.Provider,{value:t},e.children)},p="mdxType",d={inlineCode:"code",wrapper:function(e){var t=e.children;return i.createElement(i.Fragment,{},t)}},m=i.forwardRef((function(e,t){var n=e.components,a=e.mdxType,r=e.originalType,l=e.parentName,u=s(e,["components","mdxType","originalType","parentName"]),p=c(n),m=a,f=p["".concat(l,".").concat(m)]||p[m]||d[m]||r;return n?i.createElement(f,o(o({ref:t},u),{},{components:n})):i.createElement(f,o({ref:t},u))}));function f(e,t){var n=arguments,a=t&&t.mdxType;if("string"==typeof e||a){var r=n.length,o=new Array(r);o[0]=m;var s={};for(var l in t)hasOwnProperty.call(t,l)&&(s[l]=t[l]);s.originalType=e,s[p]="string"==typeof e?e:a,o[1]=s;for(var c=2;c{n.d(t,{Z:()=>o});var i=n(7294),a=n(6010);const r="tabItem_Ymn6";function o(e){let{children:t,hidden:n,className:o}=e;return i.createElement("div",{role:"tabpanel",className:(0,a.Z)(r,o),hidden:n},t)}},4866:(e,t,n)=>{n.d(t,{Z:()=>N});var i=n(7462),a=n(7294),r=n(6010),o=n(2466),s=n(6550),l=n(1980),c=n(7392),u=n(12);function p(e){return function(e){return a.Children.map(e,(e=>{if(!e||(0,a.isValidElement)(e)&&function(e){const{props:t}=e;return!!t&&"object"==typeof t&&"value"in t}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}(e).map((e=>{let{props:{value:t,label:n,attributes:i,default:a}}=e;return{value:t,label:n,attributes:i,default:a}}))}function d(e){const{values:t,children:n}=e;return(0,a.useMemo)((()=>{const e=t??p(n);return function(e){const t=(0,c.l)(e,((e,t)=>e.value===t.value));if(t.length>0)throw new Error(`Docusaurus error: Duplicate values "${t.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[t,n])}function m(e){let{value:t,tabValues:n}=e;return n.some((e=>e.value===t))}function f(e){let{queryString:t=!1,groupId:n}=e;const i=(0,s.k6)(),r=function(e){let{queryString:t=!1,groupId:n}=e;if("string"==typeof t)return t;if(!1===t)return null;if(!0===t&&!n)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return n??null}({queryString:t,groupId:n});return[(0,l._X)(r),(0,a.useCallback)((e=>{if(!r)return;const t=new URLSearchParams(i.location.search);t.set(r,e),i.replace({...i.location,search:t.toString()})}),[r,i])]}function b(e){const{defaultValue:t,queryString:n=!1,groupId:i}=e,r=d(e),[o,s]=(0,a.useState)((()=>function(e){let{defaultValue:t,tabValues:n}=e;if(0===n.length)throw new Error("Docusaurus error: the component requires at least one children component");if(t){if(!m({value:t,tabValues:n}))throw new Error(`Docusaurus error: The has a defaultValue "${t}" but none of its children has the corresponding value. Available values are: ${n.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return t}const i=n.find((e=>e.default))??n[0];if(!i)throw new Error("Unexpected error: 0 tabValues");return i.value}({defaultValue:t,tabValues:r}))),[l,c]=f({queryString:n,groupId:i}),[p,b]=function(e){let{groupId:t}=e;const n=function(e){return e?`docusaurus.tab.${e}`:null}(t),[i,r]=(0,u.Nk)(n);return[i,(0,a.useCallback)((e=>{n&&r.set(e)}),[n,r])]}({groupId:i}),h=(()=>{const e=l??p;return m({value:e,tabValues:r})?e:null})();(0,a.useLayoutEffect)((()=>{h&&s(h)}),[h]);return{selectedValue:o,selectValue:(0,a.useCallback)((e=>{if(!m({value:e,tabValues:r}))throw new Error(`Can't select invalid tab value=${e}`);s(e),c(e),b(e)}),[c,b,r]),tabValues:r}}var h=n(2389);const g="tabList__CuJ",k="tabItem_LNqP";function v(e){let{className:t,block:n,selectedValue:s,selectValue:l,tabValues:c}=e;const u=[],{blockElementScrollPositionUntilNextRender:p}=(0,o.o5)(),d=e=>{const t=e.currentTarget,n=u.indexOf(t),i=c[n].value;i!==s&&(p(t),l(i))},m=e=>{let t=null;switch(e.key){case"Enter":d(e);break;case"ArrowRight":{const n=u.indexOf(e.currentTarget)+1;t=u[n]??u[0];break}case"ArrowLeft":{const n=u.indexOf(e.currentTarget)-1;t=u[n]??u[u.length-1];break}}t?.focus()};return a.createElement("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,r.Z)("tabs",{"tabs--block":n},t)},c.map((e=>{let{value:t,label:n,attributes:o}=e;return a.createElement("li",(0,i.Z)({role:"tab",tabIndex:s===t?0:-1,"aria-selected":s===t,key:t,ref:e=>u.push(e),onKeyDown:m,onClick:d},o,{className:(0,r.Z)("tabs__item",k,o?.className,{"tabs__item--active":s===t})}),n??t)})))}function y(e){let{lazy:t,children:n,selectedValue:i}=e;const r=(Array.isArray(n)?n:[n]).filter(Boolean);if(t){const e=r.find((e=>e.props.value===i));return e?(0,a.cloneElement)(e,{className:"margin-top--md"}):null}return a.createElement("div",{className:"margin-top--md"},r.map(((e,t)=>(0,a.cloneElement)(e,{key:t,hidden:e.props.value!==i}))))}function w(e){const t=b(e);return a.createElement("div",{className:(0,r.Z)("tabs-container",g)},a.createElement(v,(0,i.Z)({},e,t)),a.createElement(y,(0,i.Z)({},e,t)))}function N(e){const t=(0,h.Z)();return a.createElement(w,(0,i.Z)({key:String(t)},e))}},8846:(e,t,n)=>{n.d(t,{Z:()=>s});var i=n(7294);const a="codeDescContainer_ie8f",r="desc_jyqI",o="example_eYlF";function s(e){let{children:t}=e,n=i.Children.toArray(t).filter((e=>e));return i.createElement("div",{className:a},i.createElement("div",{className:r},n[0]),i.createElement("div",{className:o},n[1]))}},4037:(e,t,n)=>{n.r(t),n.d(t,{assets:()=>p,contentTitle:()=>c,default:()=>f,frontMatter:()=>l,metadata:()=>u,toc:()=>d});var i=n(7462),a=(n(7294),n(3905)),r=n(8846),o=n(4866),s=n(5162);const l={},c="Lifetimes",u={unversionedId:"guides/lifetimes",id:"guides/lifetimes",title:"Lifetimes",description:"Lifetime management controls how long a service's instances will live (from instantiation to disposal) and how they will be reused between resolution requests.",source:"@site/docs/guides/lifetimes.md",sourceDirName:"guides",slug:"/guides/lifetimes",permalink:"/stashbox/docs/guides/lifetimes",draft:!1,editUrl:"https://github.com/z4kn4fein/stashbox/edit/master/docs/docs/guides/lifetimes.md",tags:[],version:"current",lastUpdatedBy:"dependabot[bot]",lastUpdatedAt:1697644420,formattedLastUpdatedAt:"Oct 18, 2023",frontMatter:{},sidebar:"docs",previous:{title:"Service resolution",permalink:"/stashbox/docs/guides/service-resolution"},next:{title:"Scopes",permalink:"/stashbox/docs/guides/scopes"}},p={},d=[{value:"Default lifetime",id:"default-lifetime",level:2},{value:"Transient lifetime",id:"transient-lifetime",level:2},{value:"Singleton lifetime",id:"singleton-lifetime",level:2},{value:"Scoped lifetime",id:"scoped-lifetime",level:2},{value:"Named scope lifetime",id:"named-scope-lifetime",level:2},{value:"Per-request lifetime",id:"per-request-lifetime",level:2},{value:"Per-scoped request lifetime",id:"per-scoped-request-lifetime",level:2},{value:"Custom lifetime",id:"custom-lifetime",level:2}],m={toc:d};function f(e){let{components:t,...n}=e;return(0,a.kt)("wrapper",(0,i.Z)({},m,n,{components:t,mdxType:"MDXLayout"}),(0,a.kt)("h1",{id:"lifetimes"},"Lifetimes"),(0,a.kt)("p",null,"Lifetime management controls how long a service's instances will live (from instantiation to ",(0,a.kt)("a",{parentName:"p",href:"/docs/guides/scopes#disposal"},"disposal"),") and how they will be reused between resolution requests."),(0,a.kt)("admonition",{type:"info"},(0,a.kt)("p",{parentName:"admonition"},"Choosing the right lifetime helps you avoid ",(0,a.kt)("a",{parentName:"p",href:"/docs/diagnostics/validation#lifetime-validation"},"captive dependencies"),".")),(0,a.kt)("h2",{id:"default-lifetime"},"Default lifetime"),(0,a.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"When you are not specifying a lifetime during registration, Stashbox will use the default lifetime. By default, it's set to ",(0,a.kt)("a",{parentName:"p",href:"#transient-lifetime"},"Transient"),", but you can override it with the ",(0,a.kt)("inlineCode",{parentName:"p"},".WithDefaultLifetime()")," ",(0,a.kt)("a",{parentName:"p",href:"/docs/configuration/container-configuration#default-lifetime"},"container configuration option"),". "),(0,a.kt)("p",null,"You can choose either from the pre-defined lifetimes defined on the ",(0,a.kt)("inlineCode",{parentName:"p"},"Lifetimes")," static class or use a ",(0,a.kt)("a",{parentName:"p",href:"#custom-lifetime"},"custom lifetime"),".")),(0,a.kt)("div",null,(0,a.kt)(o.Z,{mdxType:"Tabs"},(0,a.kt)(s.Z,{value:"Transient (default)",label:"Transient (default)",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"var container = new StashboxContainer(options => options\n .WithDefaultLifetime(Lifetimes.Transient));\n"))),(0,a.kt)(s.Z,{value:"Singleton",label:"Singleton",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"var container = new StashboxContainer(options => options\n .WithDefaultLifetime(Lifetimes.Singleton));\n"))),(0,a.kt)(s.Z,{value:"Scoped",label:"Scoped",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"var container = new StashboxContainer(options => options\n .WithDefaultLifetime(Lifetimes.Scoped));\n")))))),(0,a.kt)("h2",{id:"transient-lifetime"},"Transient lifetime"),(0,a.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"A new instance is created for each resolution request. If a transient is referred by multiple consumers in the same ",(0,a.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#resolution-tree"},"resolution tree"),", each will get a new instance.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithLifetime(Lifetimes.Transient));\n")))),(0,a.kt)("admonition",{type:"info"},(0,a.kt)("p",{parentName:"admonition"},"Transient services are not tracked for disposal by default, but this feature can be turned on with the ",(0,a.kt)("inlineCode",{parentName:"p"},".WithDisposableTransientTracking()")," ",(0,a.kt)("a",{parentName:"p",href:"/docs/configuration/container-configuration#tracking-disposable-transients"},"container configuration option"),". When it's enabled, the current ",(0,a.kt)("a",{parentName:"p",href:"/docs/guides/scopes"},"scope")," on which the resolution request was initiated takes the responsibility to track and dispose transient services.")),(0,a.kt)("h2",{id:"singleton-lifetime"},"Singleton lifetime"),(0,a.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"A single instance is created and reused for each resolution request and injected into each consumer."),(0,a.kt)("admonition",{type:"note"},(0,a.kt)("p",{parentName:"admonition"},"Singleton services are disposed when the container (",(0,a.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#root-scope"},"root scope"),") is being disposed."))),(0,a.kt)("div",null,(0,a.kt)(o.Z,{groupId:"lifetime-forms",mdxType:"Tabs"},(0,a.kt)(s.Z,{value:"Longer form",label:"Longer form",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithLifetime(Lifetimes.Singleton));\n"))),(0,a.kt)(s.Z,{value:"Shorter form",label:"Shorter form",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterSingleton();\n")))))),(0,a.kt)("h2",{id:"scoped-lifetime"},"Scoped lifetime"),(0,a.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"A new instance is created for each ",(0,a.kt)("a",{parentName:"p",href:"/docs/guides/scopes"},"scope"),", which will be returned for every resolution request initiated on the given scope. It's like a singleton lifetime within a scope. "),(0,a.kt)("admonition",{type:"note"},(0,a.kt)("p",{parentName:"admonition"},"Scoped services are disposed when their scope is being disposed."))),(0,a.kt)("div",null,(0,a.kt)(o.Z,{groupId:"lifetime-forms",mdxType:"Tabs"},(0,a.kt)(s.Z,{value:"Longer form",label:"Longer form",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithLifetime(Lifetimes.Scoped));\n\nusing var scope = container.BeginScope();\nIJob job = scope.Resolve();\n"))),(0,a.kt)(s.Z,{value:"Shorter form",label:"Shorter form",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterScoped();\n\nusing var scope = container.BeginScope();\nIJob job = scope.Resolve();\n")))))),(0,a.kt)("h2",{id:"named-scope-lifetime"},"Named scope lifetime"),(0,a.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"It is the same as scoped lifetime, except the given service will be selected only when a scope with the same name initiates the resolution request."),(0,a.kt)("p",null,"You can also let a service ",(0,a.kt)("a",{parentName:"p",href:"/docs/guides/scopes#service-as-scope"},"define")," its own named scope. During registration, this scope can be referred to by its name upon using a named scope lifetime."),(0,a.kt)("admonition",{type:"note"},(0,a.kt)("p",{parentName:"admonition"},"Services with named scope lifetime are disposed when the related named scope is being disposed."))),(0,a.kt)("div",null,(0,a.kt)(o.Z,{mdxType:"Tabs"},(0,a.kt)(s.Z,{value:"Named",label:"Named",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register(options => options\n .InNamedScope("DbScope"));\n\nusing var scope = container.BeginScope("DbScope");\nIJob job = scope.Resolve();\n'))),(0,a.kt)(s.Z,{value:"Defined",label:"Defined",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .DefinesScope());\n\nontainer.Register(options => options\n .InScopeDefinedBy());\n\n// the executor will begin a new scope within itself\n// when it gets resolved and DbBackup will be selected\n// and attached to that scope instead.\nusing var scope = container.BeginScope();\nDbJobExecutor executor = scope.Resolve();\n"))),(0,a.kt)(s.Z,{value:"Defined with name",label:"Defined with name",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register(options => options\n .DefinesScope("DbScope"));\n\nontainer.Register(options => options\n .InNamedScope("DbScope"));\n\n// the executor will begin a new scope within itself\n// when it gets resolved and DbBackup will be selected\n// and attached to that scope instead.\nusing var scope = container.BeginScope();\nDbJobExecutor executor = scope.Resolve();\n')))))),(0,a.kt)("h2",{id:"per-request-lifetime"},"Per-request lifetime"),(0,a.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"The requested service will be reused within the whole resolution request. A new instance is created for each individual request .")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithPerRequestLifetime());\n")))),(0,a.kt)("h2",{id:"per-scoped-request-lifetime"},"Per-scoped request lifetime"),(0,a.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"The requested service will behave like a singleton, but only within a scoped dependency request. This means every scoped service will get a new exclusive instance that will be used by its sub-dependencies as well.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithPerScopedRequestLifetime());\n")))),(0,a.kt)("h2",{id:"custom-lifetime"},"Custom lifetime"),(0,a.kt)("p",null,"If you'd like to use a custom lifetime, you can create your implementation by inheriting either from ",(0,a.kt)("inlineCode",{parentName:"p"},"FactoryLifetimeDescriptor")," or from ",(0,a.kt)("inlineCode",{parentName:"p"},"ExpressionLifetimeDescriptor"),", depending on how do you want to manage the service instances."),(0,a.kt)("ul",null,(0,a.kt)("li",{parentName:"ul"},(0,a.kt)("p",{parentName:"li"},(0,a.kt)("strong",{parentName:"p"},"ExpressionLifetimeDescriptor"),": With this, you can build your lifetime with the expression form of the service instantiation."),(0,a.kt)("pre",{parentName:"li"},(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"class CustomLifetime : ExpressionLifetimeDescriptor\n{\n protected override Expression ApplyLifetime(\n Expression expression, // The expression which describes the service creation\n ServiceRegistration serviceRegistration, \n ResolutionContext resolutionContext, \n Type requestedType)\n {\n // Lifetime managing functionality\n }\n}\n"))),(0,a.kt)("li",{parentName:"ul"},(0,a.kt)("p",{parentName:"li"},(0,a.kt)("strong",{parentName:"p"},"FactoryLifetimeDescriptor"),": With this, you can build your lifetime based on a pre-compiled factory delegate used for service instantiation."),(0,a.kt)("pre",{parentName:"li"},(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"class CustomLifetime : FactoryLifetimeDescriptor\n{\n protected override Expression ApplyLifetime(\n Func factory, // The factory used for service creation\n ServiceRegistration serviceRegistration, \n ResolutionContext resolutionContext, \n Type requestedType)\n {\n // Lifetime managing functionality\n }\n}\n")))),(0,a.kt)("p",null,"Then you can use your lifetime like this:"),(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options.WithLifetime(new CustomLifetime()));\n")))}f.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/ff10094c.b547bb1a.js b/assets/js/ff10094c.b547bb1a.js new file mode 100644 index 00000000..2a736a34 --- /dev/null +++ b/assets/js/ff10094c.b547bb1a.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkwebsite=self.webpackChunkwebsite||[]).push([[544],{3905:(e,t,n)=>{n.d(t,{Zo:()=>u,kt:()=>f});var i=n(7294);function a(e,t,n){return t in e?Object.defineProperty(e,t,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[t]=n,e}function r(e,t){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(e);t&&(i=i.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),n.push.apply(n,i)}return n}function o(e){for(var t=1;t=0||(a[n]=e[n]);return a}(e,t);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);for(i=0;i=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(a[n]=e[n])}return a}var l=i.createContext({}),c=function(e){var t=i.useContext(l),n=t;return e&&(n="function"==typeof e?e(t):o(o({},t),e)),n},u=function(e){var t=c(e.components);return i.createElement(l.Provider,{value:t},e.children)},p="mdxType",d={inlineCode:"code",wrapper:function(e){var t=e.children;return i.createElement(i.Fragment,{},t)}},m=i.forwardRef((function(e,t){var n=e.components,a=e.mdxType,r=e.originalType,l=e.parentName,u=s(e,["components","mdxType","originalType","parentName"]),p=c(n),m=a,f=p["".concat(l,".").concat(m)]||p[m]||d[m]||r;return n?i.createElement(f,o(o({ref:t},u),{},{components:n})):i.createElement(f,o({ref:t},u))}));function f(e,t){var n=arguments,a=t&&t.mdxType;if("string"==typeof e||a){var r=n.length,o=new Array(r);o[0]=m;var s={};for(var l in t)hasOwnProperty.call(t,l)&&(s[l]=t[l]);s.originalType=e,s[p]="string"==typeof e?e:a,o[1]=s;for(var c=2;c{n.d(t,{Z:()=>o});var i=n(7294),a=n(6010);const r="tabItem_Ymn6";function o(e){let{children:t,hidden:n,className:o}=e;return i.createElement("div",{role:"tabpanel",className:(0,a.Z)(r,o),hidden:n},t)}},4866:(e,t,n)=>{n.d(t,{Z:()=>N});var i=n(7462),a=n(7294),r=n(6010),o=n(2466),s=n(6550),l=n(1980),c=n(7392),u=n(12);function p(e){return function(e){return a.Children.map(e,(e=>{if(!e||(0,a.isValidElement)(e)&&function(e){const{props:t}=e;return!!t&&"object"==typeof t&&"value"in t}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}(e).map((e=>{let{props:{value:t,label:n,attributes:i,default:a}}=e;return{value:t,label:n,attributes:i,default:a}}))}function d(e){const{values:t,children:n}=e;return(0,a.useMemo)((()=>{const e=t??p(n);return function(e){const t=(0,c.l)(e,((e,t)=>e.value===t.value));if(t.length>0)throw new Error(`Docusaurus error: Duplicate values "${t.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[t,n])}function m(e){let{value:t,tabValues:n}=e;return n.some((e=>e.value===t))}function f(e){let{queryString:t=!1,groupId:n}=e;const i=(0,s.k6)(),r=function(e){let{queryString:t=!1,groupId:n}=e;if("string"==typeof t)return t;if(!1===t)return null;if(!0===t&&!n)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return n??null}({queryString:t,groupId:n});return[(0,l._X)(r),(0,a.useCallback)((e=>{if(!r)return;const t=new URLSearchParams(i.location.search);t.set(r,e),i.replace({...i.location,search:t.toString()})}),[r,i])]}function b(e){const{defaultValue:t,queryString:n=!1,groupId:i}=e,r=d(e),[o,s]=(0,a.useState)((()=>function(e){let{defaultValue:t,tabValues:n}=e;if(0===n.length)throw new Error("Docusaurus error: the component requires at least one children component");if(t){if(!m({value:t,tabValues:n}))throw new Error(`Docusaurus error: The has a defaultValue "${t}" but none of its children has the corresponding value. Available values are: ${n.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return t}const i=n.find((e=>e.default))??n[0];if(!i)throw new Error("Unexpected error: 0 tabValues");return i.value}({defaultValue:t,tabValues:r}))),[l,c]=f({queryString:n,groupId:i}),[p,b]=function(e){let{groupId:t}=e;const n=function(e){return e?`docusaurus.tab.${e}`:null}(t),[i,r]=(0,u.Nk)(n);return[i,(0,a.useCallback)((e=>{n&&r.set(e)}),[n,r])]}({groupId:i}),h=(()=>{const e=l??p;return m({value:e,tabValues:r})?e:null})();(0,a.useLayoutEffect)((()=>{h&&s(h)}),[h]);return{selectedValue:o,selectValue:(0,a.useCallback)((e=>{if(!m({value:e,tabValues:r}))throw new Error(`Can't select invalid tab value=${e}`);s(e),c(e),b(e)}),[c,b,r]),tabValues:r}}var h=n(2389);const g="tabList__CuJ",k="tabItem_LNqP";function v(e){let{className:t,block:n,selectedValue:s,selectValue:l,tabValues:c}=e;const u=[],{blockElementScrollPositionUntilNextRender:p}=(0,o.o5)(),d=e=>{const t=e.currentTarget,n=u.indexOf(t),i=c[n].value;i!==s&&(p(t),l(i))},m=e=>{let t=null;switch(e.key){case"Enter":d(e);break;case"ArrowRight":{const n=u.indexOf(e.currentTarget)+1;t=u[n]??u[0];break}case"ArrowLeft":{const n=u.indexOf(e.currentTarget)-1;t=u[n]??u[u.length-1];break}}t?.focus()};return a.createElement("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,r.Z)("tabs",{"tabs--block":n},t)},c.map((e=>{let{value:t,label:n,attributes:o}=e;return a.createElement("li",(0,i.Z)({role:"tab",tabIndex:s===t?0:-1,"aria-selected":s===t,key:t,ref:e=>u.push(e),onKeyDown:m,onClick:d},o,{className:(0,r.Z)("tabs__item",k,o?.className,{"tabs__item--active":s===t})}),n??t)})))}function y(e){let{lazy:t,children:n,selectedValue:i}=e;const r=(Array.isArray(n)?n:[n]).filter(Boolean);if(t){const e=r.find((e=>e.props.value===i));return e?(0,a.cloneElement)(e,{className:"margin-top--md"}):null}return a.createElement("div",{className:"margin-top--md"},r.map(((e,t)=>(0,a.cloneElement)(e,{key:t,hidden:e.props.value!==i}))))}function w(e){const t=b(e);return a.createElement("div",{className:(0,r.Z)("tabs-container",g)},a.createElement(v,(0,i.Z)({},e,t)),a.createElement(y,(0,i.Z)({},e,t)))}function N(e){const t=(0,h.Z)();return a.createElement(w,(0,i.Z)({key:String(t)},e))}},8846:(e,t,n)=>{n.d(t,{Z:()=>s});var i=n(7294);const a="codeDescContainer_ie8f",r="desc_jyqI",o="example_eYlF";function s(e){let{children:t}=e,n=i.Children.toArray(t).filter((e=>e));return i.createElement("div",{className:a},i.createElement("div",{className:r},n[0]),i.createElement("div",{className:o},n[1]))}},4037:(e,t,n)=>{n.r(t),n.d(t,{assets:()=>p,contentTitle:()=>c,default:()=>f,frontMatter:()=>l,metadata:()=>u,toc:()=>d});var i=n(7462),a=(n(7294),n(3905)),r=n(8846),o=n(4866),s=n(5162);const l={},c="Lifetimes",u={unversionedId:"guides/lifetimes",id:"guides/lifetimes",title:"Lifetimes",description:"Lifetime management controls how long a service's instances will live (from instantiation to disposal) and how they will be reused between resolution requests.",source:"@site/docs/guides/lifetimes.md",sourceDirName:"guides",slug:"/guides/lifetimes",permalink:"/stashbox/docs/guides/lifetimes",draft:!1,editUrl:"https://github.com/z4kn4fein/stashbox/edit/master/docs/docs/guides/lifetimes.md",tags:[],version:"current",lastUpdatedBy:"dependabot[bot]",lastUpdatedAt:1699538162,formattedLastUpdatedAt:"Nov 9, 2023",frontMatter:{},sidebar:"docs",previous:{title:"Service resolution",permalink:"/stashbox/docs/guides/service-resolution"},next:{title:"Scopes",permalink:"/stashbox/docs/guides/scopes"}},p={},d=[{value:"Default lifetime",id:"default-lifetime",level:2},{value:"Transient lifetime",id:"transient-lifetime",level:2},{value:"Singleton lifetime",id:"singleton-lifetime",level:2},{value:"Scoped lifetime",id:"scoped-lifetime",level:2},{value:"Named scope lifetime",id:"named-scope-lifetime",level:2},{value:"Per-request lifetime",id:"per-request-lifetime",level:2},{value:"Per-scoped request lifetime",id:"per-scoped-request-lifetime",level:2},{value:"Custom lifetime",id:"custom-lifetime",level:2}],m={toc:d};function f(e){let{components:t,...n}=e;return(0,a.kt)("wrapper",(0,i.Z)({},m,n,{components:t,mdxType:"MDXLayout"}),(0,a.kt)("h1",{id:"lifetimes"},"Lifetimes"),(0,a.kt)("p",null,"Lifetime management controls how long a service's instances will live (from instantiation to ",(0,a.kt)("a",{parentName:"p",href:"/docs/guides/scopes#disposal"},"disposal"),") and how they will be reused between resolution requests."),(0,a.kt)("admonition",{type:"info"},(0,a.kt)("p",{parentName:"admonition"},"Choosing the right lifetime helps you avoid ",(0,a.kt)("a",{parentName:"p",href:"/docs/diagnostics/validation#lifetime-validation"},"captive dependencies"),".")),(0,a.kt)("h2",{id:"default-lifetime"},"Default lifetime"),(0,a.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"When you are not specifying a lifetime during registration, Stashbox will use the default lifetime. By default, it's set to ",(0,a.kt)("a",{parentName:"p",href:"#transient-lifetime"},"Transient"),", but you can override it with the ",(0,a.kt)("inlineCode",{parentName:"p"},".WithDefaultLifetime()")," ",(0,a.kt)("a",{parentName:"p",href:"/docs/configuration/container-configuration#default-lifetime"},"container configuration option"),". "),(0,a.kt)("p",null,"You can choose either from the pre-defined lifetimes defined on the ",(0,a.kt)("inlineCode",{parentName:"p"},"Lifetimes")," static class or use a ",(0,a.kt)("a",{parentName:"p",href:"#custom-lifetime"},"custom lifetime"),".")),(0,a.kt)("div",null,(0,a.kt)(o.Z,{mdxType:"Tabs"},(0,a.kt)(s.Z,{value:"Transient (default)",label:"Transient (default)",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"var container = new StashboxContainer(options => options\n .WithDefaultLifetime(Lifetimes.Transient));\n"))),(0,a.kt)(s.Z,{value:"Singleton",label:"Singleton",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"var container = new StashboxContainer(options => options\n .WithDefaultLifetime(Lifetimes.Singleton));\n"))),(0,a.kt)(s.Z,{value:"Scoped",label:"Scoped",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"var container = new StashboxContainer(options => options\n .WithDefaultLifetime(Lifetimes.Scoped));\n")))))),(0,a.kt)("h2",{id:"transient-lifetime"},"Transient lifetime"),(0,a.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"A new instance is created for each resolution request. If a transient is referred by multiple consumers in the same ",(0,a.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#resolution-tree"},"resolution tree"),", each will get a new instance.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithLifetime(Lifetimes.Transient));\n")))),(0,a.kt)("admonition",{type:"info"},(0,a.kt)("p",{parentName:"admonition"},"Transient services are not tracked for disposal by default, but this feature can be turned on with the ",(0,a.kt)("inlineCode",{parentName:"p"},".WithDisposableTransientTracking()")," ",(0,a.kt)("a",{parentName:"p",href:"/docs/configuration/container-configuration#tracking-disposable-transients"},"container configuration option"),". When it's enabled, the current ",(0,a.kt)("a",{parentName:"p",href:"/docs/guides/scopes"},"scope")," on which the resolution request was initiated takes the responsibility to track and dispose transient services.")),(0,a.kt)("h2",{id:"singleton-lifetime"},"Singleton lifetime"),(0,a.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"A single instance is created and reused for each resolution request and injected into each consumer."),(0,a.kt)("admonition",{type:"note"},(0,a.kt)("p",{parentName:"admonition"},"Singleton services are disposed when the container (",(0,a.kt)("a",{parentName:"p",href:"/docs/getting-started/glossary#root-scope"},"root scope"),") is being disposed."))),(0,a.kt)("div",null,(0,a.kt)(o.Z,{groupId:"lifetime-forms",mdxType:"Tabs"},(0,a.kt)(s.Z,{value:"Longer form",label:"Longer form",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithLifetime(Lifetimes.Singleton));\n"))),(0,a.kt)(s.Z,{value:"Shorter form",label:"Shorter form",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterSingleton();\n")))))),(0,a.kt)("h2",{id:"scoped-lifetime"},"Scoped lifetime"),(0,a.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"A new instance is created for each ",(0,a.kt)("a",{parentName:"p",href:"/docs/guides/scopes"},"scope"),", which will be returned for every resolution request initiated on the given scope. It's like a singleton lifetime within a scope. "),(0,a.kt)("admonition",{type:"note"},(0,a.kt)("p",{parentName:"admonition"},"Scoped services are disposed when their scope is being disposed."))),(0,a.kt)("div",null,(0,a.kt)(o.Z,{groupId:"lifetime-forms",mdxType:"Tabs"},(0,a.kt)(s.Z,{value:"Longer form",label:"Longer form",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithLifetime(Lifetimes.Scoped));\n\nusing var scope = container.BeginScope();\nIJob job = scope.Resolve();\n"))),(0,a.kt)(s.Z,{value:"Shorter form",label:"Shorter form",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"container.RegisterScoped();\n\nusing var scope = container.BeginScope();\nIJob job = scope.Resolve();\n")))))),(0,a.kt)("h2",{id:"named-scope-lifetime"},"Named scope lifetime"),(0,a.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"It is the same as scoped lifetime, except the given service will be selected only when a scope with the same name initiates the resolution request."),(0,a.kt)("p",null,"You can also let a service ",(0,a.kt)("a",{parentName:"p",href:"/docs/guides/scopes#service-as-scope"},"define")," its own named scope. During registration, this scope can be referred to by its name upon using a named scope lifetime."),(0,a.kt)("admonition",{type:"note"},(0,a.kt)("p",{parentName:"admonition"},"Services with named scope lifetime are disposed when the related named scope is being disposed."))),(0,a.kt)("div",null,(0,a.kt)(o.Z,{mdxType:"Tabs"},(0,a.kt)(s.Z,{value:"Named",label:"Named",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register(options => options\n .InNamedScope("DbScope"));\n\nusing var scope = container.BeginScope("DbScope");\nIJob job = scope.Resolve();\n'))),(0,a.kt)(s.Z,{value:"Defined",label:"Defined",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .DefinesScope());\n\nontainer.Register(options => options\n .InScopeDefinedBy());\n\n// the executor will begin a new scope within itself\n// when it gets resolved and DbBackup will be selected\n// and attached to that scope instead.\nusing var scope = container.BeginScope();\nDbJobExecutor executor = scope.Resolve();\n"))),(0,a.kt)(s.Z,{value:"Defined with name",label:"Defined with name",mdxType:"TabItem"},(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},'container.Register(options => options\n .DefinesScope("DbScope"));\n\nontainer.Register(options => options\n .InNamedScope("DbScope"));\n\n// the executor will begin a new scope within itself\n// when it gets resolved and DbBackup will be selected\n// and attached to that scope instead.\nusing var scope = container.BeginScope();\nDbJobExecutor executor = scope.Resolve();\n')))))),(0,a.kt)("h2",{id:"per-request-lifetime"},"Per-request lifetime"),(0,a.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"The requested service will be reused within the whole resolution request. A new instance is created for each individual request .")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithPerRequestLifetime());\n")))),(0,a.kt)("h2",{id:"per-scoped-request-lifetime"},"Per-scoped request lifetime"),(0,a.kt)(r.Z,{mdxType:"CodeDescPanel"},(0,a.kt)("div",null,(0,a.kt)("p",null,"The requested service will behave like a singleton, but only within a scoped dependency request. This means every scoped service will get a new exclusive instance that will be used by its sub-dependencies as well.")),(0,a.kt)("div",null,(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options\n .WithPerScopedRequestLifetime());\n")))),(0,a.kt)("h2",{id:"custom-lifetime"},"Custom lifetime"),(0,a.kt)("p",null,"If you'd like to use a custom lifetime, you can create your implementation by inheriting either from ",(0,a.kt)("inlineCode",{parentName:"p"},"FactoryLifetimeDescriptor")," or from ",(0,a.kt)("inlineCode",{parentName:"p"},"ExpressionLifetimeDescriptor"),", depending on how do you want to manage the service instances."),(0,a.kt)("ul",null,(0,a.kt)("li",{parentName:"ul"},(0,a.kt)("p",{parentName:"li"},(0,a.kt)("strong",{parentName:"p"},"ExpressionLifetimeDescriptor"),": With this, you can build your lifetime with the expression form of the service instantiation."),(0,a.kt)("pre",{parentName:"li"},(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"class CustomLifetime : ExpressionLifetimeDescriptor\n{\n protected override Expression ApplyLifetime(\n Expression expression, // The expression which describes the service creation\n ServiceRegistration serviceRegistration, \n ResolutionContext resolutionContext, \n Type requestedType)\n {\n // Lifetime managing functionality\n }\n}\n"))),(0,a.kt)("li",{parentName:"ul"},(0,a.kt)("p",{parentName:"li"},(0,a.kt)("strong",{parentName:"p"},"FactoryLifetimeDescriptor"),": With this, you can build your lifetime based on a pre-compiled factory delegate used for service instantiation."),(0,a.kt)("pre",{parentName:"li"},(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"class CustomLifetime : FactoryLifetimeDescriptor\n{\n protected override Expression ApplyLifetime(\n Func factory, // The factory used for service creation\n ServiceRegistration serviceRegistration, \n ResolutionContext resolutionContext, \n Type requestedType)\n {\n // Lifetime managing functionality\n }\n}\n")))),(0,a.kt)("p",null,"Then you can use your lifetime like this:"),(0,a.kt)("pre",null,(0,a.kt)("code",{parentName:"pre",className:"language-cs"},"container.Register(options => options.WithLifetime(new CustomLifetime()));\n")))}f.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/runtime~main.142a6f72.js b/assets/js/runtime~main.142a6f72.js new file mode 100644 index 00000000..dc3a3c19 --- /dev/null +++ b/assets/js/runtime~main.142a6f72.js @@ -0,0 +1 @@ +(()=>{"use strict";var e,t,r,a,f,o={},d={};function n(e){var t=d[e];if(void 0!==t)return t.exports;var r=d[e]={exports:{}};return o[e].call(r.exports,r,r.exports,n),r.exports}n.m=o,e=[],n.O=(t,r,a,f)=>{if(!r){var o=1/0;for(i=0;i=f)&&Object.keys(n.O).every((e=>n.O[e](r[c])))?r.splice(c--,1):(d=!1,f0&&e[i-1][2]>f;i--)e[i]=e[i-1];e[i]=[r,a,f]},n.n=e=>{var t=e&&e.__esModule?()=>e.default:()=>e;return n.d(t,{a:t}),t},r=Object.getPrototypeOf?e=>Object.getPrototypeOf(e):e=>e.__proto__,n.t=function(e,a){if(1&a&&(e=this(e)),8&a)return e;if("object"==typeof e&&e){if(4&a&&e.__esModule)return e;if(16&a&&"function"==typeof e.then)return e}var f=Object.create(null);n.r(f);var o={};t=t||[null,r({}),r([]),r(r)];for(var d=2&a&&e;"object"==typeof d&&!~t.indexOf(d);d=r(d))Object.getOwnPropertyNames(d).forEach((t=>o[t]=()=>e[t]));return o.default=()=>e,n.d(f,o),f},n.d=(e,t)=>{for(var r in t)n.o(t,r)&&!n.o(e,r)&&Object.defineProperty(e,r,{enumerable:!0,get:t[r]})},n.f={},n.e=e=>Promise.all(Object.keys(n.f).reduce(((t,r)=>(n.f[r](e,t),t)),[])),n.u=e=>"assets/js/"+({53:"935f2afb",80:"3a87badd",153:"851a7a9e",195:"c4f5d8e4",196:"115c3c33",263:"19b690e4",302:"6d4ed487",340:"026a065b",343:"b53e7bc5",353:"9ff4038f",371:"4047545b",376:"8a74683a",514:"1be78505",524:"a1e0d343",544:"ff10094c",652:"dd45a7f1",783:"74fdf922",834:"61ac6d0c",835:"7a96ca3d",888:"a0f36b27",918:"17896441",920:"1a4e3797",946:"5b5f0b93",965:"b43f639d",995:"fd379919"}[e]||e)+"."+{53:"d6b8fbfa",80:"faed2df0",153:"1667941e",195:"020f9fb1",196:"97f612ab",263:"17fec708",302:"97626423",340:"d6d30dca",343:"b0225f24",353:"abdc75cc",371:"4c3bd074",376:"3ec71288",514:"ad28c246",524:"40bd786b",544:"b547bb1a",652:"ea398d3a",780:"af5465b3",783:"e9d5d220",804:"3e06e695",834:"69a685cf",835:"18e419c9",888:"df649e51",894:"e787cecf",918:"35819d44",920:"4ef3577b",945:"c9ff869e",946:"cfbbcee3",965:"9e4b25d8",972:"90686a00",995:"bd8013b7"}[e]+".js",n.miniCssF=e=>{},n.g=function(){if("object"==typeof globalThis)return globalThis;try{return this||new Function("return this")()}catch(e){if("object"==typeof window)return window}}(),n.o=(e,t)=>Object.prototype.hasOwnProperty.call(e,t),a={},f="website:",n.l=(e,t,r,o)=>{if(a[e])a[e].push(t);else{var d,c;if(void 0!==r)for(var b=document.getElementsByTagName("script"),i=0;i{d.onerror=d.onload=null,clearTimeout(s);var f=a[e];if(delete a[e],d.parentNode&&d.parentNode.removeChild(d),f&&f.forEach((e=>e(r))),t)return t(r)},s=setTimeout(l.bind(null,void 0,{type:"timeout",target:d}),12e4);d.onerror=l.bind(null,d.onerror),d.onload=l.bind(null,d.onload),c&&document.head.appendChild(d)}},n.r=e=>{"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(e,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(e,"__esModule",{value:!0})},n.p="/stashbox/",n.gca=function(e){return e={17896441:"918","935f2afb":"53","3a87badd":"80","851a7a9e":"153",c4f5d8e4:"195","115c3c33":"196","19b690e4":"263","6d4ed487":"302","026a065b":"340",b53e7bc5:"343","9ff4038f":"353","4047545b":"371","8a74683a":"376","1be78505":"514",a1e0d343:"524",ff10094c:"544",dd45a7f1:"652","74fdf922":"783","61ac6d0c":"834","7a96ca3d":"835",a0f36b27:"888","1a4e3797":"920","5b5f0b93":"946",b43f639d:"965",fd379919:"995"}[e]||e,n.p+n.u(e)},(()=>{var e={303:0,532:0};n.f.j=(t,r)=>{var a=n.o(e,t)?e[t]:void 0;if(0!==a)if(a)r.push(a[2]);else if(/^(303|532)$/.test(t))e[t]=0;else{var f=new Promise(((r,f)=>a=e[t]=[r,f]));r.push(a[2]=f);var o=n.p+n.u(t),d=new Error;n.l(o,(r=>{if(n.o(e,t)&&(0!==(a=e[t])&&(e[t]=void 0),a)){var f=r&&("load"===r.type?"missing":r.type),o=r&&r.target&&r.target.src;d.message="Loading chunk "+t+" failed.\n("+f+": "+o+")",d.name="ChunkLoadError",d.type=f,d.request=o,a[1](d)}}),"chunk-"+t,t)}},n.O.j=t=>0===e[t];var t=(t,r)=>{var a,f,o=r[0],d=r[1],c=r[2],b=0;if(o.some((t=>0!==e[t]))){for(a in d)n.o(d,a)&&(n.m[a]=d[a]);if(c)var i=c(n)}for(t&&t(r);b{"use strict";var e,t,r,a,o,f={},n={};function c(e){var t=n[e];if(void 0!==t)return t.exports;var r=n[e]={exports:{}};return f[e].call(r.exports,r,r.exports,c),r.exports}c.m=f,e=[],c.O=(t,r,a,o)=>{if(!r){var f=1/0;for(b=0;b=o)&&Object.keys(c.O).every((e=>c.O[e](r[d])))?r.splice(d--,1):(n=!1,o0&&e[b-1][2]>o;b--)e[b]=e[b-1];e[b]=[r,a,o]},c.n=e=>{var t=e&&e.__esModule?()=>e.default:()=>e;return c.d(t,{a:t}),t},r=Object.getPrototypeOf?e=>Object.getPrototypeOf(e):e=>e.__proto__,c.t=function(e,a){if(1&a&&(e=this(e)),8&a)return e;if("object"==typeof e&&e){if(4&a&&e.__esModule)return e;if(16&a&&"function"==typeof e.then)return e}var o=Object.create(null);c.r(o);var f={};t=t||[null,r({}),r([]),r(r)];for(var n=2&a&&e;"object"==typeof n&&!~t.indexOf(n);n=r(n))Object.getOwnPropertyNames(n).forEach((t=>f[t]=()=>e[t]));return f.default=()=>e,c.d(o,f),o},c.d=(e,t)=>{for(var r in t)c.o(t,r)&&!c.o(e,r)&&Object.defineProperty(e,r,{enumerable:!0,get:t[r]})},c.f={},c.e=e=>Promise.all(Object.keys(c.f).reduce(((t,r)=>(c.f[r](e,t),t)),[])),c.u=e=>"assets/js/"+({53:"935f2afb",80:"3a87badd",153:"851a7a9e",195:"c4f5d8e4",196:"115c3c33",263:"19b690e4",302:"6d4ed487",340:"026a065b",343:"b53e7bc5",353:"9ff4038f",371:"4047545b",376:"8a74683a",514:"1be78505",524:"a1e0d343",544:"ff10094c",652:"dd45a7f1",783:"74fdf922",834:"61ac6d0c",835:"7a96ca3d",888:"a0f36b27",918:"17896441",920:"1a4e3797",946:"5b5f0b93",965:"b43f639d",995:"fd379919"}[e]||e)+"."+{53:"d6b8fbfa",80:"15110844",153:"d73c6c03",195:"020f9fb1",196:"d8066e20",263:"17fec708",302:"af3bba9a",340:"c7fa72ce",343:"8724ef22",353:"5b6e8ac6",371:"25600c2c",376:"3ec71288",514:"ad28c246",524:"bffda49f",544:"4eb59afd",652:"4a7501f3",780:"af5465b3",783:"deb27988",804:"3e06e695",834:"a648eb65",835:"b2b825bb",888:"df649e51",894:"e787cecf",918:"35819d44",920:"4ef3577b",945:"c9ff869e",946:"f8a5ec34",965:"7859f4e2",972:"90686a00",995:"bc47ef9d"}[e]+".js",c.miniCssF=e=>{},c.g=function(){if("object"==typeof globalThis)return globalThis;try{return this||new Function("return this")()}catch(e){if("object"==typeof window)return window}}(),c.o=(e,t)=>Object.prototype.hasOwnProperty.call(e,t),a={},o="website:",c.l=(e,t,r,f)=>{if(a[e])a[e].push(t);else{var n,d;if(void 0!==r)for(var i=document.getElementsByTagName("script"),b=0;b{n.onerror=n.onload=null,clearTimeout(s);var o=a[e];if(delete a[e],n.parentNode&&n.parentNode.removeChild(n),o&&o.forEach((e=>e(r))),t)return t(r)},s=setTimeout(l.bind(null,void 0,{type:"timeout",target:n}),12e4);n.onerror=l.bind(null,n.onerror),n.onload=l.bind(null,n.onload),d&&document.head.appendChild(n)}},c.r=e=>{"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(e,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(e,"__esModule",{value:!0})},c.p="/stashbox/",c.gca=function(e){return e={17896441:"918","935f2afb":"53","3a87badd":"80","851a7a9e":"153",c4f5d8e4:"195","115c3c33":"196","19b690e4":"263","6d4ed487":"302","026a065b":"340",b53e7bc5:"343","9ff4038f":"353","4047545b":"371","8a74683a":"376","1be78505":"514",a1e0d343:"524",ff10094c:"544",dd45a7f1:"652","74fdf922":"783","61ac6d0c":"834","7a96ca3d":"835",a0f36b27:"888","1a4e3797":"920","5b5f0b93":"946",b43f639d:"965",fd379919:"995"}[e]||e,c.p+c.u(e)},(()=>{var e={303:0,532:0};c.f.j=(t,r)=>{var a=c.o(e,t)?e[t]:void 0;if(0!==a)if(a)r.push(a[2]);else if(/^(303|532)$/.test(t))e[t]=0;else{var o=new Promise(((r,o)=>a=e[t]=[r,o]));r.push(a[2]=o);var f=c.p+c.u(t),n=new Error;c.l(f,(r=>{if(c.o(e,t)&&(0!==(a=e[t])&&(e[t]=void 0),a)){var o=r&&("load"===r.type?"missing":r.type),f=r&&r.target&&r.target.src;n.message="Loading chunk "+t+" failed.\n("+o+": "+f+")",n.name="ChunkLoadError",n.type=o,n.request=f,a[1](n)}}),"chunk-"+t,t)}},c.O.j=t=>0===e[t];var t=(t,r)=>{var a,o,f=r[0],n=r[1],d=r[2],i=0;if(f.some((t=>0!==e[t]))){for(a in n)c.o(n,a)&&(c.m[a]=n[a]);if(d)var b=d(c)}for(t&&t(r);i