Udostępnij przez


Samouczek: Budowanie agentowej aplikacji internetowej w usłudze Azure App Service przy użyciu LangGraph lub Foundry Agent Service (Node.js)

W tym samouczku pokazano, jak dodać funkcję agenta do istniejącej aplikacji opartej na danych Express.js CRUD. Robi to przy użyciu dwóch różnych podejść: LangGraph i Foundry Agent Service.

Jeśli aplikacja internetowa ma już przydatne funkcje, takie jak zakupy, rezerwacja hotelowa lub zarządzanie danymi, stosunkowo proste jest dodanie funkcji agenta do aplikacji internetowej przez opakowywanie tych funkcji w wtyczki (dla LangGraph) lub jako punkt końcowy OpenAPI (dla usługi Agenta Foundry). W tym samouczku zaczniesz od prostej aplikacji typu lista to-do. Na koniec będziesz mieć możliwość tworzenia, aktualizowania i zarządzania zadaniami za pomocą agenta w aplikacji usługi App Service.

Zarówno LangGraph, jak i Foundry Agent Service umożliwiają tworzenie agentowych aplikacji internetowych z wykorzystaniem funkcji opartych na sztucznej inteligencji. Język LangGraph jest podobny do jądra semantycznego firmy Microsoft i jest zestawem SDK, ale semantyczne jądro nie obsługuje obecnie języka JavaScript. W poniższej tabeli przedstawiono niektóre zagadnienia i kompromisy:

Consideration LangGraph Usługa agenta programu Foundry
Performance Szybkie (działa lokalnie) Wolniejsze (zarządzana, zdalna usługa)
Development Pełny kod, maksymalna kontrolka Niski kod, szybka integracja
Testing Testy ręczne/jednostkowe w kodzie Wbudowany plac zabaw do szybkiego testowania
Scalability App-managed Zarządzane przez platformę Azure, autoskalowane
Wytyczne bezpieczeństwa Wymagana niestandardowa implementacja Wbudowane bezpieczeństwo i moderowanie zawartości
Tożsamość Wymagana niestandardowa implementacja Wbudowany identyfikator agenta i uwierzytelnianie
Enterprise Wymagana integracja niestandardowa Wbudowane wdrożenie platformy Microsoft 365/Teams i zintegrowane wywołania narzędzi platformy Microsoft 365.

W tym poradniku nauczysz się, jak:

  • Przekonwertuj istniejącą funkcjonalność aplikacji na wtyczkę langgraph.
  • Dodaj wtyczkę do agenta LangGraph i użyj jej w aplikacji internetowej.
  • Przekonwertuj istniejące funkcje aplikacji na punkt końcowy interfejsu OpenAPI dla usługi Foundry Agent Service.
  • Wywołaj agenta Foundry w aplikacji internetowej.
  • Przyznaj wymagane uprawnienia dla łączności tożsamości zarządzanej.

Prerequisites

Otwieranie przykładu za pomocą usługi Codespaces

Najprostszym sposobem rozpoczęcia pracy jest użycie usługi GitHub Codespaces, która udostępnia kompletne środowisko programistyczne ze wszystkimi wymaganymi wstępnie zainstalowanymi narzędziami.

  1. Przejdź do repozytorium GitHub pod adresem https://github.com/Azure-Samples/app-service-agentic-langgraph-foundry-node.

  2. Wybierz przycisk Kod , wybierz kartę Codespaces i wybierz pozycję Utwórz przestrzeń kodu w obszarze głównym.

  3. Zaczekaj chwilę na zainicjowanie usługi Codespace. Gdy wszystko będzie gotowe, zobaczysz w przeglądarce w pełni skonfigurowane środowisko programistyczne.

  4. Uruchom aplikację lokalnie:

    npm install
    npm run build
    npm start
    
  5. Gdy zobaczysz, że aplikacja uruchomiona na porcie 3000 jest dostępna, wybierz pozycję Otwórz w przeglądarce i dodaj kilka zadań.

    Agenci nie są w pełni skonfigurowani, więc jeszcze nie działają. Skonfigurujesz je później.

Przeanalizować kod agenta

Oba podejścia używają tego samego wzorca implementacji, w którym agent jest inicjowany podczas uruchamiania aplikacji i odpowiada na komunikaty użytkowników według żądań POST.

