File size: 6,998 Bytes
5856d68
3578090
506c5f2
5856d68
506c5f2
5856d68
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
506c5f2
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
5856d68
506c5f2
 
 
5856d68
506c5f2
 
5856d68
 
506c5f2
5856d68
506c5f2
5856d68
 
 
 
506c5f2
5856d68
 
 
 
 
 
 
506c5f2
5856d68
 
 
 
3578090
5856d68
 
 
 
 
506c5f2
5856d68
 
 
 
506c5f2
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
5856d68
 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
import { backendBaseUrl } from '@/lib/constants'
import { errorMessage } from '@/lib/utils'
import { useSettingsStore } from '@/stores/settings'

// Types
export type LightragNodeType = {
  id: string
  labels: string[]
  properties: Record<string, any>
}

export type LightragEdgeType = {
  id: string
  source: string
  target: string
  type: string
  properties: Record<string, any>
}

export type LightragGraphType = {
  nodes: LightragNodeType[]
  edges: LightragEdgeType[]
}

export type LightragStatus = {
  status: 'healthy'
  working_directory: string
  input_directory: string
  indexed_files: string[]
  indexed_files_count: number
  configuration: {
    llm_binding: string
    llm_binding_host: string
    llm_model: string
    embedding_binding: string
    embedding_binding_host: string
    embedding_model: string
    max_tokens: number
    kv_storage: string
    doc_status_storage: string
    graph_storage: string
    vector_storage: string
  }
}

export type LightragDocumentsScanProgress = {
  is_scanning: boolean
  current_file: string
  indexed_count: number
  total_files: number
  progress: number
}

export type QueryMode = 'naive' | 'local' | 'global' | 'hybrid' | 'mix'

export type QueryRequest = {
  query: string
  mode: QueryMode
  stream?: boolean
  only_need_context?: boolean
}

export type QueryResponse = {
  response: string
}

export const InvalidApiKeyError = 'Invalid API Key'
export const RequireApiKeError = 'API Key required'

// Helper functions
const getResponseContent = async (response: Response) => {
  const contentType = response.headers.get('content-type')
  if (contentType) {
    if (contentType.includes('application/json')) {
      const data = await response.json()
      return JSON.stringify(data, undefined, 2)
    } else if (contentType.startsWith('text/')) {
      return await response.text()
    } else if (contentType.includes('application/xml') || contentType.includes('text/xml')) {
      return await response.text()
    } else if (contentType.includes('application/octet-stream')) {
      const buffer = await response.arrayBuffer()
      const decoder = new TextDecoder('utf-8', { fatal: false, ignoreBOM: true })
      return decoder.decode(buffer)
    } else {
      try {
        return await response.text()
      } catch (error) {
        console.warn('Failed to decode as text, may be binary:', error)
        return `[Could not decode response body. Content-Type: ${contentType}]`
      }
    }
  } else {
    try {
      return await response.text()
    } catch (error) {
      console.warn('Failed to decode as text, may be binary:', error)
      return '[Could not decode response body. No Content-Type header.]'
    }
  }
  return ''
}

const fetchWithAuth = async (url: string, options: RequestInit = {}): Promise<Response> => {
  const apiKey = useSettingsStore.getState().apiKey
  const headers = {
    ...(options.headers || {}),
    ...(apiKey ? { 'X-API-Key': apiKey } : {})
  }

  const response = await fetch(backendBaseUrl + url, {
    ...options,
    headers
  })

  if (!response.ok) {
    throw new Error(
      `${response.status} ${response.statusText}\n${await getResponseContent(response)}\n${response.url}`
    )
  }

  return response
}

// API methods
export const queryGraphs = async (label: string): Promise<LightragGraphType> => {
  const response = await fetchWithAuth(`/graphs?label=${label}`)
  return await response.json()
}

export const getGraphLabels = async (): Promise<string[]> => {
  const response = await fetchWithAuth('/graph/label/list')
  return await response.json()
}

export const checkHealth = async (): Promise<
  LightragStatus | { status: 'error'; message: string }
> => {
  try {
    const response = await fetchWithAuth('/health')
    return await response.json()
  } catch (e) {
    return {
      status: 'error',
      message: errorMessage(e)
    }
  }
}

export const getDocuments = async (): Promise<string[]> => {
  const response = await fetchWithAuth('/documents')
  return await response.json()
}

export const getDocumentsScanProgress = async (): Promise<LightragDocumentsScanProgress> => {
  const response = await fetchWithAuth('/documents/scan-progress')
  return await response.json()
}

export const uploadDocument = async (
  file: File
): Promise<{
  status: string
  message: string
  total_documents: number
}> => {
  const formData = new FormData()
  formData.append('file', file)

  const response = await fetchWithAuth('/documents/upload', {
    method: 'POST',
    body: formData
  })
  return await response.json()
}

export const startDocumentScan = async (): Promise<{ status: string }> => {
  const response = await fetchWithAuth('/documents/scan', {
    method: 'POST'
  })
  return await response.json()
}

export const queryText = async (request: QueryRequest): Promise<QueryResponse> => {
  const response = await fetchWithAuth('/query', {
    method: 'POST',
    headers: {
      'Content-Type': 'application/json'
    },
    body: JSON.stringify(request)
  })
  return await response.json()
}

export const queryTextStream = async (request: QueryRequest, onChunk: (chunk: string) => void) => {
  const response = await fetchWithAuth('/query/stream', {
    method: 'POST',
    headers: {
      'Content-Type': 'application/json'
    },
    body: JSON.stringify(request)
  })

  const reader = response.body?.getReader()
  if (!reader) throw new Error('No response body')

  const decoder = new TextDecoder()
  while (true) {
    const { done, value } = await reader.read()
    if (done) break

    const chunk = decoder.decode(value)
    const lines = chunk.split('\n')
    for (const line of lines) {
      if (line) {
        try {
          const data = JSON.parse(line)
          if (data.response) {
            onChunk(data.response)
          }
        } catch (e) {
          console.error('Error parsing stream chunk:', e)
        }
      }
    }
  }
}

// Text insertion API
export const insertText = async (
  text: string,
  description?: string
): Promise<{
  status: string
  message: string
  document_count: number
}> => {
  const response = await fetchWithAuth('/documents/text', {
    method: 'POST',
    headers: {
      'Content-Type': 'application/json'
    },
    body: JSON.stringify({ text, description })
  })
  return await response.json()
}

// Batch file upload API
export const uploadBatchDocuments = async (
  files: File[]
): Promise<{
  status: string
  message: string
  document_count: number
}> => {
  const formData = new FormData()
  files.forEach((file) => {
    formData.append('files', file)
  })

  const response = await fetchWithAuth('/documents/batch', {
    method: 'POST',
    body: formData
  })
  return await response.json()
}

// Clear all documents API
export const clearDocuments = async (): Promise<{
  status: string
  message: string
  document_count: number
}> => {
  const response = await fetchWithAuth('/documents', {
    method: 'DELETE'
  })
  return await response.json()
}