WebSearch.svelte 9.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319
  1. <script lang="ts">
  2. import { getRAGConfig, updateRAGConfig } from '$lib/apis/retrieval';
  3. import Switch from '$lib/components/common/Switch.svelte';
  4. import { models } from '$lib/stores';
  5. import { onMount, getContext } from 'svelte';
  6. import { toast } from 'svelte-sonner';
  7. import SensitiveInput from '$lib/components/common/SensitiveInput.svelte';
  8. const i18n = getContext('i18n');
  9. export let saveHandler: Function;
  10. let webConfig = null;
  11. let webSearchEngines = [
  12. 'searxng',
  13. 'google_pse',
  14. 'brave',
  15. 'serpstack',
  16. 'serper',
  17. 'serply',
  18. 'searchapi',
  19. 'duckduckgo',
  20. 'tavily',
  21. 'jina'
  22. ];
  23. let youtubeLanguage = 'en';
  24. let youtubeTranslation = null;
  25. const submitHandler = async () => {
  26. const res = await updateRAGConfig(localStorage.token, {
  27. web: webConfig,
  28. youtube: {
  29. language: youtubeLanguage.split(',').map((lang) => lang.trim()),
  30. translation: youtubeTranslation
  31. }
  32. });
  33. };
  34. onMount(async () => {
  35. const res = await getRAGConfig(localStorage.token);
  36. if (res) {
  37. webConfig = res.web;
  38. youtubeLanguage = res.youtube.language.join(',');
  39. youtubeTranslation = res.youtube.translation;
  40. }
  41. });
  42. </script>
  43. <form
  44. class="flex flex-col h-full justify-between space-y-3 text-sm"
  45. on:submit|preventDefault={async () => {
  46. await submitHandler();
  47. saveHandler();
  48. }}
  49. >
  50. <div class=" space-y-3 overflow-y-scroll scrollbar-hidden h-full">
  51. {#if webConfig}
  52. <div>
  53. <div class=" mb-1 text-sm font-medium">
  54. {$i18n.t('Web Search')}
  55. </div>
  56. <div>
  57. <div class=" py-0.5 flex w-full justify-between">
  58. <div class=" self-center text-xs font-medium">
  59. {$i18n.t('Enable Web Search')}
  60. </div>
  61. <Switch bind:state={webConfig.search.enabled} />
  62. </div>
  63. </div>
  64. <div class=" py-0.5 flex w-full justify-between">
  65. <div class=" self-center text-xs font-medium">{$i18n.t('Web Search Engine')}</div>
  66. <div class="flex items-center relative">
  67. <select
  68. class="dark:bg-gray-900 w-fit pr-8 rounded px-2 p-1 text-xs bg-transparent outline-none text-right"
  69. bind:value={webConfig.search.engine}
  70. placeholder={$i18n.t('Select a engine')}
  71. required
  72. >
  73. <option disabled selected value="">{$i18n.t('Select a engine')}</option>
  74. {#each webSearchEngines as engine}
  75. <option value={engine}>{engine}</option>
  76. {/each}
  77. </select>
  78. </div>
  79. </div>
  80. {#if webConfig.search.engine !== ''}
  81. <div class="mt-1.5">
  82. {#if webConfig.search.engine === 'searxng'}
  83. <div>
  84. <div class=" self-center text-xs font-medium mb-1">
  85. {$i18n.t('Searxng Query URL')}
  86. </div>
  87. <div class="flex w-full">
  88. <div class="flex-1">
  89. <input
  90. class="w-full rounded-lg py-2 px-4 text-sm bg-gray-50 dark:text-gray-300 dark:bg-gray-850 outline-none"
  91. type="text"
  92. placeholder={$i18n.t('Enter Searxng Query URL')}
  93. bind:value={webConfig.search.searxng_query_url}
  94. autocomplete="off"
  95. />
  96. </div>
  97. </div>
  98. </div>
  99. {:else if webConfig.search.engine === 'google_pse'}
  100. <div>
  101. <div class=" self-center text-xs font-medium mb-1">
  102. {$i18n.t('Google PSE API Key')}
  103. </div>
  104. <SensitiveInput
  105. placeholder={$i18n.t('Enter Google PSE API Key')}
  106. bind:value={webConfig.search.google_pse_api_key}
  107. />
  108. </div>
  109. <div class="mt-1.5">
  110. <div class=" self-center text-xs font-medium mb-1">
  111. {$i18n.t('Google PSE Engine Id')}
  112. </div>
  113. <div class="flex w-full">
  114. <div class="flex-1">
  115. <input
  116. class="w-full rounded-lg py-2 px-4 text-sm bg-gray-50 dark:text-gray-300 dark:bg-gray-850 outline-none"
  117. type="text"
  118. placeholder={$i18n.t('Enter Google PSE Engine Id')}
  119. bind:value={webConfig.search.google_pse_engine_id}
  120. autocomplete="off"
  121. />
  122. </div>
  123. </div>
  124. </div>
  125. {:else if webConfig.search.engine === 'brave'}
  126. <div>
  127. <div class=" self-center text-xs font-medium mb-1">
  128. {$i18n.t('Brave Search API Key')}
  129. </div>
  130. <SensitiveInput
  131. placeholder={$i18n.t('Enter Brave Search API Key')}
  132. bind:value={webConfig.search.brave_search_api_key}
  133. />
  134. </div>
  135. {:else if webConfig.search.engine === 'serpstack'}
  136. <div>
  137. <div class=" self-center text-xs font-medium mb-1">
  138. {$i18n.t('Serpstack API Key')}
  139. </div>
  140. <SensitiveInput
  141. placeholder={$i18n.t('Enter Serpstack API Key')}
  142. bind:value={webConfig.search.serpstack_api_key}
  143. />
  144. </div>
  145. {:else if webConfig.search.engine === 'serper'}
  146. <div>
  147. <div class=" self-center text-xs font-medium mb-1">
  148. {$i18n.t('Serper API Key')}
  149. </div>
  150. <SensitiveInput
  151. placeholder={$i18n.t('Enter Serper API Key')}
  152. bind:value={webConfig.search.serper_api_key}
  153. />
  154. </div>
  155. {:else if webConfig.search.engine === 'serply'}
  156. <div>
  157. <div class=" self-center text-xs font-medium mb-1">
  158. {$i18n.t('Serply API Key')}
  159. </div>
  160. <SensitiveInput
  161. placeholder={$i18n.t('Enter Serply API Key')}
  162. bind:value={webConfig.search.serply_api_key}
  163. />
  164. </div>
  165. {:else if webConfig.search.engine === 'searchapi'}
  166. <div>
  167. <div class=" self-center text-xs font-medium mb-1">
  168. {$i18n.t('SearchApi API Key')}
  169. </div>
  170. <SensitiveInput
  171. placeholder={$i18n.t('Enter SearchApi API Key')}
  172. bind:value={webConfig.search.searchapi_api_key}
  173. />
  174. </div>
  175. <div class="mt-1.5">
  176. <div class=" self-center text-xs font-medium mb-1">
  177. {$i18n.t('SearchApi Engine')}
  178. </div>
  179. <div class="flex w-full">
  180. <div class="flex-1">
  181. <input
  182. class="w-full rounded-lg py-2 px-4 text-sm bg-gray-50 dark:text-gray-300 dark:bg-gray-850 outline-none"
  183. type="text"
  184. placeholder={$i18n.t('Enter SearchApi Engine')}
  185. bind:value={webConfig.search.searchapi_engine}
  186. autocomplete="off"
  187. />
  188. </div>
  189. </div>
  190. </div>
  191. {:else if webConfig.search.engine === 'tavily'}
  192. <div>
  193. <div class=" self-center text-xs font-medium mb-1">
  194. {$i18n.t('Tavily API Key')}
  195. </div>
  196. <SensitiveInput
  197. placeholder={$i18n.t('Enter Tavily API Key')}
  198. bind:value={webConfig.search.tavily_api_key}
  199. />
  200. </div>
  201. {/if}
  202. </div>
  203. {/if}
  204. {#if webConfig.search.enabled}
  205. <div class="mt-2 flex gap-2 mb-1">
  206. <div class="w-full">
  207. <div class=" self-center text-xs font-medium mb-1">
  208. {$i18n.t('Search Result Count')}
  209. </div>
  210. <input
  211. class="w-full rounded-lg py-2 px-4 text-sm bg-gray-50 dark:text-gray-300 dark:bg-gray-850 outline-none"
  212. placeholder={$i18n.t('Search Result Count')}
  213. bind:value={webConfig.search.result_count}
  214. required
  215. />
  216. </div>
  217. <div class="w-full">
  218. <div class=" self-center text-xs font-medium mb-1">
  219. {$i18n.t('Concurrent Requests')}
  220. </div>
  221. <input
  222. class="w-full rounded-lg py-2 px-4 text-sm bg-gray-50 dark:text-gray-300 dark:bg-gray-850 outline-none"
  223. placeholder={$i18n.t('Concurrent Requests')}
  224. bind:value={webConfig.search.concurrent_requests}
  225. required
  226. />
  227. </div>
  228. </div>
  229. {/if}
  230. </div>
  231. <hr class=" dark:border-gray-850 my-2" />
  232. <div>
  233. <div class=" mb-1 text-sm font-medium">
  234. {$i18n.t('Web Loader Settings')}
  235. </div>
  236. <div>
  237. <div class=" py-0.5 flex w-full justify-between">
  238. <div class=" self-center text-xs font-medium">
  239. {$i18n.t('Bypass SSL verification for Websites')}
  240. </div>
  241. <button
  242. class="p-1 px-3 text-xs flex rounded transition"
  243. on:click={() => {
  244. webConfig.ssl_verification = !webConfig.ssl_verification;
  245. submitHandler();
  246. }}
  247. type="button"
  248. >
  249. {#if webConfig.ssl_verification === true}
  250. <span class="ml-2 self-center">{$i18n.t('On')}</span>
  251. {:else}
  252. <span class="ml-2 self-center">{$i18n.t('Off')}</span>
  253. {/if}
  254. </button>
  255. </div>
  256. </div>
  257. <div class=" mt-2 mb-1 text-sm font-medium">
  258. {$i18n.t('Youtube Loader Settings')}
  259. </div>
  260. <div>
  261. <div class=" py-0.5 flex w-full justify-between">
  262. <div class=" w-20 text-xs font-medium self-center">{$i18n.t('Language')}</div>
  263. <div class=" flex-1 self-center">
  264. <input
  265. class="w-full rounded-lg py-2 px-4 text-sm bg-gray-50 dark:text-gray-300 dark:bg-gray-850 outline-none"
  266. type="text"
  267. placeholder={$i18n.t('Enter language codes')}
  268. bind:value={youtubeLanguage}
  269. autocomplete="off"
  270. />
  271. </div>
  272. </div>
  273. </div>
  274. </div>
  275. {/if}
  276. </div>
  277. <div class="flex justify-end pt-3 text-sm font-medium">
  278. <button
  279. class=" px-4 py-2 bg-emerald-700 hover:bg-emerald-800 text-gray-100 transition rounded-lg"
  280. type="submit"
  281. >
  282. {$i18n.t('Save')}
  283. </button>
  284. </div>
  285. </form>