Skip to content

sunholo-data/emissary

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

97 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Emissary

https://em.sunholo.com/

Emissary is an AI assisted messenger service that can be used to create an Emissary to speak on your behalf.

Emissaries are generated with a unique link that can be sent to the receiver. The receiver can then converse with the Emissary who has been given the relevant documents, persona and instructions by you the sender. Receivers can then reply to you or the Emissary.

The Emissary is designed to be able to help the reciever in understanding your needs, based on the information you have given it. Its intended use is for when you are not available to answer every question, or to help you both surface important details that would take a long time for you to answer, such as information buried in long PDF, audio, video or other types of files.

The Emissaries are not intended as generalists. Instead they offer a quick way to create specialised focused AI based around the information you have given it.

Tool that the Emissary can use to help faciliate the conversation are:

  • Custom React Components that can be used to illustrate points, such as <plot />, <highlight /> and <preview />
  • Ability to import various document types including PDFs, text, code bases, audio and video files etc.
  • Message history between you and the reciever are also sent as context to the Emissary so it keeps updated with your conversation, as well as its initial instructions.

Under the hood, the Emissary is powered by Gemini 1.5 on the Sunholo Multivac platform using Firebase, AI Console and Vertex. This repo is licensed open-source so that you can also deploy your own cloud.

Install

Clone this repository:

git clone https://github.com/sunholo-data/emissary.git
cd emissary
  • The default dev branch is experimental.
  • The test branch is testing before deployment to prod
  • The prod branch is what is running on https://em.sunholo.com

Node.js frontend

Check you have node.js installed

node -v
#v19.9.0
npm -v
9.6.3

If not install it from https://nodejs.org/

Install the dependencies:

npm install

This will create a node_modules directory containing all the dependencies required for the app.

Run via npm run dev after you have installed the backend below. The frontend will be available on http://localhost:3000

Backend - python

