diff --git a/crawler/crawler.go b/crawler/crawler.go index 6d4db79..f987767 100644 --- a/crawler/crawler.go +++ b/crawler/crawler.go @@ -69,6 +69,9 @@ type Crawler struct { // 运行时活跃线程计数(atomic,每轮 epoch 自动归零前重新开始计数) activeWorkers int64 + // 本轮发现的新链接计数(atomic,供前端实时监控) + newLinksCount int64 + // ---- Priority Worker(独立 goroutine,不受主 workers 限制)---- priorityCh chan string // Priority URL 任务队列(用户手动添加) priorityChildCh chan string // Priority 子链接队列(子 URL 继续由 priority worker 爬取) @@ -93,7 +96,8 @@ type CrawlStatus struct { QueueLength int `json:"queue_length"` // 本轮队列长度 CompletedCount int `json:"completed_count"` // 本轮已完成的 URL 数 VisitedTotal int `json:"visited_total"` // 已收录 URL 总数 - NextPoolSize int `json:"next_pool_size"` // 下一轮链接池大小(newLinks 调度后的队列长度) + NewLinksCount int64 `json:"new_links_count"` // 本轮已发现的新链接数(实时更新) + NextPoolSize int `json:"next_pool_size"` // 下一轮链接池大小(调度后的队列长度) IsRunning bool `json:"is_running"` // 是否正在运行 } @@ -457,11 +461,15 @@ func (c *Crawler) Run(entryURL string, maxEpoch int) { // 每轮 epoch 从 config 读取最新 workers 值,支持运行时动态调整 workers := config.CrawlerWorkers() + // 重置新链接计数器 + atomic.StoreInt64(&c.newLinksCount, 0) + // 更新爬取状态:新一轮开始 c.updateCrawlStatus(func(cs *CrawlStatus) { cs.CurrentEpoch = ep + 1 cs.QueueLength = len(queue) cs.CompletedCount = 0 + cs.NewLinksCount = 0 }) // 每轮开始前:拉取 priority URLs,插入队列前端 @@ -526,11 +534,16 @@ func (c *Crawler) Run(entryURL string, maxEpoch int) { // 分配权重 w := 1.0 / float64(n) - // 子 URL 进入 newLinks 调度,由普通 BFS 下一轮处理。 + // 子 URL 进入 newLinks 调度,由普通 BFS 下一轮处理。 // (priority worker 通过 priorityChildCh 独立爬取子 URL,两者互不干扰) mu.Lock() for _, h := range children { newLinks = append(newLinks, URLWeight{URL: h, Weight: w}) + // 实时自增计数(供前端监控) + atomic.AddInt64(&c.newLinksCount, 1) + c.updateCrawlStatus(func(cs *CrawlStatus) { + cs.NewLinksCount = atomic.LoadInt64(&c.newLinksCount) + }) } mu.Unlock() }(u) @@ -545,6 +558,10 @@ func (c *Crawler) Run(entryURL string, maxEpoch int) { select { case gc := <-c.normalChildCh: newLinks = append(newLinks, gc) + atomic.AddInt64(&c.newLinksCount, 1) // 孙链接也要计入 + c.updateCrawlStatus(func(cs *CrawlStatus) { + cs.NewLinksCount = atomic.LoadInt64(&c.newLinksCount) + }) case <-timeout: drained = true } diff --git a/dist/assets/index-yf_Ps55i.js b/dist/assets/index-G5PISGmH.js similarity index 82% rename from dist/assets/index-yf_Ps55i.js rename to dist/assets/index-G5PISGmH.js index cfcc44a..a070877 100644 --- a/dist/assets/index-yf_Ps55i.js +++ b/dist/assets/index-G5PISGmH.js @@ -4,4 +4,4 @@ var e=Object.defineProperty,t=(t,n)=>{let r={};for(var i in t)e(r,i,{get:t[i],en `+e.map(Yc).join(` `):` `+Yc(e[0]):`as no adapter specified`),`ERR_NOT_SUPPORT`)}return i}var Qc={getAdapter:Zc,adapters:Jc};function $c(e){if(e.cancelToken&&e.cancelToken.throwIfRequested(),e.signal&&e.signal.aborted)throw new _c(null,e)}function el(e){return $c(e),e.headers=Z.from(e.headers),e.data=hc.call(e,e.transformRequest),[`post`,`put`,`patch`].indexOf(e.method)!==-1&&e.headers.setContentType(`application/x-www-form-urlencoded`,!1),Qc.getAdapter(e.adapter||ic.adapter,e)(e).then(function(t){return $c(e),t.data=hc.call(e,e.transformResponse,t),t.headers=Z.from(t.headers),t},function(t){return gc(t)||($c(e),t&&t.response&&(t.response.data=hc.call(e,e.transformResponse,t.response),t.response.headers=Z.from(t.response.headers))),Promise.reject(t)})}var tl=`1.14.0`,nl={};[`object`,`boolean`,`number`,`function`,`string`,`symbol`].forEach((e,t)=>{nl[e]=function(n){return typeof n===e||`a`+(t<1?`n `:` `)+e}});var rl={};nl.transitional=function(e,t,n){function r(e,t){return`[Axios v`+tl+`] Transitional option '`+e+`'`+t+(n?`. `+n:``)}return(n,i,a)=>{if(e===!1)throw new Y(r(i,` has been removed`+(t?` in `+t:``)),Y.ERR_DEPRECATED);return t&&!rl[i]&&(rl[i]=!0,console.warn(r(i,` has been deprecated since v`+t+` and will be removed in the near future`))),e?e(n,i,a):!0}},nl.spelling=function(e){return(t,n)=>(console.warn(`${n} is likely a misspelling of ${e}`),!0)};function il(e,t,n){if(typeof e!=`object`)throw new Y(`options must be an object`,Y.ERR_BAD_OPTION_VALUE);let r=Object.keys(e),i=r.length;for(;i-- >0;){let a=r[i],o=t[a];if(o){let t=e[a],n=t===void 0||o(t,a,e);if(n!==!0)throw new Y(`option `+a+` must be `+n,Y.ERR_BAD_OPTION_VALUE);continue}if(n!==!0)throw new Y(`Unknown option `+a,Y.ERR_BAD_OPTION)}}var al={assertOptions:il,validators:nl},ol=al.validators,sl=class{constructor(e){this.defaults=e||{},this.interceptors={request:new Ws,response:new Ws}}async request(e,t){try{return await this._request(e,t)}catch(e){if(e instanceof Error){let t={};Error.captureStackTrace?Error.captureStackTrace(t):t=Error();let n=t.stack?t.stack.replace(/^.+\n/,``):``;try{e.stack?n&&!String(e.stack).endsWith(n.replace(/^.+\n.+\n/,``))&&(e.stack+=` `+n):e.stack=n}catch{}}throw e}}_request(e,t){typeof e==`string`?(t||={},t.url=e):t=e||{},t=jc(this.defaults,t);let{transitional:n,paramsSerializer:r,headers:i}=t;n!==void 0&&al.assertOptions(n,{silentJSONParsing:ol.transitional(ol.boolean),forcedJSONParsing:ol.transitional(ol.boolean),clarifyTimeoutError:ol.transitional(ol.boolean),legacyInterceptorReqResOrdering:ol.transitional(ol.boolean)},!1),r!=null&&(J.isFunction(r)?t.paramsSerializer={serialize:r}:al.assertOptions(r,{encode:ol.function,serialize:ol.function},!0)),t.allowAbsoluteUrls!==void 0||(this.defaults.allowAbsoluteUrls===void 0?t.allowAbsoluteUrls=!0:t.allowAbsoluteUrls=this.defaults.allowAbsoluteUrls),al.assertOptions(t,{baseUrl:ol.spelling(`baseURL`),withXsrfToken:ol.spelling(`withXSRFToken`)},!0),t.method=(t.method||this.defaults.method||`get`).toLowerCase();let a=i&&J.merge(i.common,i[t.method]);i&&J.forEach([`delete`,`get`,`head`,`post`,`put`,`patch`,`common`],e=>{delete i[e]}),t.headers=Z.concat(a,i);let o=[],s=!0;this.interceptors.request.forEach(function(e){if(typeof e.runWhen==`function`&&e.runWhen(t)===!1)return;s&&=e.synchronous;let n=t.transitional||Gs;n&&n.legacyInterceptorReqResOrdering?o.unshift(e.fulfilled,e.rejected):o.push(e.fulfilled,e.rejected)});let c=[];this.interceptors.response.forEach(function(e){c.push(e.fulfilled,e.rejected)});let l,u=0,d;if(!s){let e=[el.bind(this),void 0];for(e.unshift(...o),e.push(...c),d=e.length,l=Promise.resolve(t);u{if(!n._listeners)return;let t=n._listeners.length;for(;t-- >0;)n._listeners[t](e);n._listeners=null}),this.promise.then=e=>{let t,r=new Promise(e=>{n.subscribe(e),t=e}).then(e);return r.cancel=function(){n.unsubscribe(t)},r},e(function(e,r,i){n.reason||(n.reason=new _c(e,r,i),t(n.reason))})}throwIfRequested(){if(this.reason)throw this.reason}subscribe(e){if(this.reason){e(this.reason);return}this._listeners?this._listeners.push(e):this._listeners=[e]}unsubscribe(e){if(!this._listeners)return;let t=this._listeners.indexOf(e);t!==-1&&this._listeners.splice(t,1)}toAbortSignal(){let e=new AbortController,t=t=>{e.abort(t)};return this.subscribe(t),e.signal.unsubscribe=()=>this.unsubscribe(t),e.signal}static source(){let t;return{token:new e(function(e){t=e}),cancel:t}}};function ll(e){return function(t){return e.apply(null,t)}}function ul(e){return J.isObject(e)&&e.isAxiosError===!0}var dl={Continue:100,SwitchingProtocols:101,Processing:102,EarlyHints:103,Ok:200,Created:201,Accepted:202,NonAuthoritativeInformation:203,NoContent:204,ResetContent:205,PartialContent:206,MultiStatus:207,AlreadyReported:208,ImUsed:226,MultipleChoices:300,MovedPermanently:301,Found:302,SeeOther:303,NotModified:304,UseProxy:305,Unused:306,TemporaryRedirect:307,PermanentRedirect:308,BadRequest:400,Unauthorized:401,PaymentRequired:402,Forbidden:403,NotFound:404,MethodNotAllowed:405,NotAcceptable:406,ProxyAuthenticationRequired:407,RequestTimeout:408,Conflict:409,Gone:410,LengthRequired:411,PreconditionFailed:412,PayloadTooLarge:413,UriTooLong:414,UnsupportedMediaType:415,RangeNotSatisfiable:416,ExpectationFailed:417,ImATeapot:418,MisdirectedRequest:421,UnprocessableEntity:422,Locked:423,FailedDependency:424,TooEarly:425,UpgradeRequired:426,PreconditionRequired:428,TooManyRequests:429,RequestHeaderFieldsTooLarge:431,UnavailableForLegalReasons:451,InternalServerError:500,NotImplemented:501,BadGateway:502,ServiceUnavailable:503,GatewayTimeout:504,HttpVersionNotSupported:505,VariantAlsoNegotiates:506,InsufficientStorage:507,LoopDetected:508,NotExtended:510,NetworkAuthenticationRequired:511,WebServerIsDown:521,ConnectionTimedOut:522,OriginIsUnreachable:523,TimeoutOccurred:524,SslHandshakeFailed:525,InvalidSslCertificate:526};Object.entries(dl).forEach(([e,t])=>{dl[t]=e});function fl(e){let t=new sl(e),n=Co(sl.prototype.request,t);return J.extend(n,sl.prototype,t,{allOwnKeys:!0}),J.extend(n,t,null,{allOwnKeys:!0}),n.create=function(t){return fl(jc(e,t))},n}var Q=fl(ic);Q.Axios=sl,Q.CanceledError=_c,Q.CancelToken=cl,Q.isCancel=gc,Q.VERSION=tl,Q.toFormData=Rs,Q.AxiosError=Y,Q.Cancel=Q.CanceledError,Q.all=function(e){return Promise.all(e)},Q.spread=ll,Q.isAxiosError=ul,Q.mergeConfig=jc,Q.AxiosHeaders=Z,Q.formToJSON=e=>nc(J.isHTMLForm(e)?new FormData(e):e),Q.getAdapter=Qc.getAdapter,Q.HttpStatusCode=dl,Q.default=Q;var $=``;async function pl(e=50){let{data:t}=await Q.get(`${$}/admin/recent`,{params:{limit:e},timeout:15e3});return t}async function ml(){let{data:e}=await Q.get(`${$}/admin/stats`,{timeout:15e3});return e}async function hl(e){let{data:t}=await Q.post(`${$}/admin/priority`,{url:e},{timeout:15e3});return t}async function gl(){let{data:e}=await Q.post(`${$}/admin/flush`,null,{timeout:6e4});return e}async function _l(){let{data:e}=await Q.get(`${$}/admin/flush/status`,{timeout:5e3});return e}async function vl(){let{data:e}=await Q.get(`${$}/admin/workers`,{timeout:1e4});return e}async function yl(e){let{data:t}=await Q.post(`${$}/admin/workers`,{workers:e},{timeout:1e4});return t}async function bl(){let{data:e}=await Q.get(`${$}/admin/backlink`,{timeout:1e4});return e}async function xl(){let{data:e}=await Q.post(`${$}/admin/backlink`,null,{timeout:1e4});return e}async function Sl(){let{data:e}=await Q.get(`${$}/admin/priority/status`,{timeout:1e4});return e}async function Cl(){let{data:e}=await Q.get(`${$}/admin/crawl/status`,{timeout:1e4});return e}async function wl(e){let{data:t}=await Q.get(`${$}/admin/url/keywords`,{params:{url:e},timeout:5e3});return t}async function Tl(){let{data:e}=await Q.get(`${$}/admin/url/keywords/stats`,{timeout:5e3});return e}var El={class:`p-4 md:p-8`},Dl={key:0,class:`flex items-center justify-center h-48`},Ol={key:1,class:`bg-red-900/30 border border-red-800 rounded-lg p-4 text-red-300`},kl={class:`grid grid-cols-2 md:grid-cols-5 gap-3 md:gap-5 mb-6 md:mb-8`},Al={class:`bg-gray-900 border border-gray-800 rounded-xl p-5`},jl={class:`text-3xl font-bold text-white`},Ml={class:`bg-gray-900 border border-gray-800 rounded-xl p-5`},Nl={class:`text-3xl font-bold text-white`},Pl={class:`bg-gray-900 border border-gray-800 rounded-xl p-5`},Fl={class:`text-3xl font-bold text-white`},Il={class:`bg-gray-900 border border-gray-800 rounded-xl p-5`},Ll={class:`bg-gray-900 border border-gray-800 rounded-xl p-5 flex flex-col justify-between`},Rl=[`disabled`],zl={key:0,class:`bg-gray-900 border border-gray-800 rounded-xl p-4 md:p-5 mb-6 md:mb-8`},Bl={class:`grid grid-cols-2 md:grid-cols-4 gap-4`},Vl={class:`bg-gray-800/50 rounded-lg p-4 text-center`},Hl={class:`bg-gray-800/50 rounded-lg p-4 text-center`},Ul={class:`bg-gray-800/50 rounded-lg p-4 text-center`},Wl={class:`bg-gray-800/50 rounded-lg p-4 text-center`},Gl={class:`text-2xl font-bold text-blue-400`},Kl={class:`mt-4 flex gap-2`},ql=[`disabled`],Jl={key:1,class:`bg-gray-900 border border-gray-800 rounded-xl p-4 md:p-5 mb-6 md:mb-8`},Yl={class:`flex flex-col md:flex-row md:items-center md:justify-between gap-4 mb-4`},Xl={class:`flex items-center gap-2`},Zl={class:`grid grid-cols-2 md:grid-cols-5 gap-4`},Ql={class:`text-center`},$l={class:`text-3xl font-bold text-white`},eu={class:`text-lg text-gray-500`},tu={class:`text-center`},nu={class:`text-center`},ru={class:`text-3xl font-bold text-green-400`},iu={class:`text-xs text-gray-600 mt-1`},au={class:`text-center`},ou={class:`text-center`},su={class:`text-3xl font-bold text-white`},cu={key:0,class:`mt-4`},lu={class:`bg-gray-800 rounded-full h-2 overflow-hidden`},uu={class:`bg-gray-900 border border-gray-800 rounded-xl p-4 md:p-5 mb-6 md:mb-8`},du={class:`flex flex-col md:flex-row md:items-center md:justify-between gap-4`},fu={class:`flex items-baseline gap-4`},pu={class:`text-3xl font-bold text-white`},mu={class:`flex items-center gap-2`},hu=[`disabled`],gu=[`disabled`],_u=[`disabled`],vu={key:2,class:`bg-gray-900 border border-gray-800 rounded-xl p-4 md:p-5 mb-6 md:mb-8`},yu={class:`flex flex-col md:flex-row md:items-center md:justify-between gap-4`},bu={class:`flex-1`},xu={class:`grid grid-cols-1 sm:grid-cols-2 gap-3`},Su={class:`text-lg font-bold text-white`},Cu=[`title`],wu=[`disabled`],Tu={class:`grid grid-cols-1 md:grid-cols-2 gap-4 md:gap-5`},Eu={class:`bg-gray-900 border border-gray-800 rounded-xl p-4 md:p-5`},Du={class:`space-y-2`},Ou=[`title`],ku={class:`flex-1 bg-gray-800 rounded-full h-4 md:h-5 overflow-hidden`},Au={class:`w-12 md:w-16 text-xs text-gray-500 text-right shrink-0`},ju={class:`bg-gray-900 border border-gray-800 rounded-xl p-4 md:p-5`},Mu={class:`space-y-2`},Nu={class:`w-8 md:w-10 text-xs text-gray-400 shrink-0 font-mono`},Pu={class:`flex-1 bg-gray-800 rounded-full h-4 md:h-5 overflow-hidden`},Fu={class:`w-12 md:w-16 text-xs text-gray-500 text-right shrink-0`},Iu={__name:`Dashboard`,setup(e){let t=F(null),n=F(!0),r=F(!1),i=F(null),a=null,o=F(0),s=F(0),c=F(0),l=F(!1),u=F(null),d=F(``),f=F(!1),p=F(null);ir(async()=>{await Promise.all([m(),h(),_(),v(),y()]),a=setInterval(()=>{m(),h(),_(),v(),y()},5e3)}),cr(()=>{a&&clearInterval(a)});async function m(){try{t.value=await ml(),i.value=null}catch(e){i.value=`无法加载统计数据,可能人服务器未启动或端口不对`,console.error(e)}finally{n.value=!1}}async function h(){try{let e=await vl();o.value=e.configured,s.value=e.active,c.value=e.configured}catch(e){console.error(`Failed to load workers:`,e)}}async function g(){let e=parseInt(c.value,10);if(isNaN(e)||e<1||e>500){i.value=`线程数必须在 1~500 之间`;return}l.value=!0;try{await yl(e),o.value=e,i.value=null}catch(e){i.value=`修改线程数失败: `+e.message}finally{l.value=!1}}async function _(){try{backlinkStatus.value=await bl()}catch(e){console.error(`Failed to load backlink status:`,e)}}async function v(){try{u.value=await Sl()}catch(e){console.error(`Failed to load priority status:`,e)}}async function y(){try{p.value=await Cl()}catch(e){console.error(`Failed to load crawl status:`,e)}}async function b(){backlinkTriggering.value=!0;try{await xl(),backlinkStatus.value=await bl(),i.value=null}catch(e){i.value=`触发反链计算失败: `+e.message}finally{backlinkTriggering.value=!1}}function x(e){if(!e)return`--`;let t=new Date(e),n=t-new Date,r=Math.floor(n/6e4),i=Math.floor(r/60),a=Math.floor(i/24),o=t.toLocaleTimeString(`zh-CN`,{hour:`2-digit`,minute:`2-digit`}),s=t.toLocaleDateString(`zh-CN`,{month:`short`,day:`numeric`});return a>0?`${s} ${o} (${a}天后)`:i>0?`${s} ${o} (${i}小时后)`:r>0?`${o} (${r}分钟后)`:`${o} (即将执行)`}function S(e){return e?new Date(e).toLocaleString(`zh-CN`,{month:`short`,day:`numeric`,hour:`2-digit`,minute:`2-digit`}):`从未执行`}async function C(){r.value=!0;try{for(await gl();await new Promise(e=>setTimeout(e,1e3)),(await _l()).flushing;);t.value=await ml()}catch(e){i.value=`刷盘失败: `+e.message}finally{r.value=!1}}function w(e){return!e&&e!==0?`0`:Number(e).toLocaleString()}function ee(e){return e?Object.entries(e).sort((e,t)=>t[1]-e[1]).slice(0,10):[]}function te(e){let t={zh:`#e53e3e`,en:`#3182ce`,ja:`#e53e3e`,ko:`#3182ce`,fr:`#38a169`,de:`#d69e2e`,es:`#38a169`,ru:`#805ad5`,other:`#718096`};return t[e]||t.other}function ne(e){try{let t=new URL(e);return t.protocol===`http:`||t.protocol===`https:`}catch{return!1}}async function re(){let e=d.value.split(` -`).map(e=>e.trim()).filter(e=>e).filter(ne);if(e.length===0){i.value=`未检测到有效 URL`;return}f.value=!0;let t=0,n=0;try{for(let r of e)try{await hl(r),t++}catch{n++}d.value=``,await v(),i.value=null,n>0&&(i.value=`添加完成:${t} 成功,${n} 失败`)}finally{f.value=!1}}return(e,a)=>(H(),U(`div`,El,[a[31]||=W(`h1`,{class:`text-xl md:text-2xl font-semibold text-white mb-6 md:mb-8`},`概览`,-1),n.value?(H(),U(`div`,Dl,[...a[4]||=[W(`div`,{class:`text-gray-400 animate-pulse`},`加载中...`,-1)]])):i.value?(H(),U(`div`,Ol,O(i.value),1)):t.value?(H(),U(B,{key:2},[W(`div`,kl,[W(`div`,Al,[a[5]||=W(`div`,{class:`text-sm text-gray-500 mb-2`},`已爬取 URL`,-1),W(`div`,jl,O(w(t.value.total_urls)),1)]),W(`div`,Ml,[a[6]||=W(`div`,{class:`text-sm text-gray-500 mb-2`},`总词数`,-1),W(`div`,Nl,O(w(t.value.total_words)),1)]),W(`div`,Pl,[a[7]||=W(`div`,{class:`text-sm text-gray-500 mb-2`},`域名数量`,-1),W(`div`,Fl,O(w(t.value.total_domains)),1)]),W(`div`,Il,[a[8]||=W(`div`,{class:`text-sm text-gray-500 mb-2`},`待重爬`,-1),W(`div`,{class:D([`text-3xl font-bold`,t.value.recrawl_eligible>0?`text-orange-400`:`text-green-400`])},O(w(t.value.recrawl_eligible)),3)]),W(`div`,Ll,[W(`div`,null,[a[9]||=W(`div`,{class:`text-sm text-gray-500 mb-2`},`待刷盘`,-1),W(`div`,{class:D([`text-3xl font-bold`,t.value.pending>0?`text-yellow-400`:`text-green-400`])},O(w(t.value.pending)),3)]),W(`button`,{class:`mt-3 w-full bg-blue-700 hover:bg-blue-600 disabled:bg-gray-700 disabled:text-gray-500 text-white text-sm font-medium py-1.5 px-3 rounded transition-colors cursor-pointer`,disabled:r.value||!t.value.pending,onClick:C},O(r.value?`刷盘中...`:`立即刷盘`),9,Rl)])]),u.value?(H(),U(`div`,zl,[a[14]||=W(`h2`,{class:`text-sm font-semibold text-gray-300 uppercase tracking-wider mb-4`},`优先爬取队列`,-1),W(`div`,Bl,[W(`div`,Vl,[a[10]||=W(`div`,{class:`text-xs text-gray-500 mb-1`},`一级队列`,-1),W(`div`,{class:D([`text-2xl font-bold`,(u.value.level1||0)>0?`text-yellow-400`:`text-gray-500`])},O(w(u.value.level1||0)),3)]),W(`div`,Hl,[a[11]||=W(`div`,{class:`text-xs text-gray-500 mb-1`},`二级队列`,-1),W(`div`,{class:D([`text-2xl font-bold`,(u.value.level2_queue||0)+(u.value.level2_inflight||0)>0?`text-yellow-400`:`text-gray-500`])},O(w((u.value.level2_queue||0)+(u.value.level2_inflight||0))),3)]),W(`div`,Ul,[a[12]||=W(`div`,{class:`text-xs text-gray-500 mb-1`},`线程使用`,-1),W(`div`,{class:D([`text-2xl font-bold`,u.value.active>0?`text-orange-400`:`text-gray-500`])},O(w(u.value.active)),3)]),W(`div`,Wl,[a[13]||=W(`div`,{class:`text-xs text-gray-500 mb-1`},`线程总数`,-1),W(`div`,Gl,O(u.value.max_workers),1)])]),W(`div`,Kl,[An(W(`textarea`,{"onUpdate:modelValue":a[0]||=e=>d.value=e,rows:`3`,placeholder:`批量添加 URL(每行一个,仅发送有效链接)`,class:`flex-1 bg-gray-800 border border-gray-700 text-gray-200 text-sm rounded-lg px-3 py-2 resize-none focus:outline-none focus:border-blue-500 placeholder-gray-600`},null,512),[[so,d.value]]),W(`button`,{class:`bg-blue-700 hover:bg-blue-600 disabled:bg-gray-700 text-white text-sm font-medium px-4 py-2 rounded-lg transition-colors cursor-pointer whitespace-nowrap self-end`,disabled:f.value||!d.value.trim(),onClick:re},O(f.value?`添加中...`:`批量添加`),9,ql)])])):G(``,!0),p.value&&p.value.max_epoch>0?(H(),U(`div`,Jl,[W(`div`,Yl,[a[16]||=W(`h2`,{class:`text-sm font-semibold text-gray-300 uppercase tracking-wider`},`爬取进度`,-1),W(`div`,Xl,[a[15]||=W(`span`,{class:`text-xs text-gray-500`},`状态:`,-1),W(`span`,{class:D([`px-2 py-0.5 rounded text-xs font-medium`,p.value.is_running?`bg-green-900/50 text-green-400`:`bg-gray-800 text-gray-500`])},O(p.value.is_running?`运行中`:`已停止`),3)])]),W(`div`,Zl,[W(`div`,Ql,[a[17]||=W(`div`,{class:`text-xs text-gray-500 mb-1`},`当前轮次`,-1),W(`div`,$l,[Ui(O(p.value.current_epoch),1),W(`span`,eu,`/`+O(p.value.max_epoch),1)])]),W(`div`,tu,[a[18]||=W(`div`,{class:`text-xs text-gray-500 mb-1`},`本轮队列`,-1),W(`div`,{class:D([`text-3xl font-bold`,p.value.queue_length>0?`text-blue-400`:`text-gray-500`])},O(w(p.value.queue_length)),3)]),W(`div`,nu,[a[19]||=W(`div`,{class:`text-xs text-gray-500 mb-1`},`已完成`,-1),W(`div`,ru,O(w(p.value.completed_count)),1),W(`div`,iu,O(p.value.queue_length>0?Math.round(p.value.completed_count/p.value.queue_length*100)+`%`:`--`),1)]),W(`div`,au,[a[20]||=W(`div`,{class:`text-xs text-gray-500 mb-1`},`下一轮链接池`,-1),W(`div`,{class:D([`text-3xl font-bold`,p.value.next_pool_size>0?`text-purple-400`:`text-gray-500`])},O(w(p.value.next_pool_size)),3)]),W(`div`,ou,[a[21]||=W(`div`,{class:`text-xs text-gray-500 mb-1`},`已收录总计`,-1),W(`div`,su,O(w(p.value.visited_total)),1)])]),p.value.queue_length>0?(H(),U(`div`,cu,[W(`div`,lu,[W(`div`,{class:`h-full bg-gradient-to-r from-blue-600 to-green-500 rounded-full transition-all duration-500`,style:me({width:`${p.value.completed_count/p.value.queue_length*100}%`})},null,4)])])):G(``,!0)])):G(``,!0),W(`div`,uu,[W(`div`,du,[W(`div`,null,[a[25]||=W(`h2`,{class:`text-sm font-semibold text-gray-300 uppercase tracking-wider mb-2`},`爬虫线程`,-1),W(`div`,fu,[W(`div`,null,[a[22]||=W(`div`,{class:`text-xs text-gray-500 mb-0.5`},`实际运行`,-1),W(`div`,{class:D([`text-3xl font-bold`,s.value>0?`text-green-400`:`text-gray-500`])},O(s.value),3)]),a[24]||=W(`div`,{class:`text-gray-700 text-xl`},`/`,-1),W(`div`,null,[a[23]||=W(`div`,{class:`text-xs text-gray-500 mb-0.5`},`设定上限`,-1),W(`div`,pu,O(o.value),1)])])]),W(`div`,mu,[W(`button`,{class:`w-9 h-9 flex items-center justify-center rounded-lg bg-gray-800 hover:bg-gray-700 text-gray-300 hover:text-white text-lg font-bold transition-colors cursor-pointer`,onClick:a[1]||=e=>c.value=Math.max(1,c.value-1),disabled:l.value},`−`,8,hu),An(W(`input`,{type:`number`,"onUpdate:modelValue":a[2]||=e=>c.value=e,min:`1`,max:`500`,class:`w-20 h-9 text-center bg-gray-800 border border-gray-700 rounded-lg text-white text-sm focus:outline-none focus:border-blue-500 transition-colors [appearance:textfield] [&::-webkit-outer-spin-button]:appearance-none [&::-webkit-inner-spin-button]:appearance-none`,onKeyup:go(g,[`enter`])},null,544),[[so,c.value,void 0,{number:!0}]]),W(`button`,{class:`w-9 h-9 flex items-center justify-center rounded-lg bg-gray-800 hover:bg-gray-700 text-gray-300 hover:text-white text-lg font-bold transition-colors cursor-pointer`,onClick:a[3]||=e=>c.value=Math.min(500,c.value+1),disabled:l.value},`+`,8,gu),W(`button`,{class:`h-9 px-4 bg-blue-700 hover:bg-blue-600 disabled:bg-gray-700 disabled:text-gray-500 text-white text-sm font-medium rounded-lg transition-colors cursor-pointer`,disabled:l.value||c.value===o.value,onClick:g},O(l.value?`保存中...`:`应用`),9,_u)])])]),e.backlinkStatus?(H(),U(`div`,vu,[W(`div`,yu,[W(`div`,bu,[a[28]||=W(`h2`,{class:`text-sm font-semibold text-gray-300 uppercase tracking-wider mb-3`},`反链计算(PageRank)`,-1),W(`div`,xu,[W(`div`,null,[a[26]||=W(`div`,{class:`text-xs text-gray-500 mb-0.5`},`下次自动执行`,-1),W(`div`,{class:D([`text-lg font-bold`,e.backlinkStatus.running?`text-yellow-400`:`text-white`])},O(e.backlinkStatus.running?`计算中...`:x(e.backlinkStatus.next_run)),3)]),W(`div`,null,[a[27]||=W(`div`,{class:`text-xs text-gray-500 mb-0.5`},`上次完成`,-1),W(`div`,Su,O(S(e.backlinkStatus.last_run)),1)])]),e.backlinkStatus.last_error?(H(),U(`div`,{key:0,class:`mt-2 text-xs text-red-400 truncate`,title:e.backlinkStatus.last_error},` 上次错误:`+O(e.backlinkStatus.last_error),9,Cu)):G(``,!0)]),W(`button`,{class:`h-9 px-4 bg-blue-700 hover:bg-blue-600 disabled:bg-gray-700 disabled:text-gray-500 text-white text-sm font-medium rounded-lg transition-colors cursor-pointer whitespace-nowrap`,disabled:e.backlinkTriggering||e.backlinkStatus.running,onClick:b},O(e.backlinkTriggering?`已触发...`:e.backlinkStatus.running?`计算中...`:`立即执行`),9,wu)])])):G(``,!0),W(`div`,Tu,[W(`div`,Eu,[a[29]||=W(`h2`,{class:`text-sm font-semibold text-gray-300 mb-3 md:mb-4 uppercase tracking-wider`},`域名分布 Top 10`,-1),W(`div`,Du,[(H(!0),U(B,null,mr(ee(t.value.domains),([e,n])=>(H(),U(`div`,{key:e,class:`flex items-center gap-2 md:gap-3`},[W(`div`,{class:`w-24 md:w-36 text-xs text-gray-400 truncate shrink-0`,title:e},O(e),9,Ou),W(`div`,ku,[W(`div`,{class:`h-full bg-blue-600 rounded-full transition-all duration-500`,style:me({width:`${n/t.value.domains[Object.keys(t.value.domains)[0]]*100}%`})},null,4)]),W(`div`,Au,O(w(n)),1)]))),128))])]),W(`div`,ju,[a[30]||=W(`h2`,{class:`text-sm font-semibold text-gray-300 mb-3 md:mb-4 uppercase tracking-wider`},`语种分布`,-1),W(`div`,Mu,[(H(!0),U(B,null,mr(Object.entries(t.value.languages||{}).sort((e,t)=>t[1]-e[1]),([e,n])=>(H(),U(`div`,{key:e,class:`flex items-center gap-2 md:gap-3`},[W(`div`,Nu,O(e),1),W(`div`,Pu,[W(`div`,{class:`h-full rounded-full transition-all duration-500`,style:me({width:`${n/t.value.total_urls*100}%`,backgroundColor:te(e)})},null,4)]),W(`div`,Fu,O(w(n)),1)]))),128))])])])],64)):G(``,!0)]))}},Lu={class:`p-4 md:p-8`},Ru={class:`flex flex-col md:flex-row md:items-center justify-between mb-4 md:mb-6 gap-3`},zu={class:`text-sm text-gray-500`},Bu={class:`flex items-center gap-2 md:gap-3`},Vu={class:`bg-gray-900 border border-gray-700 rounded-lg px-3 py-2 flex items-center gap-2`},Hu={class:`flex flex-col`},Uu={class:`text-sm font-medium text-gray-300`},Wu={class:`w-12 h-8 relative`},Gu={viewBox:`0 0 36 36`,class:`w-full h-full transform -rotate-90`},Ku=[`stroke-dasharray`],qu={class:`absolute inset-0 flex items-center justify-center text-[8px] font-medium text-gray-400`},Ju=[`value`],Yu={class:`flex flex-col sm:flex-row items-stretch sm:items-center gap-3 mb-4 md:mb-5`},Xu={class:`relative flex-1 max-w-full sm:max-w-sm`},Zu={key:0,class:`flex items-center justify-center h-48`},Qu={key:1,class:`bg-red-900/30 border border-red-800 rounded-lg p-4 text-red-300`},$u={key:2,class:`bg-gray-900 border border-gray-800 rounded-xl overflow-hidden`},ed={class:`hidden md:table w-full text-sm`},td={class:`px-5 py-3.5`},nd=[`href`],rd={class:`font-medium text-gray-200 group-hover:text-white line-clamp-2`},id={class:`text-xs text-gray-600 mt-0.5 break-all line-clamp-1`},ad={key:0,class:`text-xs text-gray-500 mt-1 line-clamp-1`},od=[`onClick`],sd={key:0,class:`text-gray-500`},cd={key:1,class:`mt-2`},ld={class:`flex flex-wrap gap-1.5`},ud=[`title`],dd={class:`text-gray-500 text-[10px] ml-0.5`},fd={class:`text-xs text-gray-600 mt-1`},pd={key:1,class:`text-xs text-gray-600`},md={class:`px-5 py-3.5`},hd={class:`text-gray-400 text-xs font-mono`},gd={class:`px-5 py-3.5`},_d={key:1,class:`text-xs text-gray-600`},vd={class:`px-5 py-3.5 text-gray-500 text-xs tabular-nums`},yd={class:`px-5 py-3.5 text-gray-500 text-xs`},bd={key:0},xd={class:`md:hidden divide-y divide-gray-800`},Sd=[`href`],Cd={class:`font-medium text-gray-200 text-sm line-clamp-2 mb-1`},wd={class:`text-xs text-gray-500 break-all line-clamp-1 mb-2`},Td={class:`flex items-center gap-2 text-xs mb-2`},Ed={class:`text-gray-400 font-mono`},Dd={class:`text-gray-600 ml-auto`},Od=[`onClick`],kd={key:0,class:`text-gray-500`},Ad={key:0,class:`mb-2`},jd={class:`flex flex-wrap gap-1`},Md=[`title`],Nd={class:`text-gray-500 text-[9px] ml-0.5`},Pd={class:`text-[10px] text-gray-600 mt-1`},Fd={key:1,class:`text-xs text-gray-600`},Id={key:0,class:`p-8 text-center text-gray-600`},Ld={key:3,class:`mt-3 text-xs text-gray-600 text-right pb-4 md:pb-0`},Rd={__name:`RecentCrawls`,setup(e){let t=F([]),n=F(0),r=F(!0),i=F(null),a=F(``),o=F(``),s=[20,50,100,200],c=F(50),l=F(new Set),u=F({}),d=F(new Set),f=F({size:0,max_size:1e4,usage:0});ir(async()=>{await p()});async function p(){r.value=!0,i.value=null;try{let[e,r]=await Promise.all([pl(c.value),Tl().catch(()=>({size:0,max_size:1e4,usage:0}))]);t.value=e.items||[],n.value=e.total||0,f.value=r}catch(e){i.value=`无法加载数据,可能人服务器未启动`,console.error(e)}finally{r.value=!1}}async function m(e){c.value=e,await p()}let h=ga(()=>{let e=t.value;if(a.value){let t=a.value.toLowerCase();e=e.filter(e=>e.title?.toLowerCase().includes(t)||e.url?.toLowerCase().includes(t)||e.domain?.toLowerCase().includes(t))}return o.value&&(e=e.filter(e=>Object.keys(e.language||{}).includes(o.value))),e});function g(e){return e?new Date(e*1e3).toLocaleString(`zh-CN`,{year:`numeric`,month:`2-digit`,day:`2-digit`,hour:`2-digit`,minute:`2-digit`,second:`2-digit`,hour12:!1}):`-`}function _(e){return{zh:{label:`中文`,cls:`bg-red-900/60 text-red-300`},en:{label:`EN`,cls:`bg-blue-900/60 text-blue-300`},ja:{label:`日`,cls:`bg-pink-900/60 text-pink-300`},ko:{label:`한`,cls:`bg-blue-900/60 text-blue-300`},fr:{label:`FR`,cls:`bg-green-900/60 text-green-300`},de:{label:`DE`,cls:`bg-yellow-900/60 text-yellow-300`},es:{label:`ES`,cls:`bg-green-900/60 text-green-300`},ru:{label:`RU`,cls:`bg-purple-900/60 text-purple-300`}}[e]||{label:e,cls:`bg-gray-800 text-gray-400`}}function v(e){return e?Object.entries(e).sort((e,t)=>t[1]-e[1])[0]:null}async function y(e){if(l.value.has(e)){l.value.delete(e);return}if(l.value.add(e),!u.value[e]){d.value.add(e);try{let t=await wl(e);u.value[e]=t.keywords||[]}catch(t){console.error(`Failed to load keywords:`,t),u.value[e]=[]}finally{d.value.delete(e)}}}return(e,t)=>(H(),U(`div`,Lu,[W(`div`,Ru,[W(`div`,null,[t[4]||=W(`h1`,{class:`text-xl md:text-2xl font-semibold text-white mb-1`},`最近爬取`,-1),W(`p`,zu,`共 `+O(n.value.toLocaleString())+` 条记录`,1)]),W(`div`,Bu,[W(`div`,Vu,[W(`div`,Hu,[t[5]||=W(`span`,{class:`text-[10px] text-gray-500 uppercase`},`关键词缓存`,-1),W(`span`,Uu,O(f.value.size.toLocaleString())+` / `+O(f.value.max_size.toLocaleString()),1)]),W(`div`,Wu,[(H(),U(`svg`,Gu,[t[6]||=W(`path`,{class:`text-gray-800`,d:`M18 2.0845 a 15.9155 15.9155 0 0 1 0 31.831 a 15.9155 15.9155 0 0 1 0 -31.831`,fill:`none`,stroke:`currentColor`,"stroke-width":`4`},null,-1),W(`path`,{class:D(f.value.usage>.9?`text-red-500`:f.value.usage>.7?`text-yellow-500`:`text-green-500`),d:`M18 2.0845 a 15.9155 15.9155 0 0 1 0 31.831 a 15.9155 15.9155 0 0 1 0 -31.831`,fill:`none`,stroke:`currentColor`,"stroke-width":`4`,"stroke-dasharray":`${(f.value.usage*100).toFixed(0)}, 100`},null,10,Ku)])),W(`span`,qu,O((f.value.usage*100).toFixed(0))+`% `,1)])]),An(W(`select`,{"onUpdate:modelValue":t[0]||=e=>c.value=e,onChange:t[1]||=e=>m(c.value),class:`bg-gray-900 border border-gray-700 text-gray-300 text-sm rounded-lg px-3 py-2 focus:border-blue-500 focus:outline-none`},[(H(),U(B,null,mr(s,e=>W(`option`,{key:e,value:e},`显示 `+O(e)+` 条`,9,Ju)),64))],544),[[co,c.value]]),W(`button`,{onClick:p,class:`bg-blue-600 hover:bg-blue-700 text-white text-sm px-4 py-2 rounded-lg transition-colors`},` 刷新 `)])]),W(`div`,Yu,[W(`div`,Xu,[An(W(`input`,{"onUpdate:modelValue":t[2]||=e=>a.value=e,type:`text`,placeholder:`搜索标题、URL、域名...`,class:`w-full bg-gray-900 border border-gray-700 text-gray-200 text-sm rounded-lg pl-10 pr-4 py-2 focus:border-blue-500 focus:outline-none placeholder-gray-600`},null,512),[[so,a.value]]),t[7]||=W(`span`,{class:`absolute left-3 top-1/2 -translate-y-1/2 text-gray-500`},`🔍`,-1)]),An(W(`select`,{"onUpdate:modelValue":t[3]||=e=>o.value=e,class:`bg-gray-900 border border-gray-700 text-gray-300 text-sm rounded-lg px-3 py-2 focus:border-blue-500 focus:outline-none`},[...t[8]||=[Wi(``,9)]],512),[[co,o.value]])]),r.value?(H(),U(`div`,Zu,[...t[9]||=[W(`div`,{class:`text-gray-400 animate-pulse`},`加载中...`,-1)]])):i.value?(H(),U(`div`,Qu,O(i.value),1)):(H(),U(`div`,$u,[W(`table`,ed,[t[11]||=W(`thead`,null,[W(`tr`,{class:`border-b border-gray-800`},[W(`th`,{class:`text-left px-5 py-3 text-gray-500 font-medium text-xs uppercase tracking-wider`},`标题`),W(`th`,{class:`text-left px-5 py-3 text-gray-500 font-medium text-xs uppercase tracking-wider w-28`},`域名`),W(`th`,{class:`text-left px-5 py-3 text-gray-500 font-medium text-xs uppercase tracking-wider w-16`},`语种`),W(`th`,{class:`text-left px-5 py-3 text-gray-500 font-medium text-xs uppercase tracking-wider w-20`},`字数`),W(`th`,{class:`text-left px-5 py-3 text-gray-500 font-medium text-xs uppercase tracking-wider w-48`},`爬取时间`)])],-1),W(`tbody`,null,[(H(!0),U(B,null,mr(h.value,e=>(H(),U(`tr`,{key:e.url,class:`border-b border-gray-800/50 hover:bg-gray-800/40 transition-colors group`},[W(`td`,td,[W(`a`,{href:e.url,target:`_blank`,rel:`noopener noreferrer`,class:`block hover:opacity-80 transition-opacity`},[W(`div`,rd,O(e.title||`(无标题)`),1),W(`div`,id,O(e.url),1)],8,nd),e.description?(H(),U(`div`,ad,O(e.description),1)):G(``,!0),W(`button`,{onClick:mo(t=>y(e.url),[`prevent`]),class:`mt-2 text-xs text-blue-400 hover:text-blue-300 flex items-center gap-1`},[W(`span`,null,O(l.value.has(e.url)?`▼`:`▶`)+` 关键词`,1),d.value.has(e.url)?(H(),U(`span`,sd,`加载中...`)):G(``,!0)],8,od),l.value.has(e.url)?(H(),U(`div`,cd,[u.value[e.url]?.length?(H(),U(B,{key:0},[W(`div`,ld,[(H(!0),U(B,null,mr(u.value[e.url],e=>(H(),U(`span`,{key:e.word,class:`text-xs px-2 py-0.5 rounded bg-gray-800 text-gray-300 border border-gray-700`,title:`权重: ${e.weight.toFixed(4)}`},[Ui(O(e.word)+` `,1),W(`span`,dd,O(e.weight.toFixed(2)),1)],8,ud))),128))]),W(`div`,fd,` 共 `+O(u.value[e.url].length)+` 个关键词 `,1)],64)):d.value.has(e.url)?G(``,!0):(H(),U(`span`,pd,` 暂无关键词(服务重启后缓存已清空) `))])):G(``,!0)]),W(`td`,md,[W(`span`,hd,O(e.domain),1)]),W(`td`,gd,[v(e.language)?(H(),U(`span`,{key:0,class:D([`text-xs px-1.5 py-0.5 rounded font-medium`,_(v(e.language)[0]).cls])},O(_(v(e.language)[0]).label)+` `+O((v(e.language)[1]*100).toFixed(0))+`% `,3)):(H(),U(`span`,_d,`-`))]),W(`td`,vd,O(e.word_count.toLocaleString()),1),W(`td`,yd,O(g(e.crawled_at)),1)]))),128)),h.value.length?G(``,!0):(H(),U(`tr`,bd,[...t[10]||=[W(`td`,{colspan:`5`,class:`px-5 py-12 text-center text-gray-600`},` 没有找到匹配的记录 `,-1)]]))])]),W(`div`,xd,[(H(!0),U(B,null,mr(h.value,e=>(H(),U(`div`,{key:e.url,class:`block p-4 hover:bg-gray-800/40 transition-colors`},[W(`a`,{href:e.url,target:`_blank`,rel:`noopener noreferrer`,class:`block`},[W(`div`,Cd,O(e.title||`(无标题)`),1),W(`div`,wd,O(e.url),1)],8,Sd),W(`div`,Td,[W(`span`,Ed,O(e.domain),1),v(e.language)?(H(),U(`span`,{key:0,class:D([`px-1.5 py-0.5 rounded font-medium`,_(v(e.language)[0]).cls])},O(_(v(e.language)[0]).label),3)):G(``,!0),W(`span`,Dd,O(g(e.crawled_at)),1)]),W(`button`,{onClick:mo(t=>y(e.url),[`prevent`]),class:`text-xs text-blue-400 hover:text-blue-300 flex items-center gap-1 mb-2`},[W(`span`,null,O(l.value.has(e.url)?`▼`:`▶`)+` 关键词`,1),d.value.has(e.url)?(H(),U(`span`,kd,`加载中...`)):G(``,!0)],8,Od),l.value.has(e.url)?(H(),U(`div`,Ad,[u.value[e.url]?.length?(H(),U(B,{key:0},[W(`div`,jd,[(H(!0),U(B,null,mr(u.value[e.url],e=>(H(),U(`span`,{key:e.word,class:`text-[10px] px-1.5 py-0.5 rounded bg-gray-800 text-gray-300 border border-gray-700`,title:`权重: ${e.weight.toFixed(4)}`},[Ui(O(e.word)+` `,1),W(`span`,Nd,O(e.weight.toFixed(2)),1)],8,Md))),128))]),W(`div`,Pd,` 共 `+O(u.value[e.url].length)+` 个 `,1)],64)):d.value.has(e.url)?G(``,!0):(H(),U(`span`,Fd,` 暂无关键词 `))])):G(``,!0)]))),128)),h.value.length?G(``,!0):(H(),U(`div`,Id,` 没有找到匹配的记录 `))])])),!r.value&&!i.value&&h.value.length?(H(),U(`div`,Ld,` 筛选后 `+O(h.value.length)+` 条 / 共 `+O(n.value.toLocaleString())+` 条 `,1)):G(``,!0)]))}},zd={class:`flex flex-col h-full`},Bd={class:`bg-gray-950 border-b border-gray-800 px-4 md:px-8 py-4 md:py-6`},Vd={class:`max-w-3xl mx-auto`},Hd={class:`relative`},Ud={class:`flex-1 overflow-y-auto px-8 py-6`},Wd={class:`max-w-3xl mx-auto`},Gd={key:0,class:`flex items-center gap-3 text-gray-400 py-8`},Kd={key:1,class:`bg-red-900/30 border border-red-800 rounded-lg p-4 text-red-300`},qd={key:2,class:`py-16 text-center text-gray-600`},Jd={key:3,class:`py-16 text-center text-gray-600`},Yd={key:4,class:`flex flex-col sm:flex-row sm:items-center gap-2 sm:gap-4 mb-4 md:mb-5 text-sm text-gray-500`},Xd={class:`flex items-center gap-2`},Zd={class:`text-gray-300`},Qd={class:`flex flex-wrap gap-2 sm:ml-auto`},$d={class:`text-gray-300`},ef={key:5,class:`space-y-2 md:space-y-1`},tf=[`onClick`],nf={class:`flex items-start gap-2 md:gap-3 mb-2`},rf={class:`flex-1 min-w-0`},af={class:`text-blue-400 group-hover:text-blue-300 text-base md:text-lg leading-snug`},of={class:`text-xs text-gray-600 mt-0.5 truncate`},sf={class:`flex flex-col items-end gap-1 shrink-0`},cf={class:`text-xs text-gray-600`},lf={class:`w-12 md:w-14 bg-gray-800 rounded-full h-1.5 overflow-hidden`},uf={class:`text-gray-400 text-sm leading-relaxed mb-2 md:mb-3`},df={class:`flex flex-wrap items-center gap-2 md:gap-3 text-xs`},ff={class:`flex flex-wrap gap-1.5`},pf=[`title`],mf={class:`text-blue-400`},hf={class:`text-gray-600 ml-1`},gf={key:0,class:`text-gray-600`},_f={class:`text-gray-700 ml-0 md:ml-auto`},vf={class:`text-xs text-gray-700 mt-1 truncate`},yf={key:6,class:`mt-6 text-center`},bf=``,xf=10,Sf={__name:`SearchView`,setup(e){let t=F(``),n=F([]),r=F(0),i=F({}),a=F(!1),o=F(null),s=F(0),c=null;async function l(e,t=!1){if(!e.trim()){n.value=[],r.value=0,i.value={};return}t||(s.value=0),a.value=!0,o.value=null;try{let a=encodeURIComponent(e),o=s.value*xf,c=await fetch(`${bf}/search?qh=${a}&slice=${o}:${o+xf}`);if(!c.ok)throw Error(`HTTP ${c.status}`);let l=await c.json();t?n.value=[...n.value,...l.results||[]]:n.value=l.results||[],r.value=l.total||0,i.value=l.counts||{}}catch(e){o.value=`搜索失败,可能是人服务器未启动`,console.error(e)}finally{a.value=!1}}function u(){clearTimeout(c),c=setTimeout(()=>l(t.value),400)}function d(e){e.key===`Enter`&&(clearTimeout(c),l(t.value))}function f(e,t=120){return e?e.length>t?e.slice(0,t)+`…`:e:``}function p(e){return e?e.toLocaleString():`0`}function m(e){return Math.min(100,Math.round(e*100))}function h(e){return{zh:`中文`,en:`EN`,ja:`JA`,ko:`KO`,fr:`FR`,de:`DE`,es:`ES`,ru:`RU`}[e]||e?.toUpperCase()}function g(e){try{return new URL(e).hostname}catch{return e}}function _(e){window.open(e,`_blank`)}return(e,c)=>(H(),U(`div`,zd,[W(`div`,Bd,[W(`div`,Vd,[W(`div`,Hd,[An(W(`input`,{"onUpdate:modelValue":c[0]||=e=>t.value=e,onInput:u,onKeydown:d,type:`text`,placeholder:`输入关键词搜索,或用 site:example.com 限定域名`,class:`w-full bg-gray-900 border border-gray-700 rounded-xl md:rounded-2xl px-4 md:px-6 py-3 md:py-4 pr-20 md:pr-14 text-white text-base md:text-lg placeholder-gray-600 focus:outline-none focus:border-blue-500 focus:ring-1 focus:ring-blue-500 transition`},null,544),[[so,t.value]]),W(`button`,{onClick:c[1]||=e=>l(t.value),class:`absolute right-2 md:right-3 top-1/2 -translate-y-1/2 bg-blue-600 hover:bg-blue-500 text-white rounded-lg md:rounded-xl px-3 md:px-4 py-1.5 md:py-2 text-sm font-medium transition`},` 搜索 `)])])]),W(`div`,Ud,[W(`div`,Wd,[a.value?(H(),U(`div`,Gd,[...c[3]||=[W(`div`,{class:`w-5 h-5 border-2 border-blue-400 border-t-transparent rounded-full animate-spin`},null,-1),W(`span`,null,`搜索中...`,-1)]])):o.value?(H(),U(`div`,Kd,O(o.value),1)):t.value.trim()?n.value.length===0&&!a.value?(H(),U(`div`,Jd,` 未找到相关结果 `)):n.value.length>0?(H(),U(`div`,Yd,[W(`div`,Xd,[W(`span`,null,[c[4]||=Ui(`找到约 `,-1),W(`strong`,Zd,O(p(r.value)),1),c[5]||=Ui(` 条结果`,-1)]),c[6]||=W(`span`,{class:`text-gray-700`},`|`,-1),W(`span`,null,O(n.value.length)+` 条已加载`,1)]),W(`div`,Qd,[(H(!0),U(B,null,mr(i.value,(e,t)=>(H(),U(`span`,{key:t,class:`inline-flex items-center gap-1 bg-gray-800 rounded px-2 py-0.5 text-xs text-gray-400`},[W(`span`,$d,O(t),1),W(`span`,null,O(p(e)),1)]))),128))])])):G(``,!0):(H(),U(`div`,qd,` 输入关键词开始搜索 `)),n.value.length>0?(H(),U(`div`,ef,[(H(!0),U(B,null,mr(n.value,(e,t)=>(H(),U(`div`,{key:t,onClick:t=>_(e.url),class:`group block bg-gray-900/50 hover:bg-gray-900 border border-gray-800 hover:border-gray-700 rounded-xl p-4 md:p-5 cursor-pointer transition`},[W(`div`,nf,[W(`div`,rf,[W(`div`,af,O(e.snippet?.title||g(e.url)),1),W(`div`,of,O(g(e.url)),1)]),W(`div`,sf,[W(`div`,cf,O(m(e.score))+`%`,1),W(`div`,lf,[W(`div`,{class:`h-full bg-blue-500 rounded-full`,style:me({width:m(e.score)+`%`})},null,4)])])]),W(`p`,uf,O(f(e.snippet?.description||e.snippet?.text)),1),W(`div`,df,[W(`div`,ff,[(H(!0),U(B,null,mr(e.relevance,(e,t)=>(H(),U(`span`,{key:t,class:`inline-flex items-center bg-gray-800 rounded px-1.5 py-0.5`,title:`${t}: ${(e*100).toFixed(1)}%`},[W(`span`,mf,O(t),1),W(`span`,hf,O((e*100).toFixed(0))+`%`,1)],8,pf))),128))]),e.snippet?.language?(H(),U(`span`,gf,O(h(e.snippet.language)),1)):G(``,!0),W(`span`,_f,O(p(e.domain_count))+` 个结果`,1)]),W(`div`,vf,O(e.url),1)],8,tf))),128))])):G(``,!0),n.value.length>0&&n.value.length{s.value++,l(t.value,!0)},class:`bg-gray-800 hover:bg-gray-700 text-gray-300 rounded-xl px-6 py-2.5 text-sm transition`},` 加载更多 (`+O(n.value.length)+`/`+O(p(r.value))+`) `,1)])):G(``,!0)])])]))}},Cf={class:`flex h-screen bg-gray-950 text-gray-100 font-sans`},wf={class:`hidden md:flex w-56 bg-gray-900 border-r border-gray-800 flex-col shrink-0`},Tf={class:`flex-1 py-4 px-3 space-y-1`},Ef=[`onClick`],Df={class:`flex-1 overflow-y-auto pb-16 md:pb-0`},Of={class:`md:hidden fixed bottom-0 left-0 right-0 bg-gray-900 border-t border-gray-800 flex justify-around items-center h-16 z-50`},kf=[`onClick`],Af={class:`text-lg`},jf={class:`text-[10px]`};bo({__name:`App`,setup(e){let t=F(`dashboard`),n=[{id:`dashboard`,label:`概览`,icon:`📊`},{id:`recent`,label:`最近`,icon:`🕷️`},{id:`search`,label:`搜索`,icon:`🔍`}];return(e,r)=>(H(),U(`div`,Cf,[W(`aside`,wf,[r[0]||=W(`div`,{class:`px-5 py-5 border-b border-gray-800`},[W(`div`,{class:`text-lg font-semibold text-white tracking-tight`},`SESE Admin`),W(`div`,{class:`text-xs text-gray-500 mt-0.5`},`爬取内容监控`)],-1),W(`nav`,Tf,[(H(),U(B,null,mr(n,e=>W(`button`,{key:e.id,onClick:n=>t.value=e.id,class:D([`w-full flex items-center gap-3 px-3 py-2.5 rounded-lg text-sm font-medium transition-colors`,t.value===e.id?`bg-blue-600 text-white`:`text-gray-400 hover:text-white hover:bg-gray-800`])},[W(`span`,null,O(e.icon),1),Ui(` `+O(e.label),1)],10,Ef)),64))]),r[1]||=W(`div`,{class:`px-5 py-4 border-t border-gray-800`},[W(`div`,{class:`text-xs text-gray-600`},`sese-engine v1.0`)],-1)]),W(`main`,Df,[t.value===`dashboard`?(H(),Pi(Iu,{key:0})):t.value===`recent`?(H(),Pi(Rd,{key:1})):t.value===`search`?(H(),Pi(Sf,{key:2})):G(``,!0)]),W(`nav`,Of,[(H(),U(B,null,mr(n,e=>W(`button`,{key:e.id,onClick:n=>t.value=e.id,class:D([`flex flex-col items-center justify-center gap-0.5 flex-1 h-full transition-colors`,t.value===e.id?`text-blue-400`:`text-gray-500`])},[W(`span`,Af,O(e.icon),1),W(`span`,jf,O(e.label),1)],10,kf)),64))])]))}}).mount(`#app`); \ No newline at end of file +`).map(e=>e.trim()).filter(e=>e).filter(ne);if(e.length===0){i.value=`未检测到有效 URL`;return}f.value=!0;let t=0,n=0;try{for(let r of e)try{await hl(r),t++}catch{n++}d.value=``,await v(),i.value=null,n>0&&(i.value=`添加完成:${t} 成功,${n} 失败`)}finally{f.value=!1}}return(e,a)=>(H(),U(`div`,El,[a[31]||=W(`h1`,{class:`text-xl md:text-2xl font-semibold text-white mb-6 md:mb-8`},`概览`,-1),n.value?(H(),U(`div`,Dl,[...a[4]||=[W(`div`,{class:`text-gray-400 animate-pulse`},`加载中...`,-1)]])):i.value?(H(),U(`div`,Ol,O(i.value),1)):t.value?(H(),U(B,{key:2},[W(`div`,kl,[W(`div`,Al,[a[5]||=W(`div`,{class:`text-sm text-gray-500 mb-2`},`已爬取 URL`,-1),W(`div`,jl,O(w(t.value.total_urls)),1)]),W(`div`,Ml,[a[6]||=W(`div`,{class:`text-sm text-gray-500 mb-2`},`总词数`,-1),W(`div`,Nl,O(w(t.value.total_words)),1)]),W(`div`,Pl,[a[7]||=W(`div`,{class:`text-sm text-gray-500 mb-2`},`域名数量`,-1),W(`div`,Fl,O(w(t.value.total_domains)),1)]),W(`div`,Il,[a[8]||=W(`div`,{class:`text-sm text-gray-500 mb-2`},`待重爬`,-1),W(`div`,{class:D([`text-3xl font-bold`,t.value.recrawl_eligible>0?`text-orange-400`:`text-green-400`])},O(w(t.value.recrawl_eligible)),3)]),W(`div`,Ll,[W(`div`,null,[a[9]||=W(`div`,{class:`text-sm text-gray-500 mb-2`},`待刷盘`,-1),W(`div`,{class:D([`text-3xl font-bold`,t.value.pending>0?`text-yellow-400`:`text-green-400`])},O(w(t.value.pending)),3)]),W(`button`,{class:`mt-3 w-full bg-blue-700 hover:bg-blue-600 disabled:bg-gray-700 disabled:text-gray-500 text-white text-sm font-medium py-1.5 px-3 rounded transition-colors cursor-pointer`,disabled:r.value||!t.value.pending,onClick:C},O(r.value?`刷盘中...`:`立即刷盘`),9,Rl)])]),u.value?(H(),U(`div`,zl,[a[14]||=W(`h2`,{class:`text-sm font-semibold text-gray-300 uppercase tracking-wider mb-4`},`优先爬取队列`,-1),W(`div`,Bl,[W(`div`,Vl,[a[10]||=W(`div`,{class:`text-xs text-gray-500 mb-1`},`一级队列`,-1),W(`div`,{class:D([`text-2xl font-bold`,(u.value.level1||0)>0?`text-yellow-400`:`text-gray-500`])},O(w(u.value.level1||0)),3)]),W(`div`,Hl,[a[11]||=W(`div`,{class:`text-xs text-gray-500 mb-1`},`二级队列`,-1),W(`div`,{class:D([`text-2xl font-bold`,(u.value.level2_queue||0)+(u.value.level2_inflight||0)>0?`text-yellow-400`:`text-gray-500`])},O(w((u.value.level2_queue||0)+(u.value.level2_inflight||0))),3)]),W(`div`,Ul,[a[12]||=W(`div`,{class:`text-xs text-gray-500 mb-1`},`线程使用`,-1),W(`div`,{class:D([`text-2xl font-bold`,u.value.active>0?`text-orange-400`:`text-gray-500`])},O(w(u.value.active)),3)]),W(`div`,Wl,[a[13]||=W(`div`,{class:`text-xs text-gray-500 mb-1`},`线程总数`,-1),W(`div`,Gl,O(u.value.max_workers),1)])]),W(`div`,Kl,[An(W(`textarea`,{"onUpdate:modelValue":a[0]||=e=>d.value=e,rows:`3`,placeholder:`批量添加 URL(每行一个,仅发送有效链接)`,class:`flex-1 bg-gray-800 border border-gray-700 text-gray-200 text-sm rounded-lg px-3 py-2 resize-none focus:outline-none focus:border-blue-500 placeholder-gray-600`},null,512),[[so,d.value]]),W(`button`,{class:`bg-blue-700 hover:bg-blue-600 disabled:bg-gray-700 text-white text-sm font-medium px-4 py-2 rounded-lg transition-colors cursor-pointer whitespace-nowrap self-end`,disabled:f.value||!d.value.trim(),onClick:re},O(f.value?`添加中...`:`批量添加`),9,ql)])])):G(``,!0),p.value&&p.value.max_epoch>0?(H(),U(`div`,Jl,[W(`div`,Yl,[a[16]||=W(`h2`,{class:`text-sm font-semibold text-gray-300 uppercase tracking-wider`},`爬取进度`,-1),W(`div`,Xl,[a[15]||=W(`span`,{class:`text-xs text-gray-500`},`状态:`,-1),W(`span`,{class:D([`px-2 py-0.5 rounded text-xs font-medium`,p.value.is_running?`bg-green-900/50 text-green-400`:`bg-gray-800 text-gray-500`])},O(p.value.is_running?`运行中`:`已停止`),3)])]),W(`div`,Zl,[W(`div`,Ql,[a[17]||=W(`div`,{class:`text-xs text-gray-500 mb-1`},`当前轮次`,-1),W(`div`,$l,[Ui(O(p.value.current_epoch),1),W(`span`,eu,`/`+O(p.value.max_epoch),1)])]),W(`div`,tu,[a[18]||=W(`div`,{class:`text-xs text-gray-500 mb-1`},`本轮队列`,-1),W(`div`,{class:D([`text-3xl font-bold`,p.value.queue_length>0?`text-blue-400`:`text-gray-500`])},O(w(p.value.queue_length)),3)]),W(`div`,nu,[a[19]||=W(`div`,{class:`text-xs text-gray-500 mb-1`},`已完成`,-1),W(`div`,ru,O(w(p.value.completed_count)),1),W(`div`,iu,O(p.value.queue_length>0?Math.round(p.value.completed_count/p.value.queue_length*100)+`%`:`--`),1)]),W(`div`,au,[a[20]||=W(`div`,{class:`text-xs text-gray-500 mb-1`},`本轮新发现`,-1),W(`div`,{class:D([`text-3xl font-bold`,p.value.new_links_count>0?`text-purple-400`:`text-gray-500`])},O(w(p.value.new_links_count)),3)]),W(`div`,ou,[a[21]||=W(`div`,{class:`text-xs text-gray-500 mb-1`},`已收录总计`,-1),W(`div`,su,O(w(p.value.visited_total)),1)])]),p.value.queue_length>0?(H(),U(`div`,cu,[W(`div`,lu,[W(`div`,{class:`h-full bg-gradient-to-r from-blue-600 to-green-500 rounded-full transition-all duration-500`,style:me({width:`${p.value.completed_count/p.value.queue_length*100}%`})},null,4)])])):G(``,!0)])):G(``,!0),W(`div`,uu,[W(`div`,du,[W(`div`,null,[a[25]||=W(`h2`,{class:`text-sm font-semibold text-gray-300 uppercase tracking-wider mb-2`},`爬虫线程`,-1),W(`div`,fu,[W(`div`,null,[a[22]||=W(`div`,{class:`text-xs text-gray-500 mb-0.5`},`实际运行`,-1),W(`div`,{class:D([`text-3xl font-bold`,s.value>0?`text-green-400`:`text-gray-500`])},O(s.value),3)]),a[24]||=W(`div`,{class:`text-gray-700 text-xl`},`/`,-1),W(`div`,null,[a[23]||=W(`div`,{class:`text-xs text-gray-500 mb-0.5`},`设定上限`,-1),W(`div`,pu,O(o.value),1)])])]),W(`div`,mu,[W(`button`,{class:`w-9 h-9 flex items-center justify-center rounded-lg bg-gray-800 hover:bg-gray-700 text-gray-300 hover:text-white text-lg font-bold transition-colors cursor-pointer`,onClick:a[1]||=e=>c.value=Math.max(1,c.value-1),disabled:l.value},`−`,8,hu),An(W(`input`,{type:`number`,"onUpdate:modelValue":a[2]||=e=>c.value=e,min:`1`,max:`500`,class:`w-20 h-9 text-center bg-gray-800 border border-gray-700 rounded-lg text-white text-sm focus:outline-none focus:border-blue-500 transition-colors [appearance:textfield] [&::-webkit-outer-spin-button]:appearance-none [&::-webkit-inner-spin-button]:appearance-none`,onKeyup:go(g,[`enter`])},null,544),[[so,c.value,void 0,{number:!0}]]),W(`button`,{class:`w-9 h-9 flex items-center justify-center rounded-lg bg-gray-800 hover:bg-gray-700 text-gray-300 hover:text-white text-lg font-bold transition-colors cursor-pointer`,onClick:a[3]||=e=>c.value=Math.min(500,c.value+1),disabled:l.value},`+`,8,gu),W(`button`,{class:`h-9 px-4 bg-blue-700 hover:bg-blue-600 disabled:bg-gray-700 disabled:text-gray-500 text-white text-sm font-medium rounded-lg transition-colors cursor-pointer`,disabled:l.value||c.value===o.value,onClick:g},O(l.value?`保存中...`:`应用`),9,_u)])])]),e.backlinkStatus?(H(),U(`div`,vu,[W(`div`,yu,[W(`div`,bu,[a[28]||=W(`h2`,{class:`text-sm font-semibold text-gray-300 uppercase tracking-wider mb-3`},`反链计算(PageRank)`,-1),W(`div`,xu,[W(`div`,null,[a[26]||=W(`div`,{class:`text-xs text-gray-500 mb-0.5`},`下次自动执行`,-1),W(`div`,{class:D([`text-lg font-bold`,e.backlinkStatus.running?`text-yellow-400`:`text-white`])},O(e.backlinkStatus.running?`计算中...`:x(e.backlinkStatus.next_run)),3)]),W(`div`,null,[a[27]||=W(`div`,{class:`text-xs text-gray-500 mb-0.5`},`上次完成`,-1),W(`div`,Su,O(S(e.backlinkStatus.last_run)),1)])]),e.backlinkStatus.last_error?(H(),U(`div`,{key:0,class:`mt-2 text-xs text-red-400 truncate`,title:e.backlinkStatus.last_error},` 上次错误:`+O(e.backlinkStatus.last_error),9,Cu)):G(``,!0)]),W(`button`,{class:`h-9 px-4 bg-blue-700 hover:bg-blue-600 disabled:bg-gray-700 disabled:text-gray-500 text-white text-sm font-medium rounded-lg transition-colors cursor-pointer whitespace-nowrap`,disabled:e.backlinkTriggering||e.backlinkStatus.running,onClick:b},O(e.backlinkTriggering?`已触发...`:e.backlinkStatus.running?`计算中...`:`立即执行`),9,wu)])])):G(``,!0),W(`div`,Tu,[W(`div`,Eu,[a[29]||=W(`h2`,{class:`text-sm font-semibold text-gray-300 mb-3 md:mb-4 uppercase tracking-wider`},`域名分布 Top 10`,-1),W(`div`,Du,[(H(!0),U(B,null,mr(ee(t.value.domains),([e,n])=>(H(),U(`div`,{key:e,class:`flex items-center gap-2 md:gap-3`},[W(`div`,{class:`w-24 md:w-36 text-xs text-gray-400 truncate shrink-0`,title:e},O(e),9,Ou),W(`div`,ku,[W(`div`,{class:`h-full bg-blue-600 rounded-full transition-all duration-500`,style:me({width:`${n/t.value.domains[Object.keys(t.value.domains)[0]]*100}%`})},null,4)]),W(`div`,Au,O(w(n)),1)]))),128))])]),W(`div`,ju,[a[30]||=W(`h2`,{class:`text-sm font-semibold text-gray-300 mb-3 md:mb-4 uppercase tracking-wider`},`语种分布`,-1),W(`div`,Mu,[(H(!0),U(B,null,mr(Object.entries(t.value.languages||{}).sort((e,t)=>t[1]-e[1]),([e,n])=>(H(),U(`div`,{key:e,class:`flex items-center gap-2 md:gap-3`},[W(`div`,Nu,O(e),1),W(`div`,Pu,[W(`div`,{class:`h-full rounded-full transition-all duration-500`,style:me({width:`${n/t.value.total_urls*100}%`,backgroundColor:te(e)})},null,4)]),W(`div`,Fu,O(w(n)),1)]))),128))])])])],64)):G(``,!0)]))}},Lu={class:`p-4 md:p-8`},Ru={class:`flex flex-col md:flex-row md:items-center justify-between mb-4 md:mb-6 gap-3`},zu={class:`text-sm text-gray-500`},Bu={class:`flex items-center gap-2 md:gap-3`},Vu={class:`bg-gray-900 border border-gray-700 rounded-lg px-3 py-2 flex items-center gap-2`},Hu={class:`flex flex-col`},Uu={class:`text-sm font-medium text-gray-300`},Wu={class:`w-12 h-8 relative`},Gu={viewBox:`0 0 36 36`,class:`w-full h-full transform -rotate-90`},Ku=[`stroke-dasharray`],qu={class:`absolute inset-0 flex items-center justify-center text-[8px] font-medium text-gray-400`},Ju=[`value`],Yu={class:`flex flex-col sm:flex-row items-stretch sm:items-center gap-3 mb-4 md:mb-5`},Xu={class:`relative flex-1 max-w-full sm:max-w-sm`},Zu={key:0,class:`flex items-center justify-center h-48`},Qu={key:1,class:`bg-red-900/30 border border-red-800 rounded-lg p-4 text-red-300`},$u={key:2,class:`bg-gray-900 border border-gray-800 rounded-xl overflow-hidden`},ed={class:`hidden md:table w-full text-sm`},td={class:`px-5 py-3.5`},nd=[`href`],rd={class:`font-medium text-gray-200 group-hover:text-white line-clamp-2`},id={class:`text-xs text-gray-600 mt-0.5 break-all line-clamp-1`},ad={key:0,class:`text-xs text-gray-500 mt-1 line-clamp-1`},od=[`onClick`],sd={key:0,class:`text-gray-500`},cd={key:1,class:`mt-2`},ld={class:`flex flex-wrap gap-1.5`},ud=[`title`],dd={class:`text-gray-500 text-[10px] ml-0.5`},fd={class:`text-xs text-gray-600 mt-1`},pd={key:1,class:`text-xs text-gray-600`},md={class:`px-5 py-3.5`},hd={class:`text-gray-400 text-xs font-mono`},gd={class:`px-5 py-3.5`},_d={key:1,class:`text-xs text-gray-600`},vd={class:`px-5 py-3.5 text-gray-500 text-xs tabular-nums`},yd={class:`px-5 py-3.5 text-gray-500 text-xs`},bd={key:0},xd={class:`md:hidden divide-y divide-gray-800`},Sd=[`href`],Cd={class:`font-medium text-gray-200 text-sm line-clamp-2 mb-1`},wd={class:`text-xs text-gray-500 break-all line-clamp-1 mb-2`},Td={class:`flex items-center gap-2 text-xs mb-2`},Ed={class:`text-gray-400 font-mono`},Dd={class:`text-gray-600 ml-auto`},Od=[`onClick`],kd={key:0,class:`text-gray-500`},Ad={key:0,class:`mb-2`},jd={class:`flex flex-wrap gap-1`},Md=[`title`],Nd={class:`text-gray-500 text-[9px] ml-0.5`},Pd={class:`text-[10px] text-gray-600 mt-1`},Fd={key:1,class:`text-xs text-gray-600`},Id={key:0,class:`p-8 text-center text-gray-600`},Ld={key:3,class:`mt-3 text-xs text-gray-600 text-right pb-4 md:pb-0`},Rd={__name:`RecentCrawls`,setup(e){let t=F([]),n=F(0),r=F(!0),i=F(null),a=F(``),o=F(``),s=[20,50,100,200],c=F(50),l=F(new Set),u=F({}),d=F(new Set),f=F({size:0,max_size:1e4,usage:0});ir(async()=>{await p()});async function p(){r.value=!0,i.value=null;try{let[e,r]=await Promise.all([pl(c.value),Tl().catch(()=>({size:0,max_size:1e4,usage:0}))]);t.value=e.items||[],n.value=e.total||0,f.value=r}catch(e){i.value=`无法加载数据,可能人服务器未启动`,console.error(e)}finally{r.value=!1}}async function m(e){c.value=e,await p()}let h=ga(()=>{let e=t.value;if(a.value){let t=a.value.toLowerCase();e=e.filter(e=>e.title?.toLowerCase().includes(t)||e.url?.toLowerCase().includes(t)||e.domain?.toLowerCase().includes(t))}return o.value&&(e=e.filter(e=>Object.keys(e.language||{}).includes(o.value))),e});function g(e){return e?new Date(e*1e3).toLocaleString(`zh-CN`,{year:`numeric`,month:`2-digit`,day:`2-digit`,hour:`2-digit`,minute:`2-digit`,second:`2-digit`,hour12:!1}):`-`}function _(e){return{zh:{label:`中文`,cls:`bg-red-900/60 text-red-300`},en:{label:`EN`,cls:`bg-blue-900/60 text-blue-300`},ja:{label:`日`,cls:`bg-pink-900/60 text-pink-300`},ko:{label:`한`,cls:`bg-blue-900/60 text-blue-300`},fr:{label:`FR`,cls:`bg-green-900/60 text-green-300`},de:{label:`DE`,cls:`bg-yellow-900/60 text-yellow-300`},es:{label:`ES`,cls:`bg-green-900/60 text-green-300`},ru:{label:`RU`,cls:`bg-purple-900/60 text-purple-300`}}[e]||{label:e,cls:`bg-gray-800 text-gray-400`}}function v(e){return e?Object.entries(e).sort((e,t)=>t[1]-e[1])[0]:null}async function y(e){if(l.value.has(e)){l.value.delete(e);return}if(l.value.add(e),!u.value[e]){d.value.add(e);try{let t=await wl(e);u.value[e]=t.keywords||[]}catch(t){console.error(`Failed to load keywords:`,t),u.value[e]=[]}finally{d.value.delete(e)}}}return(e,t)=>(H(),U(`div`,Lu,[W(`div`,Ru,[W(`div`,null,[t[4]||=W(`h1`,{class:`text-xl md:text-2xl font-semibold text-white mb-1`},`最近爬取`,-1),W(`p`,zu,`共 `+O(n.value.toLocaleString())+` 条记录`,1)]),W(`div`,Bu,[W(`div`,Vu,[W(`div`,Hu,[t[5]||=W(`span`,{class:`text-[10px] text-gray-500 uppercase`},`关键词缓存`,-1),W(`span`,Uu,O(f.value.size.toLocaleString())+` / `+O(f.value.max_size.toLocaleString()),1)]),W(`div`,Wu,[(H(),U(`svg`,Gu,[t[6]||=W(`path`,{class:`text-gray-800`,d:`M18 2.0845 a 15.9155 15.9155 0 0 1 0 31.831 a 15.9155 15.9155 0 0 1 0 -31.831`,fill:`none`,stroke:`currentColor`,"stroke-width":`4`},null,-1),W(`path`,{class:D(f.value.usage>.9?`text-red-500`:f.value.usage>.7?`text-yellow-500`:`text-green-500`),d:`M18 2.0845 a 15.9155 15.9155 0 0 1 0 31.831 a 15.9155 15.9155 0 0 1 0 -31.831`,fill:`none`,stroke:`currentColor`,"stroke-width":`4`,"stroke-dasharray":`${(f.value.usage*100).toFixed(0)}, 100`},null,10,Ku)])),W(`span`,qu,O((f.value.usage*100).toFixed(0))+`% `,1)])]),An(W(`select`,{"onUpdate:modelValue":t[0]||=e=>c.value=e,onChange:t[1]||=e=>m(c.value),class:`bg-gray-900 border border-gray-700 text-gray-300 text-sm rounded-lg px-3 py-2 focus:border-blue-500 focus:outline-none`},[(H(),U(B,null,mr(s,e=>W(`option`,{key:e,value:e},`显示 `+O(e)+` 条`,9,Ju)),64))],544),[[co,c.value]]),W(`button`,{onClick:p,class:`bg-blue-600 hover:bg-blue-700 text-white text-sm px-4 py-2 rounded-lg transition-colors`},` 刷新 `)])]),W(`div`,Yu,[W(`div`,Xu,[An(W(`input`,{"onUpdate:modelValue":t[2]||=e=>a.value=e,type:`text`,placeholder:`搜索标题、URL、域名...`,class:`w-full bg-gray-900 border border-gray-700 text-gray-200 text-sm rounded-lg pl-10 pr-4 py-2 focus:border-blue-500 focus:outline-none placeholder-gray-600`},null,512),[[so,a.value]]),t[7]||=W(`span`,{class:`absolute left-3 top-1/2 -translate-y-1/2 text-gray-500`},`🔍`,-1)]),An(W(`select`,{"onUpdate:modelValue":t[3]||=e=>o.value=e,class:`bg-gray-900 border border-gray-700 text-gray-300 text-sm rounded-lg px-3 py-2 focus:border-blue-500 focus:outline-none`},[...t[8]||=[Wi(``,9)]],512),[[co,o.value]])]),r.value?(H(),U(`div`,Zu,[...t[9]||=[W(`div`,{class:`text-gray-400 animate-pulse`},`加载中...`,-1)]])):i.value?(H(),U(`div`,Qu,O(i.value),1)):(H(),U(`div`,$u,[W(`table`,ed,[t[11]||=W(`thead`,null,[W(`tr`,{class:`border-b border-gray-800`},[W(`th`,{class:`text-left px-5 py-3 text-gray-500 font-medium text-xs uppercase tracking-wider`},`标题`),W(`th`,{class:`text-left px-5 py-3 text-gray-500 font-medium text-xs uppercase tracking-wider w-28`},`域名`),W(`th`,{class:`text-left px-5 py-3 text-gray-500 font-medium text-xs uppercase tracking-wider w-16`},`语种`),W(`th`,{class:`text-left px-5 py-3 text-gray-500 font-medium text-xs uppercase tracking-wider w-20`},`字数`),W(`th`,{class:`text-left px-5 py-3 text-gray-500 font-medium text-xs uppercase tracking-wider w-48`},`爬取时间`)])],-1),W(`tbody`,null,[(H(!0),U(B,null,mr(h.value,e=>(H(),U(`tr`,{key:e.url,class:`border-b border-gray-800/50 hover:bg-gray-800/40 transition-colors group`},[W(`td`,td,[W(`a`,{href:e.url,target:`_blank`,rel:`noopener noreferrer`,class:`block hover:opacity-80 transition-opacity`},[W(`div`,rd,O(e.title||`(无标题)`),1),W(`div`,id,O(e.url),1)],8,nd),e.description?(H(),U(`div`,ad,O(e.description),1)):G(``,!0),W(`button`,{onClick:mo(t=>y(e.url),[`prevent`]),class:`mt-2 text-xs text-blue-400 hover:text-blue-300 flex items-center gap-1`},[W(`span`,null,O(l.value.has(e.url)?`▼`:`▶`)+` 关键词`,1),d.value.has(e.url)?(H(),U(`span`,sd,`加载中...`)):G(``,!0)],8,od),l.value.has(e.url)?(H(),U(`div`,cd,[u.value[e.url]?.length?(H(),U(B,{key:0},[W(`div`,ld,[(H(!0),U(B,null,mr(u.value[e.url],e=>(H(),U(`span`,{key:e.word,class:`text-xs px-2 py-0.5 rounded bg-gray-800 text-gray-300 border border-gray-700`,title:`权重: ${e.weight.toFixed(4)}`},[Ui(O(e.word)+` `,1),W(`span`,dd,O(e.weight.toFixed(2)),1)],8,ud))),128))]),W(`div`,fd,` 共 `+O(u.value[e.url].length)+` 个关键词 `,1)],64)):d.value.has(e.url)?G(``,!0):(H(),U(`span`,pd,` 暂无关键词(服务重启后缓存已清空) `))])):G(``,!0)]),W(`td`,md,[W(`span`,hd,O(e.domain),1)]),W(`td`,gd,[v(e.language)?(H(),U(`span`,{key:0,class:D([`text-xs px-1.5 py-0.5 rounded font-medium`,_(v(e.language)[0]).cls])},O(_(v(e.language)[0]).label)+` `+O((v(e.language)[1]*100).toFixed(0))+`% `,3)):(H(),U(`span`,_d,`-`))]),W(`td`,vd,O(e.word_count.toLocaleString()),1),W(`td`,yd,O(g(e.crawled_at)),1)]))),128)),h.value.length?G(``,!0):(H(),U(`tr`,bd,[...t[10]||=[W(`td`,{colspan:`5`,class:`px-5 py-12 text-center text-gray-600`},` 没有找到匹配的记录 `,-1)]]))])]),W(`div`,xd,[(H(!0),U(B,null,mr(h.value,e=>(H(),U(`div`,{key:e.url,class:`block p-4 hover:bg-gray-800/40 transition-colors`},[W(`a`,{href:e.url,target:`_blank`,rel:`noopener noreferrer`,class:`block`},[W(`div`,Cd,O(e.title||`(无标题)`),1),W(`div`,wd,O(e.url),1)],8,Sd),W(`div`,Td,[W(`span`,Ed,O(e.domain),1),v(e.language)?(H(),U(`span`,{key:0,class:D([`px-1.5 py-0.5 rounded font-medium`,_(v(e.language)[0]).cls])},O(_(v(e.language)[0]).label),3)):G(``,!0),W(`span`,Dd,O(g(e.crawled_at)),1)]),W(`button`,{onClick:mo(t=>y(e.url),[`prevent`]),class:`text-xs text-blue-400 hover:text-blue-300 flex items-center gap-1 mb-2`},[W(`span`,null,O(l.value.has(e.url)?`▼`:`▶`)+` 关键词`,1),d.value.has(e.url)?(H(),U(`span`,kd,`加载中...`)):G(``,!0)],8,Od),l.value.has(e.url)?(H(),U(`div`,Ad,[u.value[e.url]?.length?(H(),U(B,{key:0},[W(`div`,jd,[(H(!0),U(B,null,mr(u.value[e.url],e=>(H(),U(`span`,{key:e.word,class:`text-[10px] px-1.5 py-0.5 rounded bg-gray-800 text-gray-300 border border-gray-700`,title:`权重: ${e.weight.toFixed(4)}`},[Ui(O(e.word)+` `,1),W(`span`,Nd,O(e.weight.toFixed(2)),1)],8,Md))),128))]),W(`div`,Pd,` 共 `+O(u.value[e.url].length)+` 个 `,1)],64)):d.value.has(e.url)?G(``,!0):(H(),U(`span`,Fd,` 暂无关键词 `))])):G(``,!0)]))),128)),h.value.length?G(``,!0):(H(),U(`div`,Id,` 没有找到匹配的记录 `))])])),!r.value&&!i.value&&h.value.length?(H(),U(`div`,Ld,` 筛选后 `+O(h.value.length)+` 条 / 共 `+O(n.value.toLocaleString())+` 条 `,1)):G(``,!0)]))}},zd={class:`flex flex-col h-full`},Bd={class:`bg-gray-950 border-b border-gray-800 px-4 md:px-8 py-4 md:py-6`},Vd={class:`max-w-3xl mx-auto`},Hd={class:`relative`},Ud={class:`flex-1 overflow-y-auto px-8 py-6`},Wd={class:`max-w-3xl mx-auto`},Gd={key:0,class:`flex items-center gap-3 text-gray-400 py-8`},Kd={key:1,class:`bg-red-900/30 border border-red-800 rounded-lg p-4 text-red-300`},qd={key:2,class:`py-16 text-center text-gray-600`},Jd={key:3,class:`py-16 text-center text-gray-600`},Yd={key:4,class:`flex flex-col sm:flex-row sm:items-center gap-2 sm:gap-4 mb-4 md:mb-5 text-sm text-gray-500`},Xd={class:`flex items-center gap-2`},Zd={class:`text-gray-300`},Qd={class:`flex flex-wrap gap-2 sm:ml-auto`},$d={class:`text-gray-300`},ef={key:5,class:`space-y-2 md:space-y-1`},tf=[`onClick`],nf={class:`flex items-start gap-2 md:gap-3 mb-2`},rf={class:`flex-1 min-w-0`},af={class:`text-blue-400 group-hover:text-blue-300 text-base md:text-lg leading-snug`},of={class:`text-xs text-gray-600 mt-0.5 truncate`},sf={class:`flex flex-col items-end gap-1 shrink-0`},cf={class:`text-xs text-gray-600`},lf={class:`w-12 md:w-14 bg-gray-800 rounded-full h-1.5 overflow-hidden`},uf={class:`text-gray-400 text-sm leading-relaxed mb-2 md:mb-3`},df={class:`flex flex-wrap items-center gap-2 md:gap-3 text-xs`},ff={class:`flex flex-wrap gap-1.5`},pf=[`title`],mf={class:`text-blue-400`},hf={class:`text-gray-600 ml-1`},gf={key:0,class:`text-gray-600`},_f={class:`text-gray-700 ml-0 md:ml-auto`},vf={class:`text-xs text-gray-700 mt-1 truncate`},yf={key:6,class:`mt-6 text-center`},bf=``,xf=10,Sf={__name:`SearchView`,setup(e){let t=F(``),n=F([]),r=F(0),i=F({}),a=F(!1),o=F(null),s=F(0),c=null;async function l(e,t=!1){if(!e.trim()){n.value=[],r.value=0,i.value={};return}t||(s.value=0),a.value=!0,o.value=null;try{let a=encodeURIComponent(e),o=s.value*xf,c=await fetch(`${bf}/search?qh=${a}&slice=${o}:${o+xf}`);if(!c.ok)throw Error(`HTTP ${c.status}`);let l=await c.json();t?n.value=[...n.value,...l.results||[]]:n.value=l.results||[],r.value=l.total||0,i.value=l.counts||{}}catch(e){o.value=`搜索失败,可能是人服务器未启动`,console.error(e)}finally{a.value=!1}}function u(){clearTimeout(c),c=setTimeout(()=>l(t.value),400)}function d(e){e.key===`Enter`&&(clearTimeout(c),l(t.value))}function f(e,t=120){return e?e.length>t?e.slice(0,t)+`…`:e:``}function p(e){return e?e.toLocaleString():`0`}function m(e){return Math.min(100,Math.round(e*100))}function h(e){return{zh:`中文`,en:`EN`,ja:`JA`,ko:`KO`,fr:`FR`,de:`DE`,es:`ES`,ru:`RU`}[e]||e?.toUpperCase()}function g(e){try{return new URL(e).hostname}catch{return e}}function _(e){window.open(e,`_blank`)}return(e,c)=>(H(),U(`div`,zd,[W(`div`,Bd,[W(`div`,Vd,[W(`div`,Hd,[An(W(`input`,{"onUpdate:modelValue":c[0]||=e=>t.value=e,onInput:u,onKeydown:d,type:`text`,placeholder:`输入关键词搜索,或用 site:example.com 限定域名`,class:`w-full bg-gray-900 border border-gray-700 rounded-xl md:rounded-2xl px-4 md:px-6 py-3 md:py-4 pr-20 md:pr-14 text-white text-base md:text-lg placeholder-gray-600 focus:outline-none focus:border-blue-500 focus:ring-1 focus:ring-blue-500 transition`},null,544),[[so,t.value]]),W(`button`,{onClick:c[1]||=e=>l(t.value),class:`absolute right-2 md:right-3 top-1/2 -translate-y-1/2 bg-blue-600 hover:bg-blue-500 text-white rounded-lg md:rounded-xl px-3 md:px-4 py-1.5 md:py-2 text-sm font-medium transition`},` 搜索 `)])])]),W(`div`,Ud,[W(`div`,Wd,[a.value?(H(),U(`div`,Gd,[...c[3]||=[W(`div`,{class:`w-5 h-5 border-2 border-blue-400 border-t-transparent rounded-full animate-spin`},null,-1),W(`span`,null,`搜索中...`,-1)]])):o.value?(H(),U(`div`,Kd,O(o.value),1)):t.value.trim()?n.value.length===0&&!a.value?(H(),U(`div`,Jd,` 未找到相关结果 `)):n.value.length>0?(H(),U(`div`,Yd,[W(`div`,Xd,[W(`span`,null,[c[4]||=Ui(`找到约 `,-1),W(`strong`,Zd,O(p(r.value)),1),c[5]||=Ui(` 条结果`,-1)]),c[6]||=W(`span`,{class:`text-gray-700`},`|`,-1),W(`span`,null,O(n.value.length)+` 条已加载`,1)]),W(`div`,Qd,[(H(!0),U(B,null,mr(i.value,(e,t)=>(H(),U(`span`,{key:t,class:`inline-flex items-center gap-1 bg-gray-800 rounded px-2 py-0.5 text-xs text-gray-400`},[W(`span`,$d,O(t),1),W(`span`,null,O(p(e)),1)]))),128))])])):G(``,!0):(H(),U(`div`,qd,` 输入关键词开始搜索 `)),n.value.length>0?(H(),U(`div`,ef,[(H(!0),U(B,null,mr(n.value,(e,t)=>(H(),U(`div`,{key:t,onClick:t=>_(e.url),class:`group block bg-gray-900/50 hover:bg-gray-900 border border-gray-800 hover:border-gray-700 rounded-xl p-4 md:p-5 cursor-pointer transition`},[W(`div`,nf,[W(`div`,rf,[W(`div`,af,O(e.snippet?.title||g(e.url)),1),W(`div`,of,O(g(e.url)),1)]),W(`div`,sf,[W(`div`,cf,O(m(e.score))+`%`,1),W(`div`,lf,[W(`div`,{class:`h-full bg-blue-500 rounded-full`,style:me({width:m(e.score)+`%`})},null,4)])])]),W(`p`,uf,O(f(e.snippet?.description||e.snippet?.text)),1),W(`div`,df,[W(`div`,ff,[(H(!0),U(B,null,mr(e.relevance,(e,t)=>(H(),U(`span`,{key:t,class:`inline-flex items-center bg-gray-800 rounded px-1.5 py-0.5`,title:`${t}: ${(e*100).toFixed(1)}%`},[W(`span`,mf,O(t),1),W(`span`,hf,O((e*100).toFixed(0))+`%`,1)],8,pf))),128))]),e.snippet?.language?(H(),U(`span`,gf,O(h(e.snippet.language)),1)):G(``,!0),W(`span`,_f,O(p(e.domain_count))+` 个结果`,1)]),W(`div`,vf,O(e.url),1)],8,tf))),128))])):G(``,!0),n.value.length>0&&n.value.length{s.value++,l(t.value,!0)},class:`bg-gray-800 hover:bg-gray-700 text-gray-300 rounded-xl px-6 py-2.5 text-sm transition`},` 加载更多 (`+O(n.value.length)+`/`+O(p(r.value))+`) `,1)])):G(``,!0)])])]))}},Cf={class:`flex h-screen bg-gray-950 text-gray-100 font-sans`},wf={class:`hidden md:flex w-56 bg-gray-900 border-r border-gray-800 flex-col shrink-0`},Tf={class:`flex-1 py-4 px-3 space-y-1`},Ef=[`onClick`],Df={class:`flex-1 overflow-y-auto pb-16 md:pb-0`},Of={class:`md:hidden fixed bottom-0 left-0 right-0 bg-gray-900 border-t border-gray-800 flex justify-around items-center h-16 z-50`},kf=[`onClick`],Af={class:`text-lg`},jf={class:`text-[10px]`};bo({__name:`App`,setup(e){let t=F(`dashboard`),n=[{id:`dashboard`,label:`概览`,icon:`📊`},{id:`recent`,label:`最近`,icon:`🕷️`},{id:`search`,label:`搜索`,icon:`🔍`}];return(e,r)=>(H(),U(`div`,Cf,[W(`aside`,wf,[r[0]||=W(`div`,{class:`px-5 py-5 border-b border-gray-800`},[W(`div`,{class:`text-lg font-semibold text-white tracking-tight`},`SESE Admin`),W(`div`,{class:`text-xs text-gray-500 mt-0.5`},`爬取内容监控`)],-1),W(`nav`,Tf,[(H(),U(B,null,mr(n,e=>W(`button`,{key:e.id,onClick:n=>t.value=e.id,class:D([`w-full flex items-center gap-3 px-3 py-2.5 rounded-lg text-sm font-medium transition-colors`,t.value===e.id?`bg-blue-600 text-white`:`text-gray-400 hover:text-white hover:bg-gray-800`])},[W(`span`,null,O(e.icon),1),Ui(` `+O(e.label),1)],10,Ef)),64))]),r[1]||=W(`div`,{class:`px-5 py-4 border-t border-gray-800`},[W(`div`,{class:`text-xs text-gray-600`},`sese-engine v1.0`)],-1)]),W(`main`,Df,[t.value===`dashboard`?(H(),Pi(Iu,{key:0})):t.value===`recent`?(H(),Pi(Rd,{key:1})):t.value===`search`?(H(),Pi(Sf,{key:2})):G(``,!0)]),W(`nav`,Of,[(H(),U(B,null,mr(n,e=>W(`button`,{key:e.id,onClick:n=>t.value=e.id,class:D([`flex flex-col items-center justify-center gap-0.5 flex-1 h-full transition-colors`,t.value===e.id?`text-blue-400`:`text-gray-500`])},[W(`span`,Af,O(e.icon),1),W(`span`,jf,O(e.label),1)],10,kf)),64))])]))}}).mount(`#app`); \ No newline at end of file diff --git a/dist/index.html b/dist/index.html index bf9bb03..91b09d9 100644 --- a/dist/index.html +++ b/dist/index.html @@ -5,7 +5,7 @@ SESE 爬取管理 - + diff --git a/sese-engine-ui b/sese-engine-ui index 63c40d6..558d91b 160000 --- a/sese-engine-ui +++ b/sese-engine-ui @@ -1 +1 @@ -Subproject commit 63c40d69d4b44d330ef718fa992e429d081ca118 +Subproject commit 558d91b0a438762288407ff636d10ac36b84151f