Szolgáltatás meghatározások

A konfiguráció az a hely, ahol utasítjuk a DI konténert, hogyan állítsa össze az egyes szolgáltatásokat, és hogyan kapcsolja össze őket más függőségekkel. A Nette egy nagyon világos és elegáns módot biztosít erre.

A NEON konfigurációs fájl services szakasza az a hely, ahol definiáljuk az egyéni szolgáltatásainkat és azok konfigurációit. Nézzünk egy egyszerű példát a database nevű szolgáltatás definiálására, amely a PDO osztály egy példányát képviseli:

services:
	database: PDO('sqlite::memory:')

Ez a konfiguráció a következő gyári metódust eredményezi a DI konténerben:

public function createServiceDatabase(): PDO
{
	return new PDO('sqlite::memory:');
}

A szolgáltatásnevek lehetővé teszik, hogy a konfigurációs fájl más részeiben hivatkozzunk rájuk a @serviceName formátumban. Ha nincs szükség a szolgáltatás elnevezésére, akkor egyszerűen használhatunk egy felsorolásjelet:

services:
	- PDO('sqlite::memory:')

Egy szolgáltatás lekérdezéséhez a DI konténerből a getService() metódust használhatjuk a szolgáltatás nevével paraméterként, vagy a getByType() metódust a szolgáltatás típusával:

$database = $container->getService('database');
$database = $container->getByType(PDO::class);

Szolgáltatás létrehozása

Leggyakrabban egy szolgáltatást egyszerűen egy adott osztály példányosításával hozunk létre. Például:

services:
	database: PDO('mysql:host=127.0.0.1;dbname=test', root, secret)

Ha a konfigurációt további kulcsokkal kell bővítenünk, a definíciót több sorban is kibővíthetjük:

services:
	database:
		create: PDO('sqlite::memory:')
		setup: ...

A create kulcsnak van egy aliasa: factory, mindkét változat gyakori a gyakorlatban. Javasoljuk azonban a create használatát.

A konstruktor argumentumai vagy a létrehozási módszer alternatívaként a arguments kulcsban is leírhatók:

services:
	database:
		create: PDO
		arguments: ['mysql:host=127.0.0.1;dbname=test', root, secret]

A szolgáltatásokat nem csak egy osztály egyszerű példányosításával kell létrehozni; létrejöhetnek statikus metódusok vagy más szolgáltatások metódusainak hívásából is:

services:
	database: DatabaseFactory::create()
	router: @routerFactory::create()

Megjegyezzük, hogy az egyszerűség kedvéért a -> helyett a :: használjuk, lásd a kifejezés eszközeit. Ezeket a gyári metódusokat generáljuk:

public function createServiceDatabase(): PDO
{
	return DatabaseFactory::create();
}

public function createServiceRouter(): RouteList
{
	return $this->getService('routerFactory')->create();
}

A DI konténernek tudnia kell a létrehozott szolgáltatás típusát. Ha olyan metódussal hozunk létre egy szolgáltatást, amelynek nincs megadott visszatérési típusa, akkor ezt a típust kifejezetten meg kell említenünk a konfigurációban:

services:
	database:
		create: DatabaseFactory::create()
		type: PDO

Érvek

A konstruktoroknak és metódusoknak a hagyományos PHP-hoz nagyon hasonló módon adunk át argumentumokat:

services:
	database: PDO('mysql:host=127.0.0.1;dbname=test', root, secret)

A jobb olvashatóság érdekében az argumentumokat külön sorokban is felsorolhatjuk. Ebben a formátumban a vesszők használata opcionális:

services:
	database: PDO(
		'mysql:host=127.0.0.1;dbname=test'
		root
		secret
	)

Az argumentumokat el is nevezhetjük, így nem kell aggódnunk a sorrendjük miatt:

services:
	database: PDO(
		username: root
		password: secret
		dsn: 'mysql:host=127.0.0.1;dbname=test'
	)

Ha bizonyos argumentumokat el akar hagyni, és azok alapértelmezett értékét kívánja használni, vagy egy szolgáltatást automatikus bekötéssel szeretne beilleszteni, használjon aláhúzást:

services:
	foo: Foo(_, %appDir%)

Az argumentumok lehetnek szolgáltatások, paraméterek és még sok más, lásd a kifejezés eszközeit.