Install uv ( from https://github.com/astral-sh/uv )

curl -LsSf https://astral.sh/uv/install.sh | sh
uv -V
# uv 0.5.2 (195f4b634 2024-11-14)

Use python 3.10 via uv .venv

cd backend
uv venv --python 3.10
source .venv/bin/activate
cd ..

If local development of sunholo-py is required (not usually)

cd backend
uv pip install -e ../../sunholo-py

Should now be able to run the backend server on http://127.0.0.1:1956 via:

# assuming in root
cd backend
uv run app.py 

Install Firebase emulators

This is quickest way to get running locally, but you will later probably want to configure a Firebase cloud project as shown later below.

Make sure you have firebase installed

firebase --version
# 13.24.2

Otherwise install via:

npm install -g firebase-tools

Configure emulators:

firebase init
# Pick firestore, storage and auth should come with that as well

The firebase emulator configuration is already available within firebase.json so use that if you can.

You can check its all running via firebase emulators:start which should start the emulator UI on http://localhost:4000

[Optional] Langfuse

Langfuse is a GenOps tracing, eval and prompt CMS that makes working with GenAI applications easier.

If on Multivac you can use the Langfuse API keys issued for https://langfuse.sunholo.com/ or create your own instance either via self-hosting or using Langfuse's cloud.

Once issued add the following to the backend's .env variables:

LANGFUSE_HOST=https://langfuse.sunholo.com (or your own host)
LANGFUSE_SECRET_KEY=<<secret>>
LANGFUSE_PUBLIC_KEY=<<public>>

When deployed in Multivac these are populated ia Secret Manager during the cloud build.

It is recommended to use Langfuse, since then you can update prompts without needing to redploy, and can analyse sessions and GenAI traces to optimise performance.

Development setup

The below commands will enable the required services, assuming they are installed as above.

  • Node.js frontend
  • Python backend
  • Firebase services locally via emulators
  • [optional] Your own Firestore project in the cloud.
npm run dev # or dev:cloud - see below

Use CTRL-C to close down the services

  • dev This is the main development script. It starts the Firebase emulators, the python backend app and the Next.js app in parallel. This way, you’ll be using the emulators when running the dev command.
  • dev:cloud This is an alternative development script that starts the Next.js dev server and python backend without the local Firebase emulators. This allows you to develop locally but use the Firebase cloud services.
  • start:emulators This runs the Firebase emulators. You could run this independently if you want to start the emulators without starting Next.js.
  • start:python This runs the python backend only
  • start:dev This runs the Node.js app

Day to day usage:

  • Local Development with Emulators: Run npm run dev. This will start the emulators alongside your Next.js app.
  • Local Development with Cloud Services: Run npm run dev:cloud. This will start the Next.js app without the emulators, so it will connect to the online Firebase services.

Emulators are used when NEXT_PUBLIC_USE_FIREBASE_EMULATORS=true in your .env.local or passed during startup, which the above scripts set if needed.

The app will launch locally at http://127.0.0.1:3000/ and the Firestore emulators are locally at http://127.0.0.1:4000/ and the Python backend is available at http://127.0.0.1:1954

Firebase Cloud setup [optional]

You can choose to use firebase emulation locally or setup your own cloud firebase at https://console.firebase.google.com

Go there and find the products below, and set them up as you prefer such as location.

Create a new web app

Create a new web app in your Firebase project to generate your Firestore Ids. You don't need hosting.

Specify the admin

This email when logged in will be the super-admin e.g. can use the welcome bot on the front page: [email protected]

Create a .env.local file

Put the details in will generate in .env.local for local development and copy them up to the FIREBASE_ENV secret on Secret Manager to use in deployments. It should look something like this (see also env.local.example - rename it to .env.local for use)

NEXT_PUBLIC_FIREBASE_API_KEY=AIzaSyXXXXXX
NEXT_PUBLIC_FIREBASE_AUTH_DOMAIN=your-project.firebaseapp.com 
NEXT_PUBLIC_FIREBASE_PROJECT_ID=your-project
NEXT_PUBLIC_FIREBASE_STORAGE_BUCKET=your-project.firebasestorage.app
NEXT_PUBLIC_FIREBASE_MESSAGING_SENDER_ID=123123123123
NEXT_PUBLIC_FIREBASE_APP_ID=1:123123123123:web:ebXXXXddceXXXX
[email protected]

Firebase authentication

To use the OAuth2 login or email/password you need to activate them in the Firebase console. See https://firebase.google.com/docs/auth/

At the moment Email/Password, Google and GitHub are supported.

You should see this once done

You will need to update the listed of the authorized domains with your URL once deployed online, e.g. em.sunholo.com

And add your domain to the frame-src allowed list in next_config.mjs

...
// add your auth domain 
"frame-src 'self' https://*.firebaseapp.com https://*.firebase.com https://*.googleapis.com https://*.sunholo.com",  
...

Firestore

Activate the Firestore service. You can start it in dev or prod mode.

Storage

Activate the storage service. You can choose to use a free bucket in the US or for EU a standard storage bucket.

Firestore Rules and Indexes

Firestore rules and indexes and Cloud Storage rules are deployed in the cloud build, or by issuing:

firebase -P <your-project> --json deploy --only firestore:rules,firestore:indexes,storage

Creating initial Emissary Templates

To seed the initial Emissary templates they need to be added to Firestore. The script to do this is in src/scripts/seed.mjs.

npm run seed

Which runs similar to this:

Loading templates from YAML...
Checking existing templates...
Found existing templates:
- Aitana (ID: c90e19f9-c394-4e71-939c-7b3648e90381)
- Hermes (ID: dc4053f6-8f7f-4b10-aea1-8eaf1e6794ee)
- Emissary Helper (ID: e53df968-8cf3-46e1-adb1-772cf11af98b)

Starting bot template seeding...
Updating template multivac with ID: e53df968-8cf3-46e1-adb1-772cf11af98b
Updating template aitana with ID: c90e19f9-c394-4e71-939c-7b3648e90381
Updating template hermes with ID: dc4053f6-8f7f-4b10-aea1-8eaf1e6794ee
Committing batch...
Bot template seeding completed successfully!
Updated templates: multivac, aitana, hermes
Seeding completed successfully

This will add template bots to Firestore.

Modify the configuration in the src/scripts/templates.yaml script to alter them.

templates:
  multivac:
    name: "Emissary Helper"
    avatar: "/images/avatars/emissary.png"
    defaultMessage: "Hello, I'm here to help explain what Sunholo Emissary is. Ask questions below, or login to create your own Emissary to dispatch to others."
    defaultInstructions: "You are named Sunholo Emissary. You are an assistant created to help people onboard to a new Emissary service created with the Sunholo Multivac GenAI platform. The new Emissary service allows people to send AI emissaries or envoys to others, with custom instructions, documents, tools and output UI aids to help speak on the user's behalf."

  aitana:
    name: "Aitana"
    avatar: "/images/avatars/aitana.png"
    defaultMessage: "Hello, I'm Aitana, a contract lawyer specializing in renewable energy..."
    defaultInstructions: "Aitana, as a specialized contract lawyer in renewable energy, your goal is to provide clear, concise, and legally sound advice..."

  hermes:
    name: "Hermes"
    avatar: "/images/avatars/hermes.png"
    defaultMessage: "Greetings, I am Hermes, your appointed messenger..."
    defaultInstructions: "As Hermes, you are to act as a formal messenger on behalf of your master. Drop references to the greek gods and myths whenever you can."

By default if the template already exists it will not overwrite it. To force overwrites, use seed:force instead.

npm run seed:force

By default it uses the project specified in .env.local NEXT_PUBLIC_FIREBASE_PROJECT_ID but you can override that with:

node src/scripts/seed.mjs --project-id=your-project-id --force

Users will be able to create Emissary bots based on the templates or their own custom emissaries.

Deployment

This repo uses the cloudbuild.yaml script to deploy to the Multivac Cloud running on Google Cloud Platform via its Cloud Build deploying to Cloud Run.

To support deployment the Artifact Registry and Secret Manager services provide docker image hosting and secrets.

The Cloud Build needs/creates these images:

  • firebase to run the firebase commands
  • ui - The React app as built via the Dockerfile
  • backend - the Python app using sunholo-py to run the GenAI calls.

The ui and backend images are deployed to one Cloud Run instance using its multi-container feature - ui listens publically on port 8080 and sends backend API requests internally to the backend at localhost:1954.

License

Copyright [2024] Holosun ApS

Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at

  http://www.apache.org/licenses/LICENSE-2.0

Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License.