Element LangGraphTaskAgent jest inicjowany w konstruktorze w pliku src/agents/LangGraphTaskAgent.ts. Kod inicjowania wykonuje następujące czynności:

    constructor(taskService: TaskService) {
        this.taskService = taskService;
        this.memory = new MemorySaver();
        try {
            const endpoint = process.env.AZURE_OPENAI_ENDPOINT;
            const deploymentName = process.env.AZURE_OPENAI_DEPLOYMENT_NAME;

            if (!endpoint || !deploymentName) {
                console.warn('Azure OpenAI configuration missing for LangGraph agent');
                return;
            }
            // Initialize Azure OpenAI client
            const credential = new DefaultAzureCredential();
            const azureADTokenProvider = getBearerTokenProvider(credential, "https://cognitiveservices.azure.com/.default");
            
            this.llm = new AzureChatOpenAI({
                azureOpenAIEndpoint: endpoint,
                azureOpenAIApiDeploymentName: deploymentName,
                azureADTokenProvider: azureADTokenProvider,
                azureOpenAIApiVersion: "2024-10-21"
            });
            // Define tools directly in the array
            const tools = [
                tool(
                    async ({ title, isComplete = false }) => {
                        const task = await this.taskService.addTask(title, isComplete);
                        return `Task created successfully: "${task.title}" (ID: ${task.id})`;
                    },
                    {
                        name: 'createTask',
                        description: 'Create a new task',
                        schema: z.object({
                            title: z.string(),
                            isComplete: z.boolean().optional()
                        }) as any
                    }
                ),
                tool(
                    async () => {
                        const tasks = await this.taskService.getAllTasks();
                        if (tasks.length === 0) {
                            return 'No tasks found.';
                        }
                        return `Found ${tasks.length} tasks:\n` + 
                               tasks.map(t => `- ${t.id}: ${t.title} (${t.isComplete ? 'Complete' : 'Incomplete'})`).join('\n');
                    },
                    {
                        name: 'getTasks',
                        description: 'Get all tasks',
                        schema: z.object({}) as any
                    }
                ),
                tool(
                    async ({ id }) => {
                        const task = await this.taskService.getTaskById(id);
                        if (!task) {
                            return `Task with ID ${id} not found.`;
                        }
                        return `Task ${task.id}: "${task.title}" - Status: ${task.isComplete ? 'Complete' : 'Incomplete'}`;
                    },
                    {
                        name: 'getTask',
                        description: 'Get a specific task by ID',
                        schema: z.object({
                            id: z.number()
                        }) as any
                    }
                ),
                tool(
                    async ({ id, title, isComplete }) => {
                        const updated = await this.taskService.updateTask(id, title, isComplete);
                        if (!updated) {
                            return `Task with ID ${id} not found.`;
                        }
                        return `Task ${id} updated successfully.`;
                    },
                    {
                        name: 'updateTask',
                        description: 'Update an existing task',
                        schema: z.object({
                            id: z.number(),
                            title: z.string().optional(),
                            isComplete: z.boolean().optional()
                        }) as any
                    }
                ),
                tool(
                    async ({ id }) => {
                        const deleted = await this.taskService.deleteTask(id);
                        if (!deleted) {
                            return `Task with ID ${id} not found.`;
                        }
                        return `Task ${id} deleted successfully.`;
                    },
                    {
                        name: 'deleteTask',
                        description: 'Delete a task',
                        schema: z.object({
                            id: z.number()
                        }) as any
                    }
                )
            ];

            // Create the ReAct agent with memory
            this.agent = createReactAgent({
                llm: this.llm,
                tools,
                checkpointSaver: this.memory,
                stateModifier: `You are an AI assistant that manages tasks using CRUD operations.
                
You have access to tools for creating, reading, updating, and deleting tasks.
Always use the appropriate tool for any task management request.
Be helpful and provide clear responses about the actions you take.

If you need more information to complete a request, ask the user for it.`
            });
        } catch (error) {
            console.error('Error initializing LangGraph agent:', error);
        }
    }

Podczas przetwarzania komunikatów użytkownika agent jest wywoływany przy użyciu invoke() komunikatu użytkownika i konfiguracji sesji na potrzeby ciągłości konwersacji:

const result = await this.agent.invoke(
    { 
        messages: [
            { role: 'user', content: message }
        ]
    },
    { 
        configurable: { 
            thread_id: currentSessionId 
        } 
    }
);

Wdrażanie aplikacji przykładowej

Przykładowe repozytorium zawiera szablon interfejsu wiersza polecenia dla deweloperów platformy Azure (AZD), który tworzy aplikację usługi App Service z tożsamością zarządzaną i wdraża przykładową aplikację.

  1. W terminalu zaloguj się do platformy Azure przy użyciu interfejsu wiersza polecenia dla deweloperów platformy Azure:

    azd auth login
    

    Postępuj zgodnie z instrukcjami, aby ukończyć proces uwierzytelniania.

  2. Wdróż aplikację usługi Azure App Service przy użyciu szablonu AZD:

    azd up
    
  3. Po wyświetleniu monitu podaj następujące odpowiedzi:

    Question Answer
    Wprowadź nową nazwę środowiska: Wpisz unikatową nazwę.
    Wybierz subskrypcję platformy Azure do użycia: Wybierz subskrypcję.
    Wybierz grupę zasobów do użycia: Wybierz pozycję Utwórz nową grupę zasobów.
    Wybierz lokalizację, w ramach których chcesz utworzyć grupę zasobów: Wybierz pozycję Szwecja Środkowa.
    Wprowadź nazwę nowej grupy zasobów: Wpisz Enter.
  4. W wynikach AZD znajdź URL swojej aplikacji i otwórz go w przeglądarce. Adres URL wygląda następująco w danych wyjściowych usługi AZD:

     Deploying services (azd deploy)
    
       (✓) Done: Deploying service web
       - Endpoint: <URL>
     
  5. Otwórz automatycznie wygenerowany schemat OpenAPI w ścieżce https://....azurewebsites.net/api/schema . Ten schemat będzie potrzebny później.

    Masz teraz aplikację usługi App Service z tożsamością zarządzaną przypisaną przez system.

