summaryrefslogtreecommitdiff
path: root/irclog/search
diff options
context:
space:
mode:
authorMistivia <i@mistivia.com>2025-11-04 02:18:30 +0800
committerMistivia <i@mistivia.com>2025-11-04 02:18:30 +0800
commite1ae9516cc1a2e0ed44932bf76256752f5003ae1 (patch)
treed8ad1b128153d1a861aa09acc2f7446643b5d96d /irclog/search
parent848f1214153f7b8af2a243ade2799b4d3fc4e2bc (diff)
add search for irclog
Diffstat (limited to 'irclog/search')
-rw-r--r--irclog/search/index.html62
-rw-r--r--irclog/search/search.js112
2 files changed, 174 insertions, 0 deletions
diff --git a/irclog/search/index.html b/irclog/search/index.html
new file mode 100644
index 0000000..066b30d
--- /dev/null
+++ b/irclog/search/index.html
@@ -0,0 +1,62 @@
+<!DOCTYPE html>
+<html lang="zh">
+<head>
+ <meta charset="UTF-8">
+ <meta name="viewport" content="width=device-width, initial-scale=1.0, viewport-fit=cover">
+ <title>IRC Log Viewer</title>
+ <style>
+ body {
+ font-family: Arial, sans-serif;
+ max-width: 1200px;
+ margin: 20px;
+ background-color: #f4f4f4;
+ }
+ #log-container {
+ border: 1px solid #ccc;
+ padding: 15px;
+ background-color: #fff;
+ white-space: pre-wrap;
+ }
+ pre {
+ font-family: Consolas, monospace;
+ margin: 0;
+ padding: 0;
+ overflow-wrap: anywhere;
+ white-space: pre-wrap;
+ }
+ .error {
+ color: red;
+ font-weight: bold;
+ }
+ .loading {
+ color: blue;
+ }
+
+ @media (max-width: 600px) {
+ pre {
+ font-size: 0.7em;
+ }
+ }
+ #searchlink {
+ visibility: hidden;
+ }
+ a:link,
+ a:visited {
+ color: blue;
+ text-decoration: underline;
+ }
+ </style>
+</head>
+<body>
+
+ <h1>IRC日志检索</h1>
+ <div id="controls">
+ <input type="text" id="query" value="">
+ <button onclick="search()">搜索</button>
+ <br>
+ <br>
+ </div>
+ <div id="log-container"></div>
+ <script src="./search.js"></script>
+</body>
+</html>
diff --git a/irclog/search/search.js b/irclog/search/search.js
new file mode 100644
index 0000000..9a574d6
--- /dev/null
+++ b/irclog/search/search.js
@@ -0,0 +1,112 @@
+const logContainer = document.getElementById('log-container');
+const queryInput = document.getElementById('query');
+
+const queryString = window.location.search;
+const urlParams = new URLSearchParams(queryString);
+let query = urlParams.get('q');
+if (query === null) {
+ query = '';
+} else {
+ queryInput.value = query.trim();
+}
+
+logContainer.innerHTML = '';
+
+function escapeHtml(unsafe) {
+ if (!unsafe) return '';
+ return unsafe
+ .replace(/&/g, "&amp;")
+ .replace(/</g, "&lt;")
+ .replace(/>/g, "&gt;")
+ .replace(/"/g, "&quot;")
+ .replace(/'/g, "&#039;");
+}
+
+function ircAction(text) {
+ const regex =
+ /^(\[[^\]]+\])(\s*<)([^>]+)(>:\s*)(\u0001ACTION\s+)([^\u0001]+)(\u0001)(.*)$/gm;
+ const replacement = '$1 * $3 $6$8';
+ return text.replace(regex, replacement);
+}
+
+function linkify(text) {
+ const urlRegex = /(\b(https?):\/\/[-A-Z0-9+&@#\/%?=~_|!:,.;]*[-A-Z0-9+&@#\/%=~_|])/ig;
+ return text.replace(urlRegex, function(url) {
+ return `<a href="${url}" target="_blank">${url}</a>`;
+ });
+}
+
+function aggrLog(data) {
+ const logsByDate = {};
+ const lines = data.trim().split('\n');
+
+ lines.forEach(line => {
+ const match = line.match(/^\.\/(\d{4}\/\d{2}-\d{2})\.txt:(.*)/);
+ if (match) {
+ const fullDate = match[1].replace('/', '-');
+ const logContent = match[2].trim();
+ if (!logsByDate[fullDate]) {
+ logsByDate[fullDate] = [];
+ }
+ logsByDate[fullDate].push(logContent);
+ }
+ });
+ const dates = Object.keys(logsByDate);
+ const sortedDates = dates.sort().reverse();
+ let ret = '';
+ sortedDates.forEach(date => {
+ year = date.substring(0, 4);
+ month = date.substring(5, 7);
+ day = date.substring(8, 10);
+ url = '../view/?chan=main&y=' + year + '&m=' + month + '&d=' + day;
+ ret = ret + '\n=== <a href=' + url + ' target=\'_blank\'>' + date + '</a> ===' + '\n';
+ logsByDate[date].forEach(log => {
+ ret = ret + log + '\n';
+ });
+ });
+ return ret;
+}
+
+function logProcess(text) {
+ text = ircAction(text);
+ text = escapeHtml(text);
+ text = aggrLog(text);
+ text = linkify(text);
+ return text;
+}
+
+function loadLog() {
+ if (query === '') return;
+ let targetUrl = 'https://raye.mistivia.com/cgi-bin/irclogsearch/?' + encodeURIComponent(query);
+ fetch(targetUrl)
+ .then(response => {
+ if (!response.ok) {
+ throw new Error(`HTTP Error: ${response.status} ${response.statusText} for hash: ${urlHash}`);
+ }
+ return response.text();
+ })
+ .then(text => {
+ logContainer.innerHTML = `<pre>${logProcess(text.trim())}</pre>`;
+ })
+ .catch(error => {
+ console.error('Fetch error:', error);
+ logContainer.innerHTML = '';
+ });
+}
+
+loadLog();
+
+function search() {
+ query = queryInput.value.trim();
+ window.history.replaceState(null, '', window.location.origin + window.location.pathname + '?q=' + encodeURIComponent(query));
+ loadLog();
+}
+
+function handleEnter(event) {
+ if (event.key === 'Enter' || event.keyCode === 13) {
+ event.preventDefault();
+ search();
+ }
+}
+
+queryInput.addEventListener('keyup', handleEnter); \ No newline at end of file