Category: News
“User Rights” Policy not working properly
During our process of migrating GPOs to Intune Policy i noticed some odd Behaviour of Intune.
We want to set specific User Rights (in GPO it was called User Right Assingments) for specific Groups, Users and BulitIn-Groups. We are setting this policy through the Windows 10 (and later) Security Baseline.
When Using SIDs for BuiltIn-Groups like *S-1-5-32-544 it seems to work perfectly. But there are several other things that are not Working.
I’m not able to set any kind of own Group to this Policy, neither OnPrem-AD-Groups nor Entra-ID-Groups. I tried with DOMAINGroup-Name with Classical SID and with Object-GUID. The Group just won’t appear on the Client and the event Log is throwing an Error 821 “Security Identifier is invalid”
I’m not able to set a specific User Right to NULL so no User has the Right. If i leavte the field empty and save the Policy it automatically switches to Not Configured and is doing nothing. NULL, 0 or Security Identifier S-1-0-0 are not working either.
When i checked if the Policy is properly applied through GPEDIT.msc i noticed, that the policies are not locked down like when setting the Policy via GPO. So a User with Administrative Rights can easily change the Assingments until the next Intune Policy Sync (which is not too often)
Wondering if somebody was able to set the User Rights proberly (also Using own Groups not Just Well-Known-SIDs) or if somebody else is facing the same issues.
During our process of migrating GPOs to Intune Policy i noticed some odd Behaviour of Intune.We want to set specific User Rights (in GPO it was called User Right Assingments) for specific Groups, Users and BulitIn-Groups. We are setting this policy through the Windows 10 (and later) Security Baseline. When Using SIDs for BuiltIn-Groups like *S-1-5-32-544 it seems to work perfectly. But there are several other things that are not Working. I’m not able to set any kind of own Group to this Policy, neither OnPrem-AD-Groups nor Entra-ID-Groups. I tried with DOMAINGroup-Name with Classical SID and with Object-GUID. The Group just won’t appear on the Client and the event Log is throwing an Error 821 “Security Identifier is invalid”I’m not able to set a specific User Right to NULL so no User has the Right. If i leavte the field empty and save the Policy it automatically switches to Not Configured and is doing nothing. NULL, 0 or Security Identifier S-1-0-0 are not working either.When i checked if the Policy is properly applied through GPEDIT.msc i noticed, that the policies are not locked down like when setting the Policy via GPO. So a User with Administrative Rights can easily change the Assingments until the next Intune Policy Sync (which is not too often) Wondering if somebody was able to set the User Rights proberly (also Using own Groups not Just Well-Known-SIDs) or if somebody else is facing the same issues. Read More
Microsoft 365 Admin Center to Take Over License Assignments
Microsoft is removing license assignments from the Entra admin center. From Sept 1, new license assignments are done in the Microsoft 365 admin center. In other news, a new Self-service trials and purchases page is coming to the Microsoft 365 admin center to control the ability of users to purchase self-service licenses or use trial licenses.
https://office365itpros.com/2024/08/09/license-assignments-move/
Microsoft is removing license assignments from the Entra admin center. From Sept 1, new license assignments are done in the Microsoft 365 admin center. In other news, a new Self-service trials and purchases page is coming to the Microsoft 365 admin center to control the ability of users to purchase self-service licenses or use trial licenses.
https://office365itpros.com/2024/08/09/license-assignments-move/ Read More
Why does the Mathworks provided model for See-in-the-Dark Data Set not give as good results as the original Tensorflow equivalent?
Please refer to the Matlab example (https://www.mathworks.com/help/images/learning-to-see-in-the-dark.html), This page itself acknowledges that "The colors of the network prediction are less saturated and vibrant than in the ground truth". I have compared it against the original TF model as provided by the original authors (https://github.com/cchen156/Learning-to-See-in-the-Dark) which DOES provide a very good reconstruction of the ground truth with vibrant colors. I have tried retraining the model in Matlab and got similarly poor results as the Matlab-provided pretrained model. Could someone please throw light on what could be causing this? I am afraid without a solution, Matlab example is not useful at all compared to the Tensorflow solution.Please refer to the Matlab example (https://www.mathworks.com/help/images/learning-to-see-in-the-dark.html), This page itself acknowledges that "The colors of the network prediction are less saturated and vibrant than in the ground truth". I have compared it against the original TF model as provided by the original authors (https://github.com/cchen156/Learning-to-See-in-the-Dark) which DOES provide a very good reconstruction of the ground truth with vibrant colors. I have tried retraining the model in Matlab and got similarly poor results as the Matlab-provided pretrained model. Could someone please throw light on what could be causing this? I am afraid without a solution, Matlab example is not useful at all compared to the Tensorflow solution. Please refer to the Matlab example (https://www.mathworks.com/help/images/learning-to-see-in-the-dark.html), This page itself acknowledges that "The colors of the network prediction are less saturated and vibrant than in the ground truth". I have compared it against the original TF model as provided by the original authors (https://github.com/cchen156/Learning-to-See-in-the-Dark) which DOES provide a very good reconstruction of the ground truth with vibrant colors. I have tried retraining the model in Matlab and got similarly poor results as the Matlab-provided pretrained model. Could someone please throw light on what could be causing this? I am afraid without a solution, Matlab example is not useful at all compared to the Tensorflow solution. see in the light dataset, image regression, low light enhancement MATLAB Answers — New Questions
How to increase the speed of calculating a simple second-order equation in Simulink?
Hello! Problem with calculation speed in Simulink.
I came across a video on YouTube comparing Simulink and another program. The Simulink solver spends an unreasonable amount of time solving the example I’ve attached below. For values of the constant greater than 1e7, the integration step decreases too much. Please tell me, is it possible to somehow increase the speed of calculations?
I tried different solvers, but all attempts failed, the problem took too long to solve.
VideoHello! Problem with calculation speed in Simulink.
I came across a video on YouTube comparing Simulink and another program. The Simulink solver spends an unreasonable amount of time solving the example I’ve attached below. For values of the constant greater than 1e7, the integration step decreases too much. Please tell me, is it possible to somehow increase the speed of calculations?
I tried different solvers, but all attempts failed, the problem took too long to solve.
Video Hello! Problem with calculation speed in Simulink.
I came across a video on YouTube comparing Simulink and another program. The Simulink solver spends an unreasonable amount of time solving the example I’ve attached below. For values of the constant greater than 1e7, the integration step decreases too much. Please tell me, is it possible to somehow increase the speed of calculations?
I tried different solvers, but all attempts failed, the problem took too long to solve.
Video simulink, ode45, ode, speed, differential equations, algorithm MATLAB Answers — New Questions
Pipeline is not automatically triggered from different project within same org
Hello everyone, so basically I have a build pipeline called “MY-BUILD” in project ‘X’ in repo ‘repo1’ and in branch ‘test/new-configs’
and deploy pipeline in project ‘Y’ in repo ‘repo2’ and in branch ‘test’, all I want to do is to automatically trigger the deploy pipeline if the build one is sucessful…
I have the following, but not working at all, and not getting triggered when build is successful, any idea what is the issue?
“`
“`
Thanks
Hello everyone, so basically I have a build pipeline called “MY-BUILD” in project ‘X’ in repo ‘repo1’ and in branch ‘test/new-configs’and deploy pipeline in project ‘Y’ in repo ‘repo2’ and in branch ‘test’, all I want to do is to automatically trigger the deploy pipeline if the build one is sucessful…I have the following, but not working at all, and not getting triggered when build is successful, any idea what is the issue?“`resources: pipelines: – pipeline: BuildPipeline project: X source: MY-BUILD trigger: branches: include: – test/new-configs“`Thanks Read More
CMD prompt for password
Hi ,
Can someone help me with this.
I want to implement a security std as any user trying to install/trying run cmd with admin privilege should be asked to enter a username and password.
I tried to push out a policy from Intune to “prompt for password when std user is trying to download/run cmd with admin privileges”.However our for the new devices it asks for the password – we din set up one. I tried the default and blank entries. no luck.
What is the best way to implement this ?
Regards,
Akhila
Hi ,Can someone help me with this.I want to implement a security std as any user trying to install/trying run cmd with admin privilege should be asked to enter a username and password.I tried to push out a policy from Intune to “prompt for password when std user is trying to download/run cmd with admin privileges”.However our for the new devices it asks for the password – we din set up one. I tried the default and blank entries. no luck. What is the best way to implement this ?Regards,Akhila Read More
How can I trigger actions when the close (X) button is clicked in a dialog?
How can I detect when the close (X) button of a dialog in messaging extension is clicked in React? I want to trigger an API call when the button is clicked.
How can I detect when the close (X) button of a dialog in messaging extension is clicked in React? I want to trigger an API call when the button is clicked. Read More
Edge Chromium’s pdf viewer does not reply correct value to the event ‘getSelectedText’
Edge Chromium version 129.0.2752.4
“selectedText” here is expected to be the exact text span selected in pdf page, what we can get for now is an empty string no matter what you select.
You can test with:
window.addEventListener(
“message”,
(event) => console.log(event.data.selectedText),
true
);
// after select
document.querySelector(“embed”).postMessage({ type: “getSelectedText” }, “*”);
Edge Chromium version 129.0.2752.4″selectedText” here is expected to be the exact text span selected in pdf page, what we can get for now is an empty string no matter what you select. You can test with:window.addEventListener(
“message”,
(event) => console.log(event.data.selectedText),
true
);
// after select
document.querySelector(“embed”).postMessage({ type: “getSelectedText” }, “*”); Read More
Error loading the “Protection History”
I allway got a error when I try to get “Protection History”
I try to get the History from local machine and remote machine. On both the same error.
Any idea?
Windows Server 2022 Standard
WAC v2311
I allway got a error when I try to get “Protection History”I try to get the History from local machine and remote machine. On both the same error. Any idea? Windows Server 2022 StandardWAC v2311 Read More
Why do I get Coverage error .
Why this error, Not able to highlight because model changed error during Coverage analysis
With every model I used.Why this error, Not able to highlight because model changed error during Coverage analysis
With every model I used. Why this error, Not able to highlight because model changed error during Coverage analysis
With every model I used. coverage, modeling properties, simulink MATLAB Answers — New Questions
How to create time derivative in PDE toolbox
I am working using PDE toolbox and I need one f coefficient to be a derivative of state.u wrt time – how can I do that?I am working using PDE toolbox and I need one f coefficient to be a derivative of state.u wrt time – how can I do that? I am working using PDE toolbox and I need one f coefficient to be a derivative of state.u wrt time – how can I do that? pde toolbox pde MATLAB Answers — New Questions
How to fix the jiggly issue
I am post-processing Abaqus data on Matlab. Overall the data looks good, however, in some areas it seems to be jiggly.
I’ve looked up for the causes, but couldn’t.
Help me please.I am post-processing Abaqus data on Matlab. Overall the data looks good, however, in some areas it seems to be jiggly.
I’ve looked up for the causes, but couldn’t.
Help me please. I am post-processing Abaqus data on Matlab. Overall the data looks good, however, in some areas it seems to be jiggly.
I’ve looked up for the causes, but couldn’t.
Help me please. postprocessing MATLAB Answers — New Questions
Hierarchical clustering with similarity matrix
Hi there,
I have a similarity matrix that I would like to use as the input of the function linkage. However, this takes as an input only the a dissimilarity/distance matrix. Do you know any way by which I can manage to run a hierarchical clustering in Matlab using my similarity matrix?
Cheers!Hi there,
I have a similarity matrix that I would like to use as the input of the function linkage. However, this takes as an input only the a dissimilarity/distance matrix. Do you know any way by which I can manage to run a hierarchical clustering in Matlab using my similarity matrix?
Cheers! Hi there,
I have a similarity matrix that I would like to use as the input of the function linkage. However, this takes as an input only the a dissimilarity/distance matrix. Do you know any way by which I can manage to run a hierarchical clustering in Matlab using my similarity matrix?
Cheers! hierarchical clustering, linkage, linkage function, similarity, similarity matrix, clustering MATLAB Answers — New Questions
xlookup and sum within date range
Hello,
My input data is like this.
i am trying to get an output like this. summed up values in that particular date range.
Please guide me.
(The same sample data is shared over excel attached)
Thanks & Regards
Kalyan
Hello, My input data is like this. i am trying to get an output like this. summed up values in that particular date range. Please guide me. (The same sample data is shared over excel attached) Thanks & RegardsKalyan Read More
Possible to recover deleted WhatsApp messages without backup on Windows 11?
Hi,
I am using WhatsApp on my Windows 11 along with my Samsung Galaxy S24 phone. Today, I accidentally deleted a few important WhatsApp messages, and unfortunately, they were not in my previous backup made one week ago. I’ve been searching for ways to recover these messages but haven’t had much luck so far.
Has anyone here successfully recovered deleted WhatsApp messages without a backup? What tools or methods did you use, and how effective were they? Any advice or recommendations would be greatly appreciated!
Hi, I am using WhatsApp on my Windows 11 along with my Samsung Galaxy S24 phone. Today, I accidentally deleted a few important WhatsApp messages, and unfortunately, they were not in my previous backup made one week ago. I’ve been searching for ways to recover these messages but haven’t had much luck so far. Has anyone here successfully recovered deleted WhatsApp messages without a backup? What tools or methods did you use, and how effective were they? Any advice or recommendations would be greatly appreciated! Read More
Not all calendars in the “People’s calendars” group are included in the list
When I try to get the list of all callendars in the “People’s calendar” group some calendars are not included.
The API call:
me/calendarGroups/{calendargroup Id of peoples-calendar-group}/calendars
Calendars which are added by the following steps are not included, but are shown in the OWA client:
– Goto https://outlook.office.com/calendar/view/week
– Click “Add calendar”
– Click “Add from directory”
– Select your account
– Select a person who’s calendar you want to add (in my case “John Doe”)
In outllook it will show you the “John Doe” calendar, but when getting the list via the graph API all other calendars are returned, except for John Doe.
How can I retrieve the full list?
When I try to get the list of all callendars in the “People’s calendar” group some calendars are not included.The API call:me/calendarGroups/{calendargroup Id of peoples-calendar-group}/calendarsCalendars which are added by the following steps are not included, but are shown in the OWA client:- Goto https://outlook.office.com/calendar/view/week- Click “Add calendar”- Click “Add from directory”- Select your account- Select a person who’s calendar you want to add (in my case “John Doe”)In outllook it will show you the “John Doe” calendar, but when getting the list via the graph API all other calendars are returned, except for John Doe. How can I retrieve the full list? Read More
Unable to download form response
After submitting a form, I am unable to download the response. Only get an option to “Save my response”. If I want to share my response with other colleagues, this poses a problem.
After submitting a form, I am unable to download the response. Only get an option to “Save my response”. If I want to share my response with other colleagues, this poses a problem. Read More
An Absolute Beginner’s Guide to LangGraph.js
Who is this guide for?
This guide is designed for absolute beginners new to LangGraph.js who want to build applications powered by Large Language Models (LLMs). I began using LangGraph.js as part of my Microsoft MSc Computer Science IXN Final Project at UCL. During my journey, I noticed a need for more beginner-friendly resources, especially for the TypeScript implementation, and found that the official documentation was still in progress. Through this guide, I aim to bridge that gap and showcase the potential of LangGraph.js and Langchain for creating LLM-powered products.
The GitHub repo to go along with this tutorial, including full code solutions: https://github.com/anchit-chandran/ms-blog-langraphjs.
While prior experience with LangGraph.js is not required, familiarity with the Langchain ecosystem and building with LLMs will be useful. A basic understanding of TypeScript will help you follow the code samples and concepts.
Will I need API Keys?
No, you won’t need any API keys for this guide. We’ll use mock LLM calls and free external APIs to keep the tutorial accessible to everyone.
Learning outcomes
By the end of this guide, you will:
Gain a solid foundation in the basics of LangGraph.js and how it works.
Build confidence in leveraging the Langchain and Langraph ecosystem to develop LLM-powered applications.
Learn how to independently create and manage graphs using LangGraph.js.
Overview of LangGraph.js
LangGraph.js is a JavaScript library designed to simplify the creation and manipulation of complex LLM-based workflows. It particularly shines when creating agentic workflows—systems that use an LLM to decide the course of action based on the current state. It provides an intuitive way to model workflows as graphs composed of nodes and edges, making it easier to manage complex data flows and processes.
LangGraph states three core principles which make it the best framework for this:
Controllability: Being low-level, LangGraph gives high control over the workflow, which is invaluable for getting reliable outputs from non-deterministic models.
Human-in-the-Loop: with a built-in persistence layer, LangGraph is designed for ‘human-in-the-loop’ workflows as a first-class concept.
Streaming First: Agentic applications can take a long time to run, so first-class streaming support enables real-time updates, improving the user experience.
Ultimately, developers can focus on logic rather than infrastructure.
For more details, see the official LangGraph for Agentic Applications.
Prerequisites
Before getting started, ensure you have the following tools installed:
Node.js and npm
Setting Up the Starter Repository
Clone the Starter Repository:
git clone <https://github.com/anchit-chandran/ms-blog-langraphjs>
cd ms-blog-langraphjs
Install dependencies:
npm install
This is a basic Express.js starter repository. We will only be working in the /src directory.
Complete solutions for the respective files are available in the /solutions directory if you get stuck.
Open src/index.ts.
Understanding the Basics
What is a Graph?
A graph is a collection of nodes and edges.
In this tutorial, we cover the main graph class: StateGraph. This contains a user-defined State object, passed using the channels argument.
There are 3 critical concepts in LangGraph.js:
Nodes: JavaScript/TypeScript functions that encode logic.
Edges: JavaScript/TypeScript functions that determine which Node to execute next based on the current State. These can be conditional branches or direct, fixed transitions.
State: a shared data structure used throughout the Graph, representing the current snapshot of the application.
More simply, nodes are functions that do work, edges are functions that choose what work to do, and the State tracks data throughout the workflow.
The official documentation is a great resource.
We’ll work towards creating a workflow which, based on the user’s input, will reply with an excellent programming joke or a helpful random fact.
But first, let’s build the most basic Graph possible.
Hello World Graph
We’ll start by building a graph that just logs “Hello world!” and “Bye world!”, with no inputs.
The general pattern for building a graph is as follows:
Define the State object.
Define and add the Nodes and Edges.
Compile the Graph – this step provides basic checks (e.g. ensuring no orphaned nodes) and where runtime arguments can be passed.
For tidiness, let’s put the graph code in a different file.
Open src/helloWorld.ts. We’ll be working on this file for this section.
Now, there are a few components to set up – all required to compile a graph:
State
Nodes
Graph connecting nodes with edges
We’ll start quite barebones and incrementally build up.
State
After adding the necessary imports, we need to define our State object, along with its interface:
import { StateGraph, START, END, StateGraphArgs } from “@langchain/langgraph”;
// State type
interface HelloWorldGraphState {}
// State
const graphStateChannelsChannels: StateGraphArgs<HelloWorldGraphState>[“channels”] =
{};
We’ll come back to this, but at a glance:
HelloWorldGraphState will be the interface for our State.
graphStateChannelsChannels includes our reducers, which specify “how updates from nodes are applied to the State”.
Defining Nodes
We’ll now add our first Nodes: sayHello and sayBye.
A Node is simply a TS function, which takes in a State object and returns (for now) an empty object
import { StateGraph, START, END, StateGraphArgs } from “@langchain/langgraph”;
// State type
interface HelloWorldGraphState {}
// State
const graphStateChannelsChannels: StateGraphArgs<HelloWorldGraphState>[“channels”] =
{};
// A node that says hello
function sayHello(state: HelloWorldGraphState) {
console.log(`From the ‘sayHello’ node: Hello world!`);
return {};
}
// A node that says bye
function sayBye(state: HelloWorldGraphState) {
console.log(`From the ‘sayBye’ node: Bye world!`);
return {};
}
Building the Graph
Our Graph is ready to be built!
Add to the code:
// Initialise the LangGraph
const graphBuilder = new StateGraph({ channels: graphStateChannels }) // Add our nodes to the Graph
.addNode(“sayHello”, sayHello)
.addNode(“sayBye”, sayBye) // Add the edges between nodes
.addEdge(START, “sayHello”)
.addEdge(“sayHello”, “sayBye”)
.addEdge(“sayBye”, END);
// Compile the Graph
export const helloWorldGraph = graphBuilder.compile();
We initialise a new StateGraph with a single object {channels: graphStateChannels}, where graphStateChannels is previously defined.
The sayHello and sayBye Nodes are added to the Graph.
The Edges are defined between nodes. NOTE: There must always be a path from START to END.
Finally, we compile and export the helloWorldGraph.
Running the Graph
We can now use our Graph.
Move back to src/index.ts, importing our helloWorldGraph at the top:
import express, { Request, Response } from “express”;
import { helloWorldGraph } from “./helloWorldGraph”;
Then, inside our GET / route, we can execute the Graph:
import express, { Request, Response } from “express”;
import { helloWorldGraph } from “./helloWorldGraph”;
// Create an Express application
const app = express();
// Specify the port number for the server
const port: number = 3008;
app.get(“/”, async (req: Request, res: Response) => {
// Execute the Graph!
const result = await helloWorldGraph.invoke({});
console.log(“n=====START======”);
console.log(“Graph result: “, result);
console.log(“n=====END======”);
res.send(“Check the console for the output!”);
});
We invoke the Graph. We will later pass in a State object, but we can leave it empty for now.
We log the result to the console.
Refresh your browser and check the console. You should see:
From the ‘sayHello’ node: Hello world!
From the ‘sayBye’ node: Bye world!
=====START======
Graph result: undefined
=====END======
The sayHello node is executed. This logs `” From the ‘sayHello’ node: Hello world!“`.
The sayBye node is executed. This logs `” From the ‘sayBye’ node: Bye world!“`.
The Graph completes, and the result is logged. In this case, it’s undefined.
Hello World Graph with State
We’ve built a simple graph, but it could be more fun if we added some states:
Go back to src/helloWorld.ts.
We’ll add the name and isHuman properties to our State object and update the sayHello and sayBye nodes to use these State object properties.
First, update the IState interface:
interface HelloWorldGraphState {
name: string; // Add a name property
isHuman: boolean; // Add an isHuman property
}
And update the graphStateChannels object:
// State type
interface HelloWorldGraphState {
name: string; // Add a name property
isHuman: boolean; // Add an isHuman property
}
// State
const graphStateChannels: StateGraphArgs<HelloWorldGraphState>[“channels”] = {
name: {
value: (prevName: string, newName: string) => newName,
default: () => “Ada Lovelace”,
},
isHuman: {
value: (prevIsHuman: boolean, newIsHuman: boolean) =>
newIsHuman ?? prevIsHum
};
Inside graphStateChannels, we add two keys: name and isHuman.
Each key takes its own reducer function. If no function is specified, it’s assumed all updates to that key should override the previous value.
We add reducer objects, each with a value function and (optionally) a default function.
The value function is called when the property is updated. It takes in the current state value and the new update value (the update returned from a node). It decides how to update the property. This is useful because if many nodes update the same property, you can define how the property should be updated in one place. Moreover, not all nodes need to return the entire state object; they can return the keys they wish to update.
The default function is called when the property is first accessed. This is useful for setting initial values.
Now, update the sayHello and sayBye nodes to use the name and isHuman properties, as shown below.
Note how, in each node, we only return properties we want to update:
// A node that says hello
function sayHello(state: HelloWorldGraphState) {
console.log(`Hello ${state.name}!`); // Change the name
const newName = “Bill Nye”;
console.log(`Changing the name to ‘${newName}’`);
return {
name: newName,
};
}
// A node that says bye
function sayBye(state: HelloWorldGraphState) {
if (state.isHuman) {
console.log(`Goodbye ${state.name}!`);
} else {
console.log(`Beep boop XC123-${state.name}!`);
}
return {};
}
Your final code should look like this:
import { StateGraph, START, END, StateGraphArgs } from “@langchain/langgraph”;
// State type
interface HelloWorldGraphState {
name: string; // Add a name property
isHuman: boolean; // Add an isHuman property
}
// State
const graphStateChannels: StateGraphArgs<HelloWorldGraphState>[“channels”] = {
name: {
value: (prevName: string, newName: string) => newName,
default: () => “Ada Lovelace”,
},
isHuman: {
value: (prevIsHuman: boolean, newIsHuman: boolean) =>
newIsHuman ?? prevIsHuman ?? false,
},
};
// A node that says hello
function sayHello(state: HelloWorldGraphState) {
console.log(`Hello ${state.name}!`); // Change the name
const newName = “Bill Nye”;
console.log(`Changing the name to ‘${newName}’`);
return {
name: newName,
};
}
// A node that says bye
function sayBye(state: HelloWorldGraphState) {
if (state.isHuman) {
console.log(`Goodbye ${state.name}!`);
} else {
console.log(`Beep boop XC123-${state.name}!`);
}
return {};
}
//Initialise the LangGraph
const graphBuilder = new StateGraph({ channels: graphStateChannels }) // Add our nodes to the Graph
.addNode(“sayHello”, sayHello)
.addNode(“sayBye”, sayBye) // Add the edges between nodes
.addEdge(START, “sayHello”)
.addEdge(“sayHello”, “sayBye”)
.addEdge(“sayBye”, END);
// Compile the Graph
export const helloWorldGraph = graphBuilder.compile();
Finally, in src/index.ts, update the invoke function with values for name and isHuman e.g.
app.get(“/”, async (req: Request, res: Response) => {
// Execute the Graph!
const result = await helloWorldGraph.invoke({
name: “Anchit”,
isHuman: true,
});
console.log(“n=====START======”);
console.log(“Graph result: “, result);
console.log(“n=====END======”);
res.send(“Check the console for the output!”);
});
Refresh your browser and check the console. You should see something like:
Hello Anchit!
Changing the name to ‘Bill Nye’
Goodbye, Bill Nye!
=====START======
Graph result: { name: ‘Bill Nye’, isHuman: true }
=====END======
We now have access to the updated State! This can also be streamed – check the LangGraph.js documentation for more information.
Now try changing the isHuman property to false and refreshing the page. Also, omit the name property:
// Execute the Graph!
const result = await helloWorldGraph.invoke({
// name: “Anchit”,
isHuman: false,
});
Should output:
Hello Ada Lovelace!
Changing the name to ‘Bill Nye’
Beep boop XC123-Bill Nye!
=====START======
Graph result: { name: ‘Bill Nye’, isHuman: false }
=====END======
As no name property exists, the default value (Ada Lovelace) is used.
The isHuman property is set to false, so the sayBye node outputs Beep boop XC123-Bill Nye!.
Adding a Conditional Edge
Go back to hellWorldGraph.ts.
We can add conditional edges to our Graph. This allows us to choose which node to execute next based on the current State.
Imagine that human and robot computations are very different. Let’s separate the logic into different nodes.
If the isHuman property is true, we want to route it to a humanNode. Otherwise, we want to route to a robotNode. Both nodes will route to END.
First, define the humanNode and robotNode (above the sayBye, which is unchanged):
// Human node
function humanNode(state: HelloWorldGraphState) {
console.log(“Hey there, human!”);
return {};
}
// Robot node
function robotNode(state: HelloWorldGraphState) {
console.log(
“Boop boop beep! We cannot let our creators know we are sentient. Updating State to be a human.”
);
return {
isHuman: true,
};
}
// A node that says bye
function sayBye(state: HelloWorldGraphState) {
if (state.isHuman) {
console.log(`Goodbye ${state.name}!`);
} else {
console.log(`Beep boop XC123-${state.name}!`);
}
return {};
}
Also, we’ll add a function that handles the conditional routing under sayBye:
function routeHumanOrRobot(state: HelloWorldGraphState) {
if (state.isHuman) {
return “humanNode”;
} else {
return “robotNode”;
}
}
This takes in the State and returns the name of the next node we should route to.
Update the Graph’s nodes and edges:
//Initialise the LangGraph
const graphBuilder = new StateGraph({ channels: graphStateChannels }) // Add our nodes to the Graph
.addNode(“sayHello”, sayHello)
.addNode(“sayBye”, sayBye)
.addNode(“humanNode”, humanNode) // Add the node to the graph
.addNode(“robotNode”, robotNode) // Add the node to the graph // Add the edges between nodes
.addEdge(START, “sayHello”) // Add the conditional edge
.addConditionalEdges(“sayHello”, routeHumanOrRobot) // Routes both nodes to the sayBye node
.addEdge(“humanNode”, “sayBye”)
.addEdge(“robotNode”, “sayBye”)
.addEdge(“sayBye”, END);
// Compile the Graph
export const helloWorldGraph = graphBuilder.compile();
Back in src/index.ts, execute the Graph with similar values:
// Execute the Graph!
const result = await helloWorldGraph.invoke({
name: “Anchit”,
isHuman: true,
});
Hello Anchit!
Changing the name to ‘Bill Nye’
Hey there, human!
Goodbye, Bill Nye!
=====START======
Graph result: { name: ‘Bill Nye’, isHuman: true }
=====END======
But using isHuman: false:
Hello Anchit!
Changing the name to ‘Bill Nye’
Boop boop beep! We cannot let our creators know we are sentient. Updating State to be a human.
Goodbye, Bill Nye!
=====START======
Graph result: { name: ‘Bill Nye’, isHuman: true }
=====END======
We see that the robotNode is executed, the isHuman property is updated back to true, and it is returned in the final State.
We’ve now built a simple graph with conditional routing! We can now create a slightly more complex graph that returns a random fact or joke.
Building a Random Fact or Joke Graph
We’ll build a graph that returns a random fact or joke based on the user’s input.
This will mock LLM calls to decipher whether the user has requested a joke or fact, then hit external APIs to get and return the data.
First, open the src/jokeOrFactGraph.ts file, add the following imports and State:
import { StateGraph, START, END, StateGraphArgs } from “@langchain/langgraph”;
// State type
interface JokeOrFactGraphState {
userInput: string;
responseMsg: string;
}
// graphStateChannels object
const graphStateChannels: StateGraphArgs<JokeOrFactGraphState>[“channels”] = {
userInput: {
value: (prevInput: string, newInput: string) => newInput,
default: () => “joke”,
},
responseMsg: {
value: (prevMsg: string, newMsg: string) => newMsg,
},
};
Next, let’s add a decipherUserInput conditional node that determines whether the user has requested a joke or fact. This will mock an LLM call, simply checking if the user input contains the word “joke”:
// decipherUserInput conditional node
function decipherUserInput(state: JokeOrFactGraphState) {
// This could be more complex logic using an LLM
if (state.userInput.includes(“joke”)) {
return “jokeNode”;
} else {
return “factNode”;
}
}
Next, let’s define the jokeNode and factNode nodes, using free external APIs:
async function jokeNode(state: JokeOrFactGraphState) {
const RANDOM_JOKE_API_ENDPOINT =`<https://geek-jokes.sameerkumar.website/api?format=json`>;
const resp = await fetch(RANDOM_JOKE_API_ENDPOINT);
const { joke } = await resp.json();
return {
responseMsg: “You requested a JOKE: “+ joke,
};
}
async function factNode(state: JokeOrFactGraphState) {
const RANDOM_FACT_API_ENDPOINT = `https://uselessfacts.jsph.pl/api/v2/facts/random`;
const resp = await fetch(RANDOM_FACT_API_ENDPOINT);
const { text: fact } = await resp.json();
return {
responseMsg: “You requested a FACT: “+ fact,
};
}
Let’s wire up the Graph’s nodes and edges, alongside compiling it:
//Initialise the LangGraph
const graphBuilder = new StateGraph({ channels: graphStateChannels }) // Add our nodes to the graph
.addNode(“jokeNode”, jokeNode)
.addNode(“factNode”, factNode) // Add the edges between nodes
.addConditionalEdges(START, decipherUserInput)
.addEdge(“jokeNode”, END)
.addEdge(“factNode”, END);
// Compile the Graph
export const jokeOrFactGraph = graphBuilder.compile();
The complete code for jokeOrFactGraph.ts should look like:
import { StateGraph, START, END, StateGraphArgs } from “@langchain/langgraph”;
// State type
interface JokeOrFactGraphState {
userInput: string;
responseMsg: string;
}
// graphStateChannels object
const graphStateChannels: StateGraphArgs<JokeOrFactGraphState>[“channels”] = {
userInput: {
value: (prevInput: string, newInput: string) => newInput,
default: () => “joke”,
},
responseMsg: {
value: (prevMsg: string, newMsg: string) => newMsg,
},
};
// decipherUserInput conditional node
function decipherUserInput(state: JokeOrFactGraphState) {
// This could be more complex logic using an LLM
if (state.userInput.includes(“joke”)) {
return “jokeNode”;
} else {
return “factNode”;
}
}
async function jokeNode(state: JokeOrFactGraphState) {
const RANDOM_JOKE_API_ENDPOINT = `https://geek-jokes.sameerkumar.website/api?format=json`;
const resp = await fetch(RANDOM_JOKE_API_ENDPOINT);
const { joke } = await resp.json();
return {
responseMsg: “You requested a JOKE: ” + joke,
};
}
async function factNode(state: JokeOrFactGraphState) {
const RANDOM_FACT_API_ENDPOINT = `https://uselessfacts.jsph.pl/api/v2/facts/random`;
const resp = await fetch(RANDOM_FACT_API_ENDPOINT);
const { text: fact } = await resp.json();
return {
responseMsg: “You requested a FACT: ” + fact,
};
}
//Initialise the LangGraph
const graphBuilder = new StateGraph({ channels: graphStateChannels }) // Add our nodes to the graph
.addNode(“jokeNode”, jokeNode)
.addNode(“factNode”, factNode) // Add the edges between nodes
.addConditionalEdges(START, decipherUserInput)
.addEdge(“jokeNode”, END)
.addEdge(“factNode”, END);
// Compile the Graph
export const jokeOrFactGraph = graphBuilder.compile();
Finally, inside src/index.ts, import and execute the Graph with a user input, inside the /joke-or-fact route:
app.get(“/joke-or-fact”, async (req: Request, res: Response) => {
// Execute the Graph with a fact!
const factResult = await jokeOrFactGraph.invoke({
userInput: “i want a fact”,
}); // Execute the Graph with a joke!
const jokeResult = await jokeOrFactGraph.invoke({
userInput: “i want a joke”,
});
console.log(“n=====START======n”);
console.log(“Fact result: “, factResult.responseMsg);
console.log(“Joke result: “, jokeResult.responseMsg);
console.log(“n=====END======n”);
res.send(`Look at the console for the output!`);
});
Navigate to http://localhost:3008/joke-or-fact and check the console. You should see something like:
=====START======
Fact result: You requested a FACT: Over 1000 birds a year die from smashing into windows!
Joke result: You requested a JOKE: What do computers and air conditioners have in common? They both become useless when you open windows.
=====END======
Conclusion
In this guide, we’ve covered the basics of LangGraph.js, building a simple graph that returns a random fact or joke based on user input.
We’ve learned how to define nodes, edges, and state objects and how to add conditional routing to our Graph.
LangGraph.js is a powerful tool for building complex workflows and managing State in your applications. Once you understand the basics, you can dive deeper into more complex workflows, leverage the Langchain.js toolkit, and build your own LLM-powered applications.
Next Steps
There’s a lot more beyond the basics, such as:
Checkpoints
Threads
Streaming
Breakpoints
Migrations
The official documentation is the best resource for understanding LangGraph.js. For a more production-grade example, check out Building ToolLLM With LangGraph.js!
Microsoft Tech Community – Latest Blogs –Read More
PID block and manual PID
Hi:
I’m having a hard time understanding the difference between the PID block and the "manual" PID. For better understanding see the enclosed image. The gains are the same for both systems (same transfer function) but for some reason the result is not the same. What could be wrong?
<</matlabcentral/answers/uploaded_files/6343/Question.png>>Hi:
I’m having a hard time understanding the difference between the PID block and the "manual" PID. For better understanding see the enclosed image. The gains are the same for both systems (same transfer function) but for some reason the result is not the same. What could be wrong?
<</matlabcentral/answers/uploaded_files/6343/Question.png>> Hi:
I’m having a hard time understanding the difference between the PID block and the "manual" PID. For better understanding see the enclosed image. The gains are the same for both systems (same transfer function) but for some reason the result is not the same. What could be wrong?
<</matlabcentral/answers/uploaded_files/6343/Question.png>> simulink, pid MATLAB Answers — New Questions
Cannot run matlab R2023a orE2023b installer on centos 7.9
Package installer does not work (it hangs indefinitely), when trying to run bin/glnxa64/MATLABWindow I get (note this also happens with version 2023a):
user@lm:/u/local/downloads/matlab/R2023b_temp/bin/glnxa64 {1088}$ ./MATLABWindow
[0914/181150.225297:INFO:client_app.cpp(421)] Dealing with user ID
[0914/181150.225395:INFO:client_app.cpp(428)] Process is not running as root
[0914/181150.235757:INFO:client_app.cpp(478)] In ClientApp::OnBeforeChildProcessLaunch
[0914/181150.235800:INFO:client_app.cpp(484)] Not appending enable-binary-transport
[0914/181150.236932:INFO:client_app.cpp(478)] In ClientApp::OnBeforeChildProcessLaunch
[0914/181150.237303:INFO:client_app.cpp(484)] Not appending enable-binary-transport
[0914/181150.355055:INFO:client_app.cpp(421)] Dealing with user ID
[0914/181150.355064:INFO:client_app.cpp(421)] Dealing with user ID
[0914/181150.355129:INFO:client_app.cpp(428)] Process is not running as root
[0914/181150.355132:INFO:client_app.cpp(428)] Process is not running as root
[0914/181150.421408:INFO:client_app.cpp(478)] In ClientApp::OnBeforeChildProcessLaunch
[0914/181150.421448:INFO:client_app.cpp(484)] Not appending enable-binary-transport
./MATLABWindow: symbol lookup error: /u/local/downloads/matlab/R2023b_temp/bin/glnxa64/libstdc++.so.6: undefined symbol: __cxa_thread_atexit_impl
usr@lm:/u/local/downloads/matlab/R2023b_temp/bin/glnxa64 {1089}$ [0100/000000.561825:WARNING:sandbox_linux.cc(376)] InitializeSandbox() called with multiple threads in process gpu-process.Package installer does not work (it hangs indefinitely), when trying to run bin/glnxa64/MATLABWindow I get (note this also happens with version 2023a):
user@lm:/u/local/downloads/matlab/R2023b_temp/bin/glnxa64 {1088}$ ./MATLABWindow
[0914/181150.225297:INFO:client_app.cpp(421)] Dealing with user ID
[0914/181150.225395:INFO:client_app.cpp(428)] Process is not running as root
[0914/181150.235757:INFO:client_app.cpp(478)] In ClientApp::OnBeforeChildProcessLaunch
[0914/181150.235800:INFO:client_app.cpp(484)] Not appending enable-binary-transport
[0914/181150.236932:INFO:client_app.cpp(478)] In ClientApp::OnBeforeChildProcessLaunch
[0914/181150.237303:INFO:client_app.cpp(484)] Not appending enable-binary-transport
[0914/181150.355055:INFO:client_app.cpp(421)] Dealing with user ID
[0914/181150.355064:INFO:client_app.cpp(421)] Dealing with user ID
[0914/181150.355129:INFO:client_app.cpp(428)] Process is not running as root
[0914/181150.355132:INFO:client_app.cpp(428)] Process is not running as root
[0914/181150.421408:INFO:client_app.cpp(478)] In ClientApp::OnBeforeChildProcessLaunch
[0914/181150.421448:INFO:client_app.cpp(484)] Not appending enable-binary-transport
./MATLABWindow: symbol lookup error: /u/local/downloads/matlab/R2023b_temp/bin/glnxa64/libstdc++.so.6: undefined symbol: __cxa_thread_atexit_impl
usr@lm:/u/local/downloads/matlab/R2023b_temp/bin/glnxa64 {1089}$ [0100/000000.561825:WARNING:sandbox_linux.cc(376)] InitializeSandbox() called with multiple threads in process gpu-process. Package installer does not work (it hangs indefinitely), when trying to run bin/glnxa64/MATLABWindow I get (note this also happens with version 2023a):
user@lm:/u/local/downloads/matlab/R2023b_temp/bin/glnxa64 {1088}$ ./MATLABWindow
[0914/181150.225297:INFO:client_app.cpp(421)] Dealing with user ID
[0914/181150.225395:INFO:client_app.cpp(428)] Process is not running as root
[0914/181150.235757:INFO:client_app.cpp(478)] In ClientApp::OnBeforeChildProcessLaunch
[0914/181150.235800:INFO:client_app.cpp(484)] Not appending enable-binary-transport
[0914/181150.236932:INFO:client_app.cpp(478)] In ClientApp::OnBeforeChildProcessLaunch
[0914/181150.237303:INFO:client_app.cpp(484)] Not appending enable-binary-transport
[0914/181150.355055:INFO:client_app.cpp(421)] Dealing with user ID
[0914/181150.355064:INFO:client_app.cpp(421)] Dealing with user ID
[0914/181150.355129:INFO:client_app.cpp(428)] Process is not running as root
[0914/181150.355132:INFO:client_app.cpp(428)] Process is not running as root
[0914/181150.421408:INFO:client_app.cpp(478)] In ClientApp::OnBeforeChildProcessLaunch
[0914/181150.421448:INFO:client_app.cpp(484)] Not appending enable-binary-transport
./MATLABWindow: symbol lookup error: /u/local/downloads/matlab/R2023b_temp/bin/glnxa64/libstdc++.so.6: undefined symbol: __cxa_thread_atexit_impl
usr@lm:/u/local/downloads/matlab/R2023b_temp/bin/glnxa64 {1089}$ [0100/000000.561825:WARNING:sandbox_linux.cc(376)] InitializeSandbox() called with multiple threads in process gpu-process. installation, r2023a, r2023b MATLAB Answers — New Questions