Skip to content

Commit

Permalink
Merge pull request #202 from Olog/CSSTUDIO-1257
Browse files Browse the repository at this point in the history
Support for log templates
  • Loading branch information
georgweiss authored Nov 27, 2024
2 parents 4b9a445 + 94003c0 commit 41929e0
Show file tree
Hide file tree
Showing 22 changed files with 1,490 additions and 164 deletions.
158 changes: 115 additions & 43 deletions src/main/java/org/phoebus/olog/ElasticConfig.java

Large diffs are not rendered by default.

25 changes: 10 additions & 15 deletions src/main/java/org/phoebus/olog/LogRepository.java
Original file line number Diff line number Diff line change
Expand Up @@ -29,7 +29,6 @@
import org.phoebus.olog.entity.SearchResult;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.data.repository.CrudRepository;
import org.springframework.http.HttpStatus;
import org.springframework.stereotype.Repository;
Expand All @@ -48,18 +47,14 @@
import java.util.logging.Logger;
import java.util.stream.Collectors;

import static org.phoebus.olog.ElasticConfig.ES_LOG_INDEX;
import static org.phoebus.olog.ElasticConfig.ES_LOG_ARCHIVE_INDEX;

@Repository
public class LogRepository implements CrudRepository<Log, String> {

private static final Logger logger = Logger.getLogger(LogRepository.class.getName());

@SuppressWarnings("unused")
@Value("${elasticsearch.log.index:olog_logs}")
private String ES_LOG_INDEX;

@Value("${elasticsearch.log.archive.index:olog_archived_logs}")
private String ES_LOG_ARCHIVE_INDEX;

@SuppressWarnings("unused")
@Autowired
@Qualifier("client")
Expand All @@ -79,7 +74,7 @@ public <S extends Log> S save(S log) {
if (log.getAttachments() != null && !log.getAttachments().isEmpty()) {
Set<Attachment> createdAttachments = new HashSet<>();
log.getAttachments().stream().filter(attachment -> attachment.getAttachment() != null).forEach(attachment ->
createdAttachments.add(attachmentRepository.save(attachment))
createdAttachments.add(attachmentRepository.save(attachment))
);
validatedLog = validatedLog.setAttachments(createdAttachments);
}
Expand Down Expand Up @@ -114,7 +109,7 @@ public <S extends Log> S save(S log) {
public <S extends Log> Iterable<S> saveAll(Iterable<S> logs) {
List<S> createdLogs = new ArrayList<>();
logs.forEach(log ->
createdLogs.add(save(log))
createdLogs.add(save(log))
);
return createdLogs;
}
Expand All @@ -124,7 +119,7 @@ public Log update(Log log) {
Log document = LogBuilder.createLog(log).build();
IndexRequest<Log> indexRequest =
IndexRequest.of(i ->
i.index(ES_LOG_INDEX)
i.index(ES_LOG_INDEX)
.id(String.valueOf(document.getId()))
.refresh(Refresh.True)
.document(document));
Expand Down Expand Up @@ -152,7 +147,7 @@ public Log archive(Log log) {
// retrieve the log version from elastic
GetResponse<Log> resp = client.get(GetRequest.of(g ->
g.index(ES_LOG_INDEX).id(String.valueOf(log.getId()))), Log.class);
if(!resp.found()) {
if (!resp.found()) {
logger.log(Level.SEVERE, () -> MessageFormat.format(TextUtil.LOG_NOT_ARCHIVED, log.getId()));
} else {
Log originalDocument = resp.source();
Expand Down Expand Up @@ -185,9 +180,9 @@ public SearchResult findArchivedById(String id) {
fb.order(SortOrder.Desc);

SearchRequest searchRequest = SearchRequest.of(s -> s.index(ES_LOG_ARCHIVE_INDEX)
.query(WildcardQuery.of(q -> q.field("id").caseInsensitive(true).value(id+"*"))._toQuery())
.timeout("60s")
.sort(SortOptions.of(so -> so.field(fb.build()))));
.query(WildcardQuery.of(q -> q.field("id").caseInsensitive(true).value(id + "*"))._toQuery())
.timeout("60s")
.sort(SortOptions.of(so -> so.field(fb.build()))));
try {
final SearchResponse<Log> searchResponse = client.search(searchRequest, Log.class);
List<Log> result = searchResponse.hits().hits().stream().map(Hit::source).collect(Collectors.toList());
Expand Down
1 change: 0 additions & 1 deletion src/main/java/org/phoebus/olog/LogResource.java
Original file line number Diff line number Diff line change
Expand Up @@ -211,7 +211,6 @@ public SearchResult search(@RequestHeader(value = OLOG_CLIENT_INFO_HEADER, requi
* @return The persisted {@link Log} object.
*/
@PutMapping()
@Deprecated
public Log createLog(@RequestHeader(value = OLOG_CLIENT_INFO_HEADER, required = false, defaultValue = "n/a") String clientInfo,
@RequestParam(value = "markup", required = false) String markup,
@RequestParam(value = "inReplyTo", required = false, defaultValue = "-1") String inReplyTo,
Expand Down
213 changes: 213 additions & 0 deletions src/main/java/org/phoebus/olog/LogTemplateRepository.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,213 @@
/*
* Copyright (c) 2010-2020 Brookhaven National Laboratory
* Copyright (c) 2010-2020 Helmholtz-Zentrum Berlin für Materialien und Energie GmbH
* All rights reserved. Use is subject to license terms and conditions.
*/
package org.phoebus.olog;

import co.elastic.clients.elasticsearch.ElasticsearchClient;
import co.elastic.clients.elasticsearch._types.FieldSort;
import co.elastic.clients.elasticsearch._types.Refresh;
import co.elastic.clients.elasticsearch._types.Result;
import co.elastic.clients.elasticsearch._types.SortOptions;
import co.elastic.clients.elasticsearch._types.query_dsl.MatchAllQuery;
import co.elastic.clients.elasticsearch.core.DeleteRequest;
import co.elastic.clients.elasticsearch.core.DeleteResponse;
import co.elastic.clients.elasticsearch.core.GetRequest;
import co.elastic.clients.elasticsearch.core.GetResponse;
import co.elastic.clients.elasticsearch.core.IndexRequest;
import co.elastic.clients.elasticsearch.core.IndexResponse;
import co.elastic.clients.elasticsearch.core.SearchRequest;
import co.elastic.clients.elasticsearch.core.SearchResponse;
import org.phoebus.olog.entity.LogTemplate;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.data.repository.CrudRepository;
import org.springframework.http.HttpStatus;
import org.springframework.stereotype.Repository;
import org.springframework.web.server.ResponseStatusException;

import java.io.IOException;
import java.text.MessageFormat;
import java.time.Instant;
import java.util.ArrayList;
import java.util.List;
import java.util.Optional;
import java.util.logging.Level;
import java.util.logging.Logger;

/**
* {@link org.springframework.data.repository.Repository} for {@link LogTemplate}s.
*/
@Repository
public class LogTemplateRepository implements CrudRepository<LogTemplate, String> {

private static final Logger logger = Logger.getLogger(LogTemplateRepository.class.getName());

@SuppressWarnings("unused")
@Autowired
@Qualifier("client")
ElasticsearchClient client;

@Override
public <S extends LogTemplate> S save(S logTemplate) {

try {
logTemplate.setCreatedDate(Instant.now());
IndexRequest<Object> indexRequest =
IndexRequest.of(i ->
i.index(ElasticConfig.ES_LOG_TEMPLATE_INDEX)
.document(logTemplate)
.refresh(Refresh.True));
IndexResponse response = client.index(indexRequest);

if (response.result().equals(Result.Created)) {
GetRequest getRequest =
GetRequest.of(g ->
g.index(ElasticConfig.ES_LOG_TEMPLATE_INDEX).id(response.id()));
GetResponse<LogTemplate> resp =
client.get(getRequest, LogTemplate.class);
LogTemplate saved = resp.source();
saved.setId(response.id());
return (S) saved;
}
} catch (Exception e) {
String message = MessageFormat.format(TextUtil.LOG_TEMPLATE_NOT_SAVED, logTemplate.getName());
logger.log(Level.SEVERE, message, e);
throw new ResponseStatusException(HttpStatus.INTERNAL_SERVER_ERROR, message);
}
return null;
}

@Override
public <S extends LogTemplate> Iterable<S> saveAll(Iterable<S> logTemplates) {
throw new ResponseStatusException(HttpStatus.NOT_FOUND, TextUtil.LOG_TEMPLATES_SAVE_ALL);
}

public LogTemplate update(LogTemplate logTemplate) {
try {
logTemplate.setModifyDate(Instant.now());
IndexRequest<LogTemplate> indexRequest =
IndexRequest.of(i ->
i.index(ElasticConfig.ES_LOG_TEMPLATE_INDEX)
.refresh(Refresh.True)
.id(logTemplate.getId())
.document(logTemplate));

IndexResponse response = client.index(indexRequest);

if (response.result().equals(Result.Updated)) {
GetRequest getRequest =
GetRequest.of(g ->
g.index(ElasticConfig.ES_LOG_TEMPLATE_INDEX).id(response.id()));
GetResponse<LogTemplate> resp =
client.get(getRequest, LogTemplate.class);
return resp.source();
}
} catch (Exception e) {
String message = MessageFormat.format(TextUtil.LOG_TEMPLATE_NOT_UPDATED, logTemplate.getName());
logger.log(Level.SEVERE, message, e);
throw new ResponseStatusException(HttpStatus.INTERNAL_SERVER_ERROR, message);
}
return null;
}

@Override
public Optional<LogTemplate> findById(String id) {
try {
GetRequest getRequest =
GetRequest.of(g ->
g.index(ElasticConfig.ES_LOG_TEMPLATE_INDEX).id(id));
GetResponse<LogTemplate> resp =
client.get(getRequest, LogTemplate.class);

if (!resp.found()) {
throw new ResponseStatusException(HttpStatus.NOT_FOUND, MessageFormat.format(TextUtil.LOG_TEMPLATE_NOT_FOUND, id));
}
return resp.source() != null ? Optional.of(resp.source()) : Optional.empty();
} catch (Exception e) {
// https://www.baeldung.com/exception-handling-for-rest-with-spring#controlleradvice
String message = MessageFormat.format(TextUtil.LOG_TEMPLATE_NOT_RETRIEVED, id);
logger.log(Level.SEVERE, message, e);
throw new ResponseStatusException(HttpStatus.NOT_FOUND, message);
}
}

@Override
public boolean existsById(String logId) {
throw new ResponseStatusException(HttpStatus.NOT_FOUND, TextUtil.LOG_TEMPLATE_EXISTS_UNSUPPORTED);
}

@Override
public Iterable<LogTemplate> findAll() {
try {
SearchRequest searchRequest =
SearchRequest.of(s ->
s.index(ElasticConfig.ES_LOG_TEMPLATE_INDEX)
.query(new MatchAllQuery.Builder().build()._toQuery())
.size(1000) // Assume number of templates is within reason, so returning "all" should be the wanted behavior
.from(0)
.sort(SortOptions.of(so -> so.field(FieldSort.of(f -> f.field("name"))))));

SearchResponse<LogTemplate> response =
client.search(searchRequest, LogTemplate.class);
List<LogTemplate> templates = new ArrayList<>();
response.hits().hits().forEach(h -> {
LogTemplate logTemplate = h.source();
logTemplate.setId(h.id());
templates.add(logTemplate);
});
return templates;
} catch (Exception e) {
logger.log(Level.SEVERE, TextUtil.LOG_TEMPLATES_NOT_RETRIEVED, e);
throw new ResponseStatusException(HttpStatus.INTERNAL_SERVER_ERROR, TextUtil.LOG_TEMPLATES_NOT_RETRIEVED);
}
}

@Override
public Iterable<LogTemplate> findAllById(Iterable<String> logIds) {
throw new ResponseStatusException(HttpStatus.NOT_FOUND, TextUtil.LOG_TEMPLATES_FIND_ALL_BY_ID);
}

@Override
public long count() {
return 0;
}

@Override
public void deleteById(String id) {
try {
DeleteRequest deleteRequest = DeleteRequest.of(d ->
d.index(ElasticConfig.ES_LOG_TEMPLATE_INDEX).id(id).refresh(Refresh.True));
DeleteResponse deleteResponse = client.delete(deleteRequest);
if (deleteResponse.result().equals(Result.Deleted)) {
logger.log(Level.INFO, MessageFormat.format(TextUtil.LOG_TEMPLATE_DELETED, id));
} else {
logger.log(Level.INFO, MessageFormat.format(TextUtil.LOG_TEMPLATE_NOT_DELETED, id));
}
} catch (IOException e) {
logger.log(Level.SEVERE, MessageFormat.format(TextUtil.LOG_TEMPLATE_NOT_DELETED, id), e);
throw new RuntimeException(e);
}
}

@Override
public void delete(LogTemplate logTemplate) {
deleteById(logTemplate.getId());
}

@Override
public void deleteAll(Iterable<? extends LogTemplate> entities) {
throw new ResponseStatusException(HttpStatus.NOT_FOUND, TextUtil.LOG_TEMPLATE_DELETE_ALL_NOT_SUPPORTED);
}

@Override
public void deleteAll() {
throw new ResponseStatusException(HttpStatus.FORBIDDEN, TextUtil.LOG_TEMPLATE_DELETE_ALL_NOT_SUPPORTED);
}

@Override
public void deleteAllById(Iterable<? extends String> ids) {
throw new ResponseStatusException(HttpStatus.NOT_FOUND, TextUtil.LOG_TEMPLATE_DELETE_ALL_NOT_SUPPORTED);
}
}
Loading

0 comments on commit 41929e0

Please sign in to comment.