Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Adds generic search thumbnail retrieval #10266

Merged
merged 6 commits into from
Nov 28, 2023
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -189,16 +189,22 @@ function($, _, BaseFilter, bootstrap, arches, select2, ko, koMapping, GraphModel
return acc;
}, []);

this.searchResults.results.hits.hits.forEach(function(result){
this.searchResults.results.hits.hits.forEach(async function(result){
var graphdata = _.find(viewdata.graphs, function(graphdata){
return result._source.graph_id === graphdata.graphid;
});
var point = null;
if (result._source.points.length > 0) {
point = result._source.points[0].point;
}

const thumbnailUrl = `/thumbnail/${result._source.resourceinstanceid}`;
const thumbnailResponse = arches.searchThumbnails == 'True' ? await fetch(thumbnailUrl, {method: 'HEAD'}): undefined;
const thumbnail = thumbnailResponse && thumbnailResponse.ok ? thumbnailUrl: undefined;

this.results.push({
displayname: result._source.displayname,
thumbnail: thumbnail,
resourceinstanceid: result._source.resourceinstanceid,
displaydescription: result._source.displaydescription,
alternativelanguage: result._source.displayname_language != arches.activeLanguage,
Expand Down
1 change: 1 addition & 0 deletions arches/app/templates/javascript.htm
Original file line number Diff line number Diff line change
Expand Up @@ -989,6 +989,7 @@
active-language="{{ app_settings.ACTIVE_LANGUAGE }}"
active-language-dir="{{ app_settings.ACTIVE_LANGUAGE_DIR }}"
languages='{{ app_settings.LANGUAGES }}'
search-thumbnails='{{ app_settings.SEARCH_THUMBNAILS }}'
mapbox-api-key="{{ map_info.mapbox_api_key }}"
mapbox-glyphs="{{ map_info.mapbox_glyphs }}"
mapbox-sprites="{{ map_info.mapbox_sprites }}"
Expand Down
27 changes: 16 additions & 11 deletions arches/app/templates/views/components/search/search-results.htm
Original file line number Diff line number Diff line change
Expand Up @@ -5,19 +5,24 @@
<div id="search-results-list" data-bind="foreach: results, visible: true" style="display: none;">

<div class="search-listing" data-bind="event: { mouseover: mouseoverInstance, mouseout: mouseoverInstance('')}, css: {'selected': selected()}">
<h3
class="search-listing-title"
data-bind="css: {'i18n-alt': $parent.alternativelanguage}"
>
<a class="search-candidate-title" href="" data-bind="click: $parent.viewReport.bind($parent)">
<i class="search-listing-icon" data-bind="css: iconclass"></i>
<span data-bind="text: displayname"></span>
</a>
</h3>
<div style="display: flex">
<div style="flex: 1">
<h3
class="search-listing-title"
data-bind="css: {'i18n-alt': $parent.alternativelanguage}"
>
<a class="search-candidate-title" href="" data-bind="click: $parent.viewReport.bind($parent)">
<i class="search-listing-icon" data-bind="css: iconclass"></i>
<span data-bind="text: displayname"></span>
</a>
</h3>

<div class="search-listing-body" data-bind="html: displaydescription">
<div class="search-listing-body" data-bind="html: displaydescription">

</div>
</div>
<div style="display: flex; align-items: center; margin: 10px;" data-bind="visible: thumbnail"><img style="max-width: 60px;" data-bind="attr:{src: thumbnail}"></div>
</div>

<div class="search-listing-footer">
<div style="flex-grow: 1;">
<!--ko if: provisional_resource == 'true' -->
Expand Down
1 change: 1 addition & 0 deletions arches/app/utils/context_processors.py
Original file line number Diff line number Diff line change
Expand Up @@ -73,6 +73,7 @@ def app_settings(request=None):
"VERSION": __version__,
"APP_VERSION": settings.APP_VERSION,
"APP_NAME": settings.APP_NAME,
"SEARCH_THUMBNAILS": settings.SEARCH_THUMBNAILS,
"GOOGLE_ANALYTICS_TRACKING_ID": settings.GOOGLE_ANALYTICS_TRACKING_ID,
"USE_SEMANTIC_RESOURCE_RELATIONSHIPS": settings.USE_SEMANTIC_RESOURCE_RELATIONSHIPS,
"SEARCH_EXPORT_IMMEDIATE_DOWNLOAD_THRESHOLD": settings.SEARCH_EXPORT_IMMEDIATE_DOWNLOAD_THRESHOLD,
Expand Down
16 changes: 16 additions & 0 deletions arches/app/utils/search_thumbnail_fetcher.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
import base64
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It's nice to have an example, but because this file isn't yet wired in as a default, I think it would be better to remove this here and add it to the docs.

import copy
import re
from typing import List, Tuple
from urllib.error import HTTPError
from requests.exceptions import ConnectionError
import requests
from urllib.parse import urlparse, urlunparse

class SearchThumbnailFetcher(object):
def __init__(self, resource):
self.resource = resource

def get_thumbnail(self, retrieve=False):
pass

39 changes: 39 additions & 0 deletions arches/app/utils/search_thumbnail_fetcher_factory.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,39 @@
from logging import Logger
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Logger is unused

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

it is used on line 17

Copy link
Member

@chiatt chiatt Nov 15, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

On 17 you are using 'logging' which is defined on line 8 imported from line 2. Logger itself is unused. I wouldn't have caught it myself had vscode not flagged it.

import logging
from typing import Callable
from arches.app.models.resource import Resource

from arches.app.utils.search_thumbnail_fetcher import SearchThumbnailFetcher

logger = logging.getLogger(__name__)

class SearchThumbnailFetcherFactory(object):
registry = {}

@classmethod
def register(cls, name:str):
def inner_wrapper(wrapped_class: SearchThumbnailFetcher) -> Callable:
if name in cls.registry:
logger.warning('Search Thumbnail Fetcher %s already exists. Will replace it', name)
cls.registry[name] = wrapped_class
return wrapped_class

return inner_wrapper

@classmethod
def create_thumbnail_fetcher(cls, resource_id: str, **kwargs) -> SearchThumbnailFetcher:
""" Factory command to create the template engine """
resource = Resource.objects.get(resourceinstanceid=resource_id)
try:
search_thumbnail_fetcher_class = cls.registry[str(resource.graph_id)]
search_thumbnail_fetcher = search_thumbnail_fetcher_class(resource, **kwargs)
return search_thumbnail_fetcher
except KeyError:
pass # there is no thumbnail fetcher registered for the graph requested

try:
search_thumbnail_fetcher_class = cls.registry['default']
search_thumbnail_fetcher = search_thumbnail_fetcher_class(resource, **kwargs)
return search_thumbnail_fetcher
except KeyError:
return None # there is no default thumbnail fetcher registered, return false.
34 changes: 34 additions & 0 deletions arches/app/views/thumbnail.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,34 @@
from arches.app.models.resource import Resource
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Remove unused import

from django.views.generic import View
from django.http import HttpResponse, HttpResponseNotFound
from django.utils.translation import get_language
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

unused import

import requests
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

unused import

from arches.app.utils.search_thumbnail_fetcher_factory import SearchThumbnailFetcherFactory

class ThumbnailView(View):

def head(self, request, resource_id):
fetcher = self.get_thumbnail_fetcher(resource_id)
if fetcher is None:
return HttpResponseNotFound()
thumbnail = fetcher.get_thumbnail(False)
if thumbnail is not None:
return HttpResponse()
else:
return HttpResponseNotFound()

def get(self, request, resource_id):
fetcher = self.get_thumbnail_fetcher(resource_id)
if fetcher is None:
return HttpResponseNotFound()

thumbnail = fetcher.get_thumbnail(True)
if thumbnail is not None:
return HttpResponse(thumbnail[0], content_type=thumbnail[1])
else:
return HttpResponseNotFound()

def get_thumbnail_fetcher(self, resource_id):
factory = SearchThumbnailFetcherFactory()
fetcher = factory.create_thumbnail_fetcher(resource_id)
return fetcher
1 change: 1 addition & 0 deletions arches/settings.py
Original file line number Diff line number Diff line change
Expand Up @@ -66,6 +66,7 @@

ELASTICSEARCH_HTTP_PORT = 9200 # this should be in increments of 200, eg: 9400, 9600, 9800
SEARCH_BACKEND = "arches.app.search.search.SearchEngine"
SEARCH_THUMBNAILS = False
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This should also be added to the project template's settings file

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

done

# see http://elasticsearch-py.readthedocs.org/en/master/api.html#elasticsearch.Elasticsearch
ELASTICSEARCH_HOSTS = [{"scheme": "https", "host": "localhost", "port": ELASTICSEARCH_HTTP_PORT}]

Expand Down
2 changes: 2 additions & 0 deletions arches/urls.py
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@
from arches.app.views.admin import ReIndexResources, ClearUserPermissionCache
from arches.app.views.etl_manager import ETLManagerView
from arches.app.views.file import FileView, TempFileView
from arches.app.views.thumbnail import ThumbnailView
from arches.app.views.graph import (
GraphDesignerView,
GraphSettingsView,
Expand Down Expand Up @@ -277,6 +278,7 @@
re_path(r"^history/$", ResourceActivityStreamCollectionView.as_view(), name="as_stream_collection"),
re_path(r"^history/(?P<page>[0-9]+)$", ResourceActivityStreamPageView.as_view(), name="as_stream_page"),
re_path(r"^icons$", IconDataView.as_view(), name="icons"),
re_path(r"^thumbnail/(?P<resource_id>%s)$" % uuid_regex, ThumbnailView.as_view(), name="thumbnail"),
# Uncomment the admin/doc line below to enable admin documentation:
# re_path(r'^admin/doc/', include('django.contrib.admindocs.urls')),
# Uncomment the next line to enable the admin:
Expand Down
Loading