Skip to content

Commit 4445b69

Browse files
authored
Merge pull request #6 from kj455/feat/create-update-note
feat/create update note
2 parents 23d8600 + 02082d5 commit 4445b69

9 files changed

+250
-5
lines changed

.changeset/hungry-olives-obey.md

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
---
2+
'@kj455/mcp-kibela': minor
3+
---
4+
5+
create and update note

README.md

+6-4
Original file line numberDiff line numberDiff line change
@@ -10,10 +10,12 @@ A Model Context Protocol (MCP) server implementation that enables AI assistants
1010

1111
The mcp-kibela server provides the following features:
1212

13-
- 🔍 **Note Search**: Search Kibela notes by keywords
14-
- 📝 **My Notes**: Fetch your latest notes
15-
- 📖 **Note Content**: Get note content and comments by ID
16-
- 🔗 **Note by Path**: Get note content by path
13+
- **Note Search**: Search Kibela notes by keywords
14+
- **My Notes**: Fetch your latest notes
15+
- **Note Content**: Get note content and comments by ID
16+
- **Note by Path**: Get note content by path
17+
- **Create Note**: Create a new note
18+
- **Update Note Content**: Update note content by note id
1719

1820
---
1921

package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -25,7 +25,7 @@
2525
"build": "tsup src/index.ts --dts --format esm",
2626
"postbuild": "shx chmod +x dist/*.js",
2727
"check": "tsc --noEmit",
28-
"build:watch": "npm run build --watch",
28+
"build:watch": "npm run build -- --watch",
2929
"prepublishOnly": "npm run build",
3030
"format": "prettier --write .",
3131
"changeset": "changeset",

src/graphql/queries/createNote.ts

+46
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,46 @@
1+
import { TypedDocumentNode } from '@graphql-typed-document-node/core'
2+
import { gql } from 'graphql-tag'
3+
import { gqlRequest } from '../request'
4+
5+
type CreateNoteResponse = {
6+
createNote: {
7+
clientMutationId: string
8+
note: {
9+
id: string
10+
title: string
11+
content: string
12+
url: string
13+
}
14+
}
15+
}
16+
17+
type CreateNoteVariables = {
18+
input: {
19+
clientMutationId: string
20+
title: string
21+
content: string
22+
coediting: boolean
23+
groupIds: string[]
24+
folders?: string[]
25+
authorId?: string
26+
draft?: boolean
27+
}
28+
}
29+
30+
const createNoteMutation: TypedDocumentNode<CreateNoteResponse, CreateNoteVariables> = gql`
31+
mutation CreateNote($input: CreateNoteInput!) {
32+
createNote(input: $input) {
33+
clientMutationId
34+
note {
35+
id
36+
title
37+
content
38+
url
39+
}
40+
}
41+
}
42+
`
43+
44+
export async function createNote(variables: CreateNoteVariables): Promise<CreateNoteResponse> {
45+
return gqlRequest(createNoteMutation, variables)
46+
}
+36
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,36 @@
1+
import { TypedDocumentNode } from '@graphql-typed-document-node/core'
2+
import { gql } from 'graphql-tag'
3+
import { gqlRequest } from '../request'
4+
5+
type UpdateNoteContentResponse = {
6+
updateNoteContent: {
7+
clientMutationId: string
8+
note: {
9+
id: string
10+
}
11+
}
12+
}
13+
14+
type UpdateNoteContentVariables = {
15+
input: {
16+
clientMutationId: string
17+
id: string
18+
newContent: string
19+
baseContent: string
20+
}
21+
}
22+
23+
const updateNoteContentMutation: TypedDocumentNode<UpdateNoteContentResponse, UpdateNoteContentVariables> = gql`
24+
mutation UpdateNoteContent($input: UpdateNoteContentInput!) {
25+
updateNoteContent(input: $input) {
26+
clientMutationId
27+
note {
28+
id
29+
}
30+
}
31+
}
32+
`
33+
34+
export async function updateNoteContent(variables: UpdateNoteContentVariables): Promise<UpdateNoteContentResponse> {
35+
return gqlRequest(updateNoteContentMutation, variables)
36+
}

src/tools/createNote.ts

