Redigera

Dela via


Händelsedrivet arkitekturformat

Azure Stream Analytics
Azure Functions
Azure Service Bus

En händelsedriven arkitektur består av händelseproducenter som genererar en ström av händelser, händelsekonsumenter som lyssnar efter dessa händelser och händelsekanaler som överför händelser från producenter till konsumenter.

Diagram över ett händelsedrivet arkitekturformat

Händelser levereras nära nog i realtid, så att förbrukarna kan svara omedelbart på händelser när de inträffar. Producenterna frikopplas från konsumenterna – en producent vet inte vilka konsumenter som lyssnar. Förbrukarna är också fristående från varandra och varje förbrukare ser alla händelser. Detta skiljer sig från mönstret Konkurrerande förbrukare där förbrukaren hämtar meddelanden från en kö och ett meddelande bearbetas bara en gång (förutsatt att det inte uppstår några fel). I vissa system, t.ex IoT, måste händelser matas in i mycket stora volymer.

En händelsedriven arkitektur kan använda en modell för publicering/prenumeration (kallas även pub/sub) eller en händelseströmsmodell.

  • Pub/sub: Meddelandeinfrastrukturen håller reda på prenumerationerna. När en händelse publiceras skickas händelsen till varje prenumerant. När en händelse har tagits emot kan den inte spelas upp igen och nya prenumeranter ser inte händelsen.

  • Händelseströmning: Händelser skrivs till en logg. Händelserna är strikt sorterade (inom en partition) och beständiga. Klienterna prenumererar inte på dataströmmen, utan i stället kan en klient läsa från valfri del av strömmen. Klienten är ansvarig för avancera dess position i dataströmmen. Det innebär att en klient kan ansluta till när som helst och spela upp händelser.

Det finns vissa vanliga varianter på förbrukarsidan:

  • Enkel händelsebearbetning. En händelse utlöser direkt en åtgärd i förbrukaren. Du kan till exempel använda Azure Functions med en Service Bus-utlösare, så att en funktion körs varje gång ett meddelande har publicerats i ett Service Bus-ämne.

  • Grundläggande händelsekorrelation. En konsument måste bearbeta ett litet antal diskreta affärshändelser, vanligtvis korrelerade med någon identifierare, och spara viss information från tidigare händelser som ska användas vid bearbetning av senare händelser. Det här mönstret stöds av bibliotek som NServiceBus och MassTransit.

  • Komplex händelsebearbetning. En konsument bearbetar en serie händelser och letar efter mönster i händelsedata med hjälp av en teknik som Azure Stream Analytics. Du kan exempelvis aggregera värdena från en inbäddad enhet via ett tidsfönster och generera en avisering om glidande medelvärde överskrider ett visst tröskelvärde.

  • Händelseströmsbearbetning. Använd en dataströmningsplattform, till exempel Azure IoT Hub eller Apache Kafka, som en pipeline för att mata in händelser och mata dem till strömprocessorer. Strömprocessorerna fungerar så att de behandlar eller transformerar dataströmmen. Det kan finnas flera strömprocessorer för olika delsystem av programmet. Den här metoden passar bra för IoT-arbetsbelastningar.

Källan för händelserna som kan vara externa för systemet, till exempel fysiska enheter i en IoT-lösning. I så fall måste systemet kunna att mata in data med den volym och det dataflöde som krävs av datakällan.

Det finns två huvudsakliga metoder för att strukturera händelsenyttolaster. När du har kontroll över dina händelsekonsumenter kan du fatta det här beslutet om nyttolaststruktur per konsument. blandningsmetoder efter behov inom en enda arbetsbelastning.

  • Inklusive alla obligatoriska attribut i nyttolasten: Den här metoden används när du vill att konsumenterna ska ha all tillgänglig information utan att behöva fråga en extern datakälla. Det kan dock leda till problem med datakonsekvens på grund av flera postsystem, särskilt efter uppdateringar. Avtalshantering och versionshantering kan också bli komplexa.

  • Inkludera endast nyckel(er) i nyttolasten: I den här metoden hämtar konsumenterna nödvändiga attribut, till exempel en primärnyckel, för att oberoende hämta återstående data från en datakälla. Även om den här metoden ger bättre datakonsekvens på grund av ett enda postsystem kan den prestera sämre än den första metoden eftersom konsumenterna ofta måste fråga datakällan. Det finns färre problem med koppling, bandbredd, kontraktshantering eller versionshantering, eftersom händelser är mindre och kontrakt enklare.

