Add search

This commit is contained in:
2022-12-10 17:15:11 +01:00
parent 6d028919b7
commit 5ea85d9448
5 changed files with 82 additions and 2 deletions

View File

@ -63,3 +63,5 @@ def send_to_discord(feed=None) -> None:
response: Response = webhook.execute()
if not response.ok:
reader.set_entry_read(entry, False)
reader.update_search()

View File

@ -37,7 +37,7 @@ from fastapi import FastAPI, Form, HTTPException, Request
from fastapi.responses import FileResponse, HTMLResponse
from fastapi.staticfiles import StaticFiles
from fastapi.templating import Jinja2Templates
from reader import Entry, EntryCounts, Feed, FeedCounts
from reader import Entry, EntryCounts, EntrySearchResult, Feed, FeedCounts, HighlightedString
from starlette.templating import _TemplateResponse
from tomlkit.toml_document import TOMLDocument
@ -100,6 +100,8 @@ async def create_feed(feed_url: str = Form(), webhook_dropdown: str = Form()) ->
reader.set_tag(feed_url, "webhook", webhook_url)
reader.get_tag(feed_url, "webhook")
reader.update_search()
# TODO: Go to the feed page.
return {"feed_url": str(feed_url), "status": "added"}
@ -224,11 +226,70 @@ async def remove_feed(request: Request, feed_url: str = Form()):
"""
reader.delete_feed(feed_url)
reader.update_search()
context = make_context_index(request)
return templates.TemplateResponse("index.html", context)
@app.get("/search", response_class=HTMLResponse)
async def search(request: Request, query: str) -> _TemplateResponse:
"""
Get entries matching a full-text search query.
Args:
request: The request.
query: The query to search for.
Returns:
HTMLResponse: The HTML response.
"""
reader.update_search()
search_results = reader.search_entries(query)
search_amount = reader.search_entry_counts(query)
def add_span_with_slice(highlighted_string: HighlightedString):
"""Add a span with the highlighted string."""
for txt_slice in highlighted_string.highlights:
first = f"{highlighted_string.value[: txt_slice.start]}"
second = f"<span class='bg-warning'>{highlighted_string.value[txt_slice.start: txt_slice.stop]}</span>"
third = f"{highlighted_string.value[txt_slice.stop:]}"
return f"{first}{second}{third}"
def create_html_for_search_results(search_results: Iterable[EntrySearchResult]) -> str:
"""Create HTML for the search results.
Args:
search_results: The search results.
Returns:
str: The HTML.
"""
html = ""
for result in search_results:
if ".summary" in result.content:
result_summary = add_span_with_slice(result.content[".summary"])
feed = reader.get_feed(result.feed_url)
feed_url = encode_url(feed.url)
html += f"""
<a class="text-muted" href="/feed?feed_url={feed_url}">
<h2>{result.metadata[".title"]}</h2>
</a>
<blockquote>
{result_summary}
</blockquote>
<hr>
"""
return html
search_html = create_html_for_search_results(search_results)
return templates.TemplateResponse(
"search.html", {"request": request, "search_html": search_html, "query": query, "search_amount": search_amount}
)
@app.on_event("startup")
def startup() -> None:
"""This is called when the server starts.

View File

@ -85,7 +85,6 @@
</div>
</div>
{% endif %}
{% endfor %}
</div>
{% endblock %}

View File

@ -13,6 +13,14 @@
<a class="nav-link" href="/add">Add new feed</a>
</li>
</ul>
{# Search #}
<form action="/search" method="get" class="d-flex col-sm-7 ms-auto">
<input name="query" class="form-control me-1" type="search" placeholder="Search">
<button class="btn btn-outline-warning" type="submit">Search</button>
</form>
{# Donate button #}
<ul class="navbar-nav ms-auto">
<li class="nav-item">
<a class="nav-link" href="https://github.com/sponsors/TheLovinator1">Donate ❤️</a>

View File

@ -0,0 +1,10 @@
{% extends "base.html" %}
{% block title %} | Search{% endblock %}
{% block content %}
<div class="p-2 border border-dark text-muted" style="background:#0F0F0F">
Your search for "{{ query }}" returned {{ search_amount.total }} results.
</div>
<div class="p-2 border border-dark text-muted" style="background:#0F0F0F">
{{ search_html | safe }}
</div>
{% endblock %}