What would a copilot for writing and thinking look like? To try answering this question, I built a prototype: Obsidian-Copilot. Given a section header, it helps draft a few paragraphs via retrieval-augmented generation. Also, if you write a daily journal, it can help you reflect on the past week and plan for the week ahead.
Hereās a short 2-minute demo. The code is available at obsidian-copilot.
We start by parsing documents into chunks. A sensible default is to chunk documents by token length, typically 1,500 to 3,000 tokens per chunk. However, I found that this didnāt work very well. A better approach might be to chunk by paragraphs (e.g., split on \n\n
).
Given that my notes are mostly in bullet form, I chunk by top-level bullets: Each chunk is made up of a single top-level bullet and its sub-bullets. There are usually 5 to 10 sub-bullets per top-level bullet making each chunk similar in length to a paragraph.
chunks = defaultdict()
current_chunk = []
chunk_idx = 0
current_header = None
for line in lines:
if '##' in line: # Chunk header = Section header
current_header = line
if line.startswith('- '): # Top-level bullet
if current_chunk: # If chunks accumulated, add it to chunks
if len(current_chunk) >= min_chunk_lines:
chunks[chunk_idx] = current_chunk
chunk_idx += 1
current_chunk = [] # Reset current chunk
if current_header:
current_chunk.append(current_header)
current_chunk.append(line)
Next, we build an OpenSearch index and a semantic index on these chunks. In a previous experiment, I found that embedding-based retrieval alone might be insufficient and thus added classical search (i.e., BM25 via OpenSearch) in this prototype.
For OpenSearch, we start by configuring filters and fields. We include filters such as stripping HTML, removing possessives (i.e., the trailing ās in words), removing stopwords, and basic stemming. These filters are applied on both documents (during indexing) and queries. We also specify the fields we want to index and their respective types. Types matter because filters are applied on text fields (e.g., title, chunk) but not on keyword fields (e.g., path, document type). We donāt apply preprocessing on file paths to keep them as they are.
'mappings': {
'properties': {
'title': {'type': 'text', 'analyzer': 'english_custom'},
'type': {'type': 'keyword'},
'path': {'type': 'keyword'},
'chunk_header': {'type': 'text', 'analyzer': 'english_custom'},
'chunk': {'type': 'text', 'analyzer': 'english_custom'},
}
}
When querying, we apply boosts to make some fields count more towards the relevance score. In this prototype, I arbitrarily boosted titles by 5x and chunk headers (i.e., top-level bullets) by 2x. Retrieval can be improved by tweaking these boosts as well as other features.
For semantic search, we start by picking an embedding model. I referred to the Massive Text Embedding Benchmark Leaderboard, sorted it on descending order of retrieval score, and picked a model that had a good balance of embedding dimension and score.
This led me to e5-small-v2. Currently, itās ranked a respectable 7th, right below text-embedding-ada-002. Whatās impressive is its embedding size of 384 which is far smaller than what most models have (768 - 1,536). And while it supports a maximum sequence length of only 512, this is sufficient given my shorter chunks. (More details in the paper Text Embeddings by Weakly-Supervised Contrastive Pre-training.) After embedding these documents, we store them in a numpy
array.
During query time, we tokenize and embed the query, do a dot product with the document embedding array, and take the top n results (in this case, 10).
def query_semantic(query, tokenizer, model, doc_embeddings_array, n_results=10):
query_tokenized = tokenizer(f'query: {query}', max_length=512, padding=False, truncation=True, return_tensors='pt')
outputs = model(**query_tokenized)
query_embedding = average_pool(outputs.last_hidden_state, query_tokenized['attention_mask'])
query_embedding = F.normalize(query_embedding, p=2, dim=1).detach().numpy()
cos_sims = np.dot(doc_embeddings_array, query_embedding.T)
cos_sims = cos_sims.flatten()
top_indices = np.argsort(cos_sims)[-n_results:][::-1]
return top_indices
If youāre thinking of using the e5 models, remember to add the necessary prefixes during preprocessing. For documents, youāll have to prefix them with āpassage:ā
ā and for queries, youāll have to prefix them with āquery:ā
ā
The retrieval service is a FastAPI app. Given an input query, it performs both BM25 and semantic search, deduplicates the results, and returns the documentsā text and associated title. The latter is used to link source documents when generating the draft.
To start the OpenSearch node and semantic search + FastAPI server, we use a simple-docker compose file. They each run in their own containers, bridged by a common network. For convenience, we also define common commands in a Makefile.
Finally, we integrate with Obsidian via a TypeScript plugin. The obsidian-plugin-sample made it easy to get started and I added functions to display retrieved documents in a new tab, query APIs, and stream the output. (Iām new to TypeScript so feedback appreciated!)
While this prototype uses local notes and journal entries, itās not a stretch to imagine the copilot retrieving from other documents (online). For example, team documents such as product requirements and technical design docs, internal wikis, and even code. Iād guess thatās what Microsoft, Atlassian, and Notion are working on right now.
It also extends beyond personal productivity. Within my field of recommendations and search, researchers and practitioners are excited about layering LLM-based generation on top of existing systems and products to improve the customer experience. (I expect weāll see some of this in production by the end of the year.)
One idea is to try LLMs with larger context sizes that allow us to feed in entire documents instead of chunks. (This may help with retrieval recall but puts more onus on the LLM to identify the relevant context for generation.) Currently, Iām using gpt-3.5-turbo which is a good balance of speed and cost. Nonetheless, Iām excited to try claude-1.3-100k and provide entire documents as context.
Another idea is to augment retrieval with web or internal search when necessary. For example, when documents and notes go stale (e.g., based on last updated timestamp), we can look up the web or internal documents for more recent information.
Hereās the GitHub repo if youāre keen to try. Start by cloning the repo and updating the path to your obsidian-vault and huggingface hub cache. The latter saves us from downloading the tokenizer and model each time you start the containers.
git clone https://github.com/eugeneyan/obsidian-copilot.git
# Open Makefile and update the following paths
export OBSIDIAN_PATH = /Users/eugene/obsidian-vault/
export TRANSFORMER_CACHE = /Users/eugene/.cache/huggingface/hub
Then, build the image and indices before starting the retrieval app.
# Build the docker image
make build
# Start the opensearch container and wait for it to start.
# You should see something like this: [c6587bf83572] Node 'c6587bf83572' initialized
make opensearch
# In ANOTHER terminal, build your artifacts (this can take a while)
make build-artifacts
# Start the app. You should see this: Uvicorn running on http://0.0.0.0:8000
make run
Finally, install the copilot plugin, enable it in community plugin settings, and update the API key. Youāll have to restart your Obsidian app if you had it open before installation.
make install-plugin
If you tried it, I would love to hear how it went, especially where it didnāt work well and how it can be improved. Or if youāve been working with retrieval-augmented generation, Iād love to hear about your experience so far!
If you found this useful, please cite this write-up as:
Yan, Ziyou. (Jun 2023). Obsidian-Copilot: An Assistant for Writing & Reflecting. eugeneyan.com. https://eugeneyan.com/writing/obsidian-copilot/.
or
@article{yan2023copilot,
title = {Obsidian-Copilot: An Assistant for Writing & Reflecting},
author = {Yan, Ziyou},
journal = {eugeneyan.com},
year = {2023},
month = {Jun},
url = {https://eugeneyan.com/writing/obsidian-copilot/}
}
Join 8,500+ readers getting updates on machine learning, RecSys, LLMs, and engineering.