Tworzenie i konfigurowanie zasobu rozwiązania Microsoft Foundry

  1. W portalu Foundry upewnij się, że górny przycisk radiowy New Foundry jest ustawiony na aktywny i utwórz projekt.

  2. Wdróż wybrany model (zobacz Przewodnik Szybki start firmy Microsoft Foundry: tworzenie zasobów).

  3. W górnej części placu zabaw modelu skopiuj nazwę modelu.

  4. Najprostszym sposobem na uzyskanie punktu końcowego Azure OpenAI nadal jest korzystanie z portalu klasycznego. Wybierz przycisk radiowy New Foundry , a następnie pozycję Azure OpenAI, a następnie skopiuj adres URL w punkcie końcowym usługi Azure OpenAI na później.

    Zrzut ekranu przedstawiający sposób kopiowania punktu końcowego interfejsu OpenAI i punktu końcowego projektu foundry w portalu foundry.

Przypisywanie wymaganych uprawnień

  1. W górnym menu nowego portalu Foundry wybierz pozycję Obsługa, a następnie wybierz pozycję Administrator. W wierszu projektu Foundry powinny zostać wyświetlone dwa linki. Element w kolumnie Name to zasób projektu Foundry, a element w kolumnie Zasób nadrzędny to zasób Foundry.

    Zrzut ekranu przedstawiający sposób szybkiego przechodzenia do zasobu foundry lub zasobu projektu foundry.

  2. Wybierz zasób Foundry w zasobie nadrzędnym , a następnie wybierz pozycję Zarządzaj tym zasobem w witrynie Azure Portal. W witrynie Azure Portal możesz przypisać dostęp oparty na rolach dla zasobu do wdrożonej aplikacji internetowej.

  3. Dodaj następującą rolę dla zarządzanej tożsamości aplikacji App Service:

    Zasób docelowy Wymagana rola Wymagane do
    Odlewnia Użytkownik Usług Cognitive Services OpenAI Usługa uzupełniania czatu w programie Microsoft Agent Framework.

    Aby uzyskać instrukcje, zobacz temat Przypisywanie ról platformy Azure za pomocą witryny Azure Portal.

Konfigurowanie zmiennych połączenia w przykładowej aplikacji

  1. Otwórz plik env. Korzystając z wartości skopiowanych wcześniej z portalu Foundry, skonfiguruj następujące zmienne:

    Variable Description
    AZURE_OPENAI_ENDPOINT Punkt końcowy usługi Azure OpenAI (skopiowany z klasycznego portalu Foundry).
    AZURE_OPENAI_DEPLOYMENT_NAME Nazwa modelu we wdrożeniu (skopiowana z obszaru roboczego modelu w nowym portalu Foundry).

    Note

    Aby zachować prostotę samouczka, użyjesz tych zmiennych w pliku env zamiast zastępowania ich ustawieniami aplikacji w usłudze App Service.

    Note

    Aby zachować prostotę samouczka, użyjesz tych zmiennych w pliku env zamiast zastępowania ich ustawieniami aplikacji w usłudze App Service.

  2. Zaloguj się do platformy Azure przy użyciu interfejsu wiersza polecenia platformy Azure:

    az login
    

    Dzięki temu biblioteka klienta tożsamości platformy Azure w przykładowym kodzie może odbierać token uwierzytelniania dla zalogowanego użytkownika. Pamiętaj, że wcześniej dodano wymaganą rolę dla tego użytkownika.

  3. Uruchom aplikację lokalnie:

    npm run build
    npm start
    
  4. Gdy zobaczysz, że aplikacja uruchomiona na porcie 3000 jest dostępna, wybierz pozycję Otwórz w przeglądarce.

  5. Wybierz link LangGraph Agent i link Foundry Agent, aby wypróbować interfejs czatu. Jeśli otrzymasz odpowiedź, aplikacja łączy się pomyślnie z zasobem Microsoft Foundry.

  6. W usłudze GitHub codespace wdróż zmiany aplikacji.

    azd up
    
  7. Przejdź ponownie do wdrożonej aplikacji i przetestuj agentów czatu.

Uprzątnij zasoby

Po zakończeniu pracy z aplikacją możesz usunąć zasoby usługi App Service, aby uniknąć ponoszenia dodatkowych kosztów:

azd down --purge

Ponieważ szablon AZD nie zawiera zasobów rozwiązania Microsoft Foundry, należy je usunąć ręcznie, jeśli chcesz.

Więcej zasobów