Beállítás

A setup szakaszban definiáljuk azokat a metódusokat, amelyeket a szolgáltatás létrehozásakor meg kell hívni.

services:
	database:
		create: PDO(%dsn%, %user%, %password%)
		setup:
			- setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION)

PHP nyelven ez így nézne ki:

public function createServiceDatabase(): PDO
{
	$service = new PDO('...', '...', '...');
	$service->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
	return $service;
}

A metódushívások mellett értékeket is átadhat a tulajdonságoknak. Egy elem hozzáadása egy tömbhöz szintén támogatott, de azt idézőjelek közé kell zárni, hogy ne ütközzön a NEON szintaxissal:

services:
	foo:
		create: Foo
		setup:
			- $value = 123
			- '$onClick[]' = [@bar, clickHandler]

PHP-ben ez a következőképpen hangzana:

public function createServiceFoo(): Foo
{
	$service = new Foo;
	$service->value = 123;
	$service->onClick[] = [$this->getService('bar'), 'clickHandler'];
	return $service;
}

A beállításban statikus módszereket vagy más szolgáltatások metódusait is meghívhatja. Ha az aktuális szolgáltatást kell átadni argumentumként, használja a @self címet:

services:
	foo:
		create: Foo
		setup:
			- My\Helpers::initializeFoo(@self)
			- @anotherService::setFoo(@self)

Vegye figyelembe, hogy az egyszerűség kedvéért a -> helyett a ::, lásd a kifejezés eszközeit. Ez a következő gyári metódust generálja:

public function createServiceFoo(): Foo
{
	$service = new Foo;
	My\Helpers::initializeFoo($service);
	$this->getService('anotherService')->setFoo($service);
	return $service;
}

Kifejezés Eszközök

A Nette DI kivételesen gazdag kifejezési lehetőségeket biztosít számunkra, lehetővé téve számunkra, hogy szinte bármit megfogalmazzunk. A konfigurációs fájlokban használhatunk paramétereket:

# paraméter
%wwwDir%

# érték egy paraméterkulcs alatt
%mailer.user%

# paraméter egy karakterláncon belül
'%wwwDir%/images'

Objektumokat is létrehozhatunk, metódusokat és függvényeket hívhatunk:

# hozzon létre egy objektumot
DateTime()

# statikus metódus hívása
Collator::create(%locale%)

# PHP függvény hívása
::getenv(DB_USER)

A szolgáltatásokra a nevük vagy a típusuk alapján hivatkozhatunk:

# szolgáltatás név szerint
@database

# szolgáltatás típusa szerint
@Nette\Database\Connection

Első osztályú hívható szintaxis használata:

# creating a callback, equivalent to [@user, logout]
@user::logout(...)

Konstanciák használata:

# osztálykonstans
FilesystemIterator::SKIP_DOTS

# globális konstans, amelyet a PHP konstans() függvényével kapunk.
::constant(PHP_VERSION)

A metódushívások láncolhatók, akárcsak a PHP-ban. Az egyszerűség kedvéért a -> helyett a :: címet használjuk:

DateTime()::format('Y-m-d')
# PHP: (new DateTime())->format('Y-m-d')

@http.request::getUrl()::getHost()
# PHP: http.request')->getUrl()->getHost()

Ezek a kifejezések bárhol használhatók a szolgáltatások létrehozásakor, az argumentumokban, a beállítási szakaszban vagy a paraméterekben:

parameters:
	ipAddress: @http.request::getRemoteAddress()

services:
	database:
		create: DatabaseFactory::create( @anotherService::getDsn() )
		setup:
			- initialize( ::getenv('DB_USER') )

Különleges funkciók

A konfigurációs fájlokon belül használhatja ezeket a speciális funkciókat:

  • not() az érték negációjához
  • bool(), int(), float(), string() a veszteségmentes típusváltáshoz.
  • typed() a megadott típusú szolgáltatások tömbjének létrehozásához.
  • tagged() egy adott címkével rendelkező összes szolgáltatás tömbjének létrehozásához.
services:
	- Foo(
		id: int(::getenv('ProjectId'))
		productionMode: not(%debugMode%)
	)

A PHP hagyományos típuskiosztásához képest, mint például a (int), a veszteségmentes típuskiosztás kivételt dob a nem numerikus értékek esetén.