+88
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,88 @@
1+
import { createNote } from '../graphql/queries/createNote'
2+
import { uuid } from '../utils'
3+
import { ToolDefinition } from './types'
4+
5+
export type CreateNoteArgs = {
6+
title: string
7+
content: string
8+
groupIds: string[]
9+
coediting: boolean
10+
folders?: string[]
11+
authorId?: string
12+
draft?: boolean
13+
}
14+
15+
export const createNoteTool: ToolDefinition<CreateNoteArgs> = {
16+
tool: {
17+
name: 'kibela_create_note',
18+
description: 'Create a new note in Kibela.',
19+
inputSchema: {
20+
type: 'object',
21+
properties: {
22+
title: {
23+
type: 'string',
24+
description: 'required: Title of the note',
25+
},
26+
content: {
27+
type: 'string',
28+
description: 'required: Content of the note in markdown format',
29+
},
30+
groupIds: {
31+
type: 'array',
32+
items: {
33+
type: 'string',
34+
},
35+
description: 'required: IDs of the groups to create the note in.',
36+
},
37+
folders: {
38+
type: 'array',
39+
items: {
40+
type: 'string',
41+
},
42+
description: 'IDs of the folders to add the note to.',
43+
},
44+
authorId: {
45+
type: 'string',
46+
description:
47+
'ID of the author of the note. If not specified, the note will be created by the authenticated user.',
48+
},
49+
coediting: {
50+
type: 'boolean',
51+
description: 'required: Whether to enable co-editing for the note',
52+
},
53+
draft: {
54+
type: 'boolean',
55+
description: 'Whether to create the note as a draft',
56+
},
57+
},
58+
required: ['title', 'content'],
59+
},
60+
},
61+
handler: async ({ title, content, groupIds, folders, authorId, coediting, draft }) => {
62+
if (!title || !content || !groupIds || !coediting) {
63+
throw new Error('Title, content, groupIds, and coediting are required')
64+
}
65+
66+
const response = await createNote({
67+
input: {
68+
clientMutationId: uuid(),
69+
title,
70+
content,
71+
groupIds,
72+
folders,
73+
authorId,
74+
coediting,
75+
draft,
76+
},
77+
})
78+
79+
return {
80+
content: [
81+
{
82+
type: 'text',
83+
text: JSON.stringify(response.createNote, null, 2),
84+
},
85+
],
86+
}
87+
},
88+
}

src/tools/index.ts

+4
Original file line numberDiff line numberDiff line change
@@ -3,12 +3,16 @@ import { getMyNotesTool } from './getMyNotes'
33
import { getNoteContentTool } from './getNoteContent'
44
import { getNoteFromPathTool } from './getNoteFromPath'
55
import { ToolResponse } from './types'
6+
import { updateNoteContentTool } from './updateNoteContent'
7+
import { createNoteTool } from './createNote'
68

79
const toolDefinitions = {
810
kibela_search_notes: searchNotesTool,
911
kibela_get_my_notes: getMyNotesTool,
1012
kibela_get_note_content: getNoteContentTool,
1113
kibela_get_note_from_path: getNoteFromPathTool,
14+
kibela_update_note_content: updateNoteContentTool,
15+
kibela_create_note: createNoteTool,
1216
} as const
1317

1418
export type ToolName = keyof typeof toolDefinitions

src/tools/updateNoteContent.ts

+61
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,61 @@
1+
import { getNote } from '../graphql'
2+
import { updateNoteContent } from '../graphql/queries/updateNoteContent'
3+
import { uuid } from '../utils'
4+
import { ToolDefinition } from './types'
5+
6+
export type UpdateNoteContentArgs = {
7+
id: string
8+
content: string
9+
}
10+
11+
export const updateNoteContentTool: ToolDefinition<UpdateNoteContentArgs> = {
12+
tool: {
13+
name: 'kibela_update_note_content',
14+
description:
15+
'Update note content by note id. This tool allows you to modify the content of an existing Kibela note. Before updating, it fetches the current content of the note to ensure proper version control. Note that you need the note ID (not the note path) to use this tool.',
16+
inputSchema: {
17+
type: 'object',
18+
properties: {
19+
id: {
20+
type: 'string',
21+
description:
22+
'Note id - not note path (e.g. /notes/123). If you want to update note content by note path, please use kibela_get_note_from_path tool first and get note id from the response',
23+
},
24+
content: {
25+
type: 'string',
26+
description:
27+
'New content of the note in markdown format. The content will completely replace the existing note content. Make sure to include all necessary formatting, headers, and sections you want to preserve.',
28+
},
29+
},
30+
required: ['id', 'content'],
31+
},
32+
},
33+
handler: async ({ id, content }) => {
34+
if (!id || !content) {
35+
throw new Error('Note id and content are required')
36+
}
37+
38+
const noteRes = await getNote({ id })
39+
if (!noteRes.note) {
40+
throw new Error('Note not found')
41+
}
42+
43+
const response = await updateNoteContent({
44+
input: {
45+
clientMutationId: uuid(),
46+
id,
47+
newContent: content,
48+
baseContent: noteRes.note.content,
49+
},
50+
})
51+
52+
return {
53+
content: [
54+
{
55+
type: 'text',
56+
text: JSON.stringify(response.updateNoteContent, null, 2),
57+
},
58+
],
59+
}
60+
},
61+
}

src/utils.ts

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
export const uuid = (): string => {
2+
return crypto.randomUUID()
3+
}

0 commit comments

Comments
 (0)