Skip to content

Commit 36a7241

Browse files
committed
feat: create note
1 parent 0690abe commit 36a7241

File tree

6 files changed

+147
-5
lines changed

6 files changed

+147
-5
lines changed

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

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+
}

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

+2
Original file line numberDiff line numberDiff line change
@@ -4,13 +4,15 @@ import { getNoteContentTool } from './getNoteContent'
44
import { getNoteFromPathTool } from './getNoteFromPath'
55
import { ToolResponse } from './types'
66
import { updateNoteContentTool } from './updateNoteContent'
7+
import { createNoteTool } from './createNote'
78

89
const toolDefinitions = {
910
kibela_search_notes: searchNotesTool,
1011
kibela_get_my_notes: getMyNotesTool,
1112
kibela_get_note_content: getNoteContentTool,
1213
kibela_get_note_from_path: getNoteFromPathTool,
1314
kibela_update_note_content: updateNoteContentTool,
15+
kibela_create_note: createNoteTool,
1416
} as const
1517

1618
export type ToolName = keyof typeof toolDefinitions

src/tools/updateNoteContent.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
import { getNote } from '../graphql'
22
import { updateNoteContent } from '../graphql/queries/updateNoteContent'
3+
import { uuid } from '../utils'
34
import { ToolDefinition } from './types'
45

56
export type UpdateNoteContentArgs = {
@@ -41,7 +42,7 @@ export const updateNoteContentTool: ToolDefinition<UpdateNoteContentArgs> = {
4142

4243
const response = await updateNoteContent({
4344
input: {
44-
clientMutationId: 'updateNoteContent',
45+
clientMutationId: uuid(),
4546
id,
4647
newContent: content,
4748
baseContent: noteRes.note.content,

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)