A typed() függvény egy adott típusú (osztály vagy interfész) összes szolgáltatásának tömbjét hozza létre. Kizárja azokat a szolgáltatásokat, amelyeknél az automatikus kapcsolás ki van kapcsolva. Több típus is megadható, vesszővel elválasztva.

services:
	- BarsDependent( typed(Bar) )

Az autowiring használatával egy adott típusú szolgáltatások tömbjét is automatikusan átadhatja argumentumként.

A tagged() függvény létrehozza a megadott címkével rendelkező összes szolgáltatás tömbjét. Több címke is felsorolható, vesszővel elválasztva.

services:
	- LoggersDependent( tagged(logger) )

Autowiring

A autowired billentyűvel módosíthatja egy adott szolgáltatás autowiring viselkedését. További részletekért lásd az autowiring fejezetet.

services:
	foo:
		create: Foo
		autowired: false     # a foo szolgáltatás ki van zárva az automatikus bekötésből

Lusta szolgáltatások

A lusta betöltés egy olyan technika, amely a szolgáltatás létrehozását addig késlelteti, amíg arra ténylegesen szükség van. A lusta szolgáltatás létrehozását globálisan engedélyezheti a konfigurációban az összes szolgáltatás számára egyszerre. Az egyes szolgáltatások esetében ez a viselkedés felülbírálható:

services:
	foo:
		create: Foo
		lazy: false

Ha egy szolgáltatás lusta szolgáltatásként van definiálva, akkor a DI konténerből történő lekérdezés egy speciális proxy objektumot fog visszaadni. Ez a proxy úgy néz ki és úgy viselkedik, mint a tényleges szolgáltatás, de a valódi inicializálás (konstruktorhívás és beállítás) csak a metódusok vagy tulajdonságok első meghívásakor történik meg.

A lusta betöltés csak a felhasználó által definiált osztályok esetében használható, a PHP belső osztályai esetében nem. Ehhez a PHP 8.4 vagy újabb verziószámú PHP szükséges.

Címkék

A címkék a szolgáltatások kiegészítő információkkal való kiegészítésére szolgálnak. Egy vagy több címkét rendelhet egy szolgáltatáshoz:

services:
	foo:
		create: Foo
		tags:
			- cached

A címkék értékeket is hordozhatnak:

services:
	foo:
		create: Foo
		tags:
			logger: monolog.logger.event

A tagged() funkcióval lekérdezheti az adott címkékkel rendelkező összes szolgáltatást:

services:
	- LoggersDependent( tagged(logger) )

A DI konténerben a findByTag() metódus segítségével megkaphatja az összes, egy adott címkével rendelkező szolgáltatás nevét:

$names = $container->findByTag('logger');
// $names egy tömb, amely a szolgáltatás nevét és a tag értékét tartalmazza.
// pl. ['foo' => 'monolog.logger.event', ...]

Injektálás mód

A inject: true jelző használata aktiválja a függőségek átadását nyilvános változókon keresztül az inject megjegyzésekkel és az inject*() metódusokkal.

services:
	articles:
		create: App\Model\Articles
		inject: true

Alapértelmezés szerint a inject csak az előadók esetében van aktiválva.

Szolgáltatás módosítások

A DI konténer számos beépített vagy felhasználói bővítéssel hozzáadott szolgáltatást tartalmaz. Ezeknek a szolgáltatásoknak a definícióit közvetlenül a konfigurációban módosíthatja. Például megváltoztathatja a application.application szolgáltatás osztályát, amely a hagyományos Nette\Application\Application, másra:

services:
	application.application:
		create: MyApplication
		alteration: true

A alteration jelző tájékoztató jellegű, jelzi, hogy csupán egy meglévő szolgáltatást módosítunk.

Kiegészíthetjük a beállítást is:

services:
	application.application:
		create: MyApplication
		alteration: true
		setup:
			- '$onStartup[]' = [@resource, init]

Itt jön jól a reset:

services:
	application.application:
		create: MyApplication
		alteration: true
		reset:
			- arguments
			- setup
			- tags

Ha egy bővítmény által hozzáadott szolgáltatást szeretne eltávolítani, akkor ezt a következőképpen teheti meg:

services:
	cache.journal: false
verzió: 3.x