I den logiska diagrammet ovan för visas varje typ av konsument som en enda ruta. I praktiken är det vanligt att ha flera instanser av en förbrukare för att undvika att förbrukaren blir en felkritisk systemdel. Flera instanser kan också krävas för att hantera händelsernas volym och frekvens. En enskild förbrukare kan också bearbeta händelser i flera trådar. Detta kan skapa utmaningar om händelser måste bearbetas i ordning eller kräva exakt en gång semantik. Se Minimera samordning.

Det finns två primära topologier i många händelsedrivna arkitekturer:

  • Koordinatortopologi. Komponenter sänder förekomster som händelser till hela systemet, och andra komponenter agerar antingen på händelsen eller bara ignorerar händelsen. Den här topologin är användbar när händelsebearbetningsflödet är relativt enkelt. Det finns ingen central samordning eller orkestrering, så den här topologin kan vara mycket dynamisk. Den här topologin är mycket frikopplad, vilket ger skalbarhet, svarstider och feltolerans för komponenter. Ingen komponent äger eller känner till tillståndet för en affärstransaktion med flera steg och åtgärder vidtas asynkront. Därefter är distribuerade transaktioner riskfyllda eftersom det inte finns några inbyggda sätt att startas om eller spelas upp igen. Strategier för felhantering och manuella åtgärder måste övervägas noggrant eftersom den här topologin kan vara en källa till datainkonsekvens.

  • Medlartopologi. Den här topologin åtgärdar några av bristerna i koordinatortopologin. Det finns en händelsemedlare som hanterar och styr händelseflödet. Händelsemedlaren underhåller tillståndet och hanterar funktioner för felhantering och omstart. Till skillnad från koordinatortopologin sänder komponenter förekomster som kommandon och endast till avsedda kanaler, vanligtvis meddelandeköer. Dessa kommandon förväntas inte ignoreras av sina konsumenter. Den här topologin ger mer kontroll, bättre distribuerad felhantering och potentiellt bättre datakonsekvens. Den här topologin medför ökad koppling mellan komponenter och händelsemedlaren kan bli en flaskhals eller ett tillförlitlighetsproblem.

När ska den här arkitekturen användas?

  • Flera delsystem måste bearbeta samma händelser.
  • Realtidsbearbetning med minsta tidsförskjutning.
  • Komplex händelsebearbetning, till exempel mönstermatchning eller aggregering över tidsfönster.
  • Hög volym och datahastighet, till exempel IoT.

Förmåner

  • Producenter och förbrukare är fristående.
  • Inga punkt-till-punkt-integreringar. Det är lätt att lägga till nya förbrukare i systemet.
  • Förbrukarna kan reagera på händelser direkt när de tas emot.
  • Mycket skalbar, elastisk och distribuerad.
  • Delsystem har fristående vyer över händelseströmmen.

