Moves through the therapy workflow pioneered by Richard Schwartz.
- Currently deployed at https://internalfamilysystems.ai/
- AWS Fargate/ECS/Gateway
-
git clone https://github.com/JeffKatzy/interfamily-agent
-
cd interfamily-agent
-
poetry shell
-
poetry install
-
mv .env.example .env
-
Add your openai api key, and add abs path to the project in PYTHON_PATH
-
To run as flask api that connects to frontend-interfamily-agent, (or with very simple html frontend at localhost:8000) run
python3 main.py
-
To run as chainlit app, run
chainlit run app.py
Or run backend with:
python3 -i backend_runner.py
Implemented routing using JSON parsing to route between structured questions in the therapy (workflow steps) and general questions from the user.
Incoming user messages are routed to either a general response or a sequenced workflow. A workflow is the sequence of steps that the agent responds to individually.
A workflow is essentially a ViewModel. A workflow has many steps, and an application may have many workflows.
Each step in a workflow has it's own skip logic that indicates if the step is already complete. This is often a combination of the state of the underlying model (eg. the parsed information from the agent) and the workflow itself (eg. have we asked enough times and should move on.)
For example, the bot finds the next step by checking off the steps in sequence to look if the skip
lambda function returns True
. And to execute the lambda, it looks at the related model to see if it has already gathered the related information.
class PartWorkflow(BaseWorkflow):
_model: Part
find_part: WField = WField(prompt="Ask if there's a feeling, struggle, thought pattern, or part they need help with.",
skip=lambda view: bool(view._model.part))
assess_awareness: WField = WField(prompt = "Thank them, mirror using parts language. Then ask if they're aware of this part and how they sense or are aware of the part.",
skip=lambda view: bool(view._model.aware_of_part))
class Part(BaseModel):
part: str = Field("",
description="A feeling, struggle, thought pattern, or part they encounter.")
The BaseWorkflow
class's get_next_message
function will call each skip
procedure in sequence, until reaching what is yet to be completed.
def get_next_message(self):
for field, attrs in self.dict().items():
if attrs.get('skip') and not attrs['skip'](self):
return attrs['prompt']
For example:
service = PartWorkflow(Part())
service.get_next_message()
# Ask if there's a feeling, struggle, thought pattern...
Additional workflows are added by adding them to the Server
.
workflows = [ PartWorkflow(Part())]
server = Server(workflows)
asyncio.run(server.listen())
Organizing this way hopes to achieve the following:
- Add additional steps by defining additional workflows, their related steps (along with skip logic) and any changes to the model layer (data to be parsed/captured).
- Isolate model from views. Not everything we want the bot to say needs a related update to the model layer. The model layer should represent the domain.
- Keep logic about whether each step is complete (and should be skipped) with the step declaration. And allow this logic to be flexible.
- Still should look into Zep for output parsing
- Look at Cerebras for latency
- Have dynamic updates of prompt based on workflow state
- Use different model for parsing vs generation