Utmaningar

  • Garanterad leverans.

    Det är viktigt att säkerställa att händelser levereras i en del system, särskilt i IoT-scenarier.

  • Bearbetning av händelser i tur och ordning eller bara en gång.

    Varje typ av förbrukare körs vanligtvis i flera instanser för flexibilitet och skalbarhet. Detta kan skapa en utmaning om händelserna måste bearbetas i ordning (inom en konsumenttyp) eller om idempotent meddelandebearbetningslogik inte implementeras.

  • Samordna meddelanden mellan tjänster.

    Affärsprocesser omfattar ofta flera tjänster som publicerar och prenumererar på meddelanden för att uppnå ett konsekvent resultat för en hel arbetsbelastning. Arbetsflödesmönster som koreografimönstret och Saga Orchestration kan användas för att hantera meddelandeflöden på ett tillförlitligt sätt i olika tjänster.

  • Felhantering.

    Händelsedriven arkitektur använder huvudsakligen asynkron kommunikation. En utmaning med asynkron kommunikation är felhantering. Ett sätt att åtgärda det här problemet är att använda en separat felhanterarprocessor. Så när händelsekonsumenten får ett fel skickar den omedelbart och asynkront den felaktiga händelsen till felhanterarprocessorn och går vidare. Felhanterarprocessorn försöker åtgärda felet och skickar tillbaka händelsen till den ursprungliga inmatningskanalen. Men om felhanterarprocessorn misslyckas kan den skicka den felaktiga händelsen till en administratör för ytterligare kontroll. Om du använder en felhanterarprocessor bearbetas felaktiga händelser utan sekvens när de skickas på nytt.

  • Dataförlust.

    En annan utmaning med asynkron kommunikation är dataförlust. Om någon av komponenterna kraschar innan den bearbetas och överlämnas till nästa komponent tas händelsen bort och hamnar aldrig i slutmålet. För att minimera risken för dataförlust bevarar du händelser under överföring och tar bort eller tar bort händelserna endast när nästa komponent har bekräftat mottagandet av händelsen. Dessa funktioner kallas vanligtvis klient bekräftar läge och senaste deltagare support.

  • Implementera ett traditionellt mönster för begäran-svar.

    Ibland kräver händelseproducenten ett omedelbart svar från händelsekonsumenten, till exempel att skaffa en kundberättigande innan han eller hon fortsätter med en beställning. I händelsedriven arkitektur kan synkron kommunikation uppnås via meddelandehantering med begärandesvar.

    Det här mönstret implementeras vanligtvis genom att använda flera köer – en begäranskö och en svarskö. Händelseproducenten skickar en asynkron begäran till en begäranskö, pausar andra åtgärder för den uppgiften och väntar på ett svar i svarskön. effektivt omvandlar detta till en synkron process. Händelsekonsumenter bearbetar sedan begäran och skickar tillbaka svaret via en svarskö. Den här metoden använder vanligtvis ett sessions-ID för spårning, så händelseproducenten vet vilket meddelande i svarskön som är relaterat till den specifika begäran. Den ursprungliga begäran kan också ange namnet på svarskö, potentiellt tillfälliga, i ett svar-till-huvud eller ett annat ömsesidigt överenskommet anpassat attribut.

  • Upprätthålla rätt antal händelser.

    Att generera ett överdrivet antal detaljerade händelser kan mätta och överbelasta systemet, vilket gör det svårt att effektivt analysera det övergripande händelseflödet. Det här problemet förvärras när ändringar måste återställas. Omvänt kan alltför konsoliderande händelser också skapa problem, vilket resulterar i onödig bearbetning och svar från händelsekonsumenter.

    För att uppnå rätt balans bör du överväga konsekvenserna av händelser och om konsumenterna behöver inspektera händelsens nyttolaster för att fastställa deras svar. Om du till exempel har en komponent för efterlevnadskontroll kan det vara tillräckligt att endast publicera två typer av händelser: kompatibla och icke-kompatibla. Med den här metoden kan varje händelse endast bearbetas av relevanta konsumenter, vilket förhindrar onödig bearbetning.

Ytterligare överväganden

  • Mängden data som ska inkluderas i en händelse kan vara en betydande faktor som påverkar både prestanda och kostnad. Om du lägger till all relevant information som behövs för bearbetning i själva händelsen kan du förenkla bearbetningskoden och spara ytterligare sökningar. Om du placerar den minsta mängden information i en händelse, som bara ett par identifierare, minskar transporttiden och kostnaden, men bearbetningskoden måste leta upp eventuell ytterligare information som behövs. Mer information om detta finns i det här blogginlägget.
  • Även om en begäran endast är synlig för komponenten för hantering av begäran, är händelser ofta synliga för flera komponenter i en arbetsbelastning, även om dessa komponenter inte är avsedda att använda dem eller inte är avsedda att användas. Arbeta med ett tänkesätt för "anta intrång" och tänk på vilken information du tar med i händelser för att förhindra oavsiktlig informationsexponering.
  • Många program använder händelsedriven arkitektur som primär arkitektur. Den här metoden kan dock kombineras med andra arkitekturformat, vilket resulterar i hybridarkitekturer. Vanliga kombinationer är mikrotjänster och rör och filter. Integreringen av händelsedriven arkitektur förbättrar systemprestandan genom att eliminera flaskhalsar och ge tillbakatryck under stora volymer av begäranden.
  • Specifika domäner omfattar ofta flera händelseproducenter, konsumenter eller händelsekanaler. Ändringar i en viss domän kan påverka många komponenter.