-
Notifications
You must be signed in to change notification settings - Fork 297
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
[OPIK-432] Project Metrics MVP (#678)
* OPIK-432 project metrics traces count failing test [WIP] * OPIK-432 project metrics traces count failing test * OPIK-432 project metrics traces count failing test * OPIK-432 project metrics traces count failing test * OPIK-432 project metrics traces count implementation [WIP] * OPIK-432 project metrics traces count implementation [WIP] * OPIK-432 project metrics traces count implementation [WIP] * OPIK-432 project metrics traces count implementation [WIP] * OPIK-432 project metrics traces count implementation [WIP] * OPIK-432 project metrics traces count failing test green * OPIK-432 validations failing test * OPIK-432 validations failing test green * OPIK-432 refactor * OPIK-432 cover auth * OPIK-432 code style * OPIK-432 code simplify * OPIK-432 use number * OPIK-432 fix comparison * OPIK-432 pr comments * OPIK-432 pr comments * OPIK-432 pr comments * OPIK-432 pr comments
- Loading branch information
Showing
10 changed files
with
621 additions
and
0 deletions.
There are no files selected for viewing
8 changes: 8 additions & 0 deletions
8
apps/opik-backend/src/main/java/com/comet/opik/api/DataPoint.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,8 @@ | ||
package com.comet.opik.api; | ||
|
||
import lombok.Builder; | ||
|
||
import java.time.Instant; | ||
|
||
@Builder(toBuilder = true) | ||
public record DataPoint(Instant time, Number value) {} |
7 changes: 7 additions & 0 deletions
7
apps/opik-backend/src/main/java/com/comet/opik/api/TimeInterval.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,7 @@ | ||
package com.comet.opik.api; | ||
|
||
public enum TimeInterval { | ||
HOURLY, | ||
DAILY, | ||
WEEKLY, | ||
} |
7 changes: 7 additions & 0 deletions
7
apps/opik-backend/src/main/java/com/comet/opik/api/metrics/MetricType.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,7 @@ | ||
package com.comet.opik.api.metrics; | ||
|
||
public enum MetricType { | ||
FEEDBACK_SCORES, | ||
TRACE_COUNT, | ||
TOKEN_USAGE, | ||
} |
19 changes: 19 additions & 0 deletions
19
apps/opik-backend/src/main/java/com/comet/opik/api/metrics/ProjectMetricRequest.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,19 @@ | ||
package com.comet.opik.api.metrics; | ||
|
||
import com.comet.opik.api.TimeInterval; | ||
import com.fasterxml.jackson.annotation.JsonIgnoreProperties; | ||
import com.fasterxml.jackson.databind.PropertyNamingStrategies; | ||
import com.fasterxml.jackson.databind.annotation.JsonNaming; | ||
import lombok.Builder; | ||
import lombok.NonNull; | ||
|
||
import java.time.Instant; | ||
|
||
@Builder(toBuilder = true) | ||
@JsonIgnoreProperties(ignoreUnknown = true) | ||
@JsonNaming(PropertyNamingStrategies.SnakeCaseStrategy.class) | ||
public record ProjectMetricRequest( | ||
@NonNull MetricType metricType, | ||
@NonNull TimeInterval interval, | ||
Instant intervalStart, | ||
Instant intervalEnd) {} |
31 changes: 31 additions & 0 deletions
31
apps/opik-backend/src/main/java/com/comet/opik/api/metrics/ProjectMetricResponse.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,31 @@ | ||
package com.comet.opik.api.metrics; | ||
|
||
import com.comet.opik.api.DataPoint; | ||
import com.comet.opik.api.TimeInterval; | ||
import com.fasterxml.jackson.annotation.JsonIgnoreProperties; | ||
import com.fasterxml.jackson.databind.PropertyNamingStrategies; | ||
import com.fasterxml.jackson.databind.annotation.JsonNaming; | ||
import lombok.Builder; | ||
|
||
import java.time.Instant; | ||
import java.util.List; | ||
import java.util.UUID; | ||
|
||
@Builder(toBuilder = true) | ||
@JsonIgnoreProperties(ignoreUnknown = true) | ||
@JsonNaming(PropertyNamingStrategies.SnakeCaseStrategy.class) | ||
public record ProjectMetricResponse( | ||
UUID projectId, | ||
MetricType metricType, | ||
TimeInterval interval, | ||
List<Results> results) { | ||
|
||
public static final ProjectMetricResponse EMPTY = ProjectMetricResponse.builder() | ||
.results(List.of()) | ||
.build(); | ||
|
||
@Builder(toBuilder = true) | ||
@JsonIgnoreProperties(ignoreUnknown = true) | ||
@JsonNaming(PropertyNamingStrategies.SnakeCaseStrategy.class) | ||
public record Results(String name, List<DataPoint> data) {} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
79 changes: 79 additions & 0 deletions
79
apps/opik-backend/src/main/java/com/comet/opik/domain/ProjectMetricsDAO.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,79 @@ | ||
package com.comet.opik.domain; | ||
|
||
import com.comet.opik.api.DataPoint; | ||
import com.comet.opik.api.metrics.ProjectMetricRequest; | ||
import com.comet.opik.infrastructure.instrumentation.InstrumentAsyncUtils; | ||
import com.google.inject.ImplementedBy; | ||
import io.r2dbc.spi.Connection; | ||
import io.r2dbc.spi.Result; | ||
import jakarta.inject.Inject; | ||
import jakarta.inject.Singleton; | ||
import lombok.RequiredArgsConstructor; | ||
import lombok.extern.slf4j.Slf4j; | ||
import org.reactivestreams.Publisher; | ||
import org.stringtemplate.v4.ST; | ||
import reactor.core.publisher.Mono; | ||
|
||
import java.time.Instant; | ||
import java.util.List; | ||
import java.util.UUID; | ||
|
||
import static com.comet.opik.domain.AsyncContextUtils.bindWorkspaceIdToMono; | ||
import static com.comet.opik.infrastructure.instrumentation.InstrumentAsyncUtils.endSegment; | ||
import static com.comet.opik.infrastructure.instrumentation.InstrumentAsyncUtils.startSegment; | ||
import static com.comet.opik.utils.AsyncUtils.makeMonoContextAware; | ||
|
||
@ImplementedBy(ProjectMetricsDAOImpl.class) | ||
public interface ProjectMetricsDAO { | ||
Mono<List<DataPoint>> getTraceCount(UUID projectId, ProjectMetricRequest request, Connection connection); | ||
} | ||
|
||
@Slf4j | ||
@Singleton | ||
@RequiredArgsConstructor(onConstructor_ = @Inject) | ||
class ProjectMetricsDAOImpl implements ProjectMetricsDAO { | ||
private static final String GET_TRACE_COUNT = """ | ||
SELECT toStartOfInterval(start_time, toIntervalHour(1)) AS bucket, | ||
count() as count | ||
FROM traces | ||
WHERE project_id = :project_id | ||
AND workspace_id = :workspace_id | ||
AND start_time >= parseDateTime64BestEffort(:start_time, 9) | ||
AND end_time \\<= parseDateTime64BestEffort(:end_time, 9) | ||
GROUP BY bucket | ||
ORDER BY bucket | ||
WITH FILL | ||
FROM parseDateTimeBestEffort(:start_time) | ||
TO parseDateTimeBestEffort(:end_time) | ||
STEP toIntervalHour(1); | ||
"""; | ||
|
||
@Override | ||
public Mono<List<DataPoint>> getTraceCount( | ||
UUID projectId, ProjectMetricRequest request, Connection connection) { | ||
return getTracesCountForProject(projectId, request, connection) | ||
.flatMapMany(this::mapToIntDataPoint) | ||
.collectList(); | ||
} | ||
|
||
private Mono<? extends Result> getTracesCountForProject( | ||
UUID projectId, ProjectMetricRequest request, Connection connection) { | ||
var template = new ST(GET_TRACE_COUNT); | ||
var statement = connection.createStatement(template.render()) | ||
.bind("project_id", projectId) | ||
.bind("start_time", request.intervalStart().toString()) | ||
.bind("end_time", request.intervalEnd().toString()); | ||
|
||
InstrumentAsyncUtils.Segment segment = startSegment("traceCount", "Clickhouse", "get"); | ||
|
||
return makeMonoContextAware(bindWorkspaceIdToMono(statement)) | ||
.doFinally(signalType -> endSegment(segment)); | ||
} | ||
|
||
private Publisher<DataPoint> mapToIntDataPoint(Result result) { | ||
return result.map(((row, rowMetadata) -> DataPoint.builder() | ||
.time(row.get("bucket", Instant.class)) | ||
.value(row.get("count", Integer.class)) | ||
.build())); | ||
} | ||
} |
57 changes: 57 additions & 0 deletions
57
apps/opik-backend/src/main/java/com/comet/opik/domain/ProjectMetricsService.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,57 @@ | ||
package com.comet.opik.domain; | ||
|
||
import com.comet.opik.api.DataPoint; | ||
import com.comet.opik.api.metrics.ProjectMetricRequest; | ||
import com.comet.opik.api.metrics.ProjectMetricResponse; | ||
import com.comet.opik.infrastructure.db.TransactionTemplateAsync; | ||
import com.google.inject.ImplementedBy; | ||
import jakarta.inject.Inject; | ||
import jakarta.inject.Singleton; | ||
import jakarta.ws.rs.BadRequestException; | ||
import lombok.NonNull; | ||
import lombok.RequiredArgsConstructor; | ||
import lombok.extern.slf4j.Slf4j; | ||
import reactor.core.publisher.Mono; | ||
|
||
import java.util.List; | ||
import java.util.UUID; | ||
|
||
@ImplementedBy(ProjectMetricsServiceImpl.class) | ||
public interface ProjectMetricsService { | ||
String ERR_START_BEFORE_END = "'start_time' must be before 'end_time'"; | ||
|
||
Mono<ProjectMetricResponse> getProjectMetrics(UUID projectId, ProjectMetricRequest request); | ||
} | ||
|
||
@Slf4j | ||
@Singleton | ||
@RequiredArgsConstructor(onConstructor_ = @Inject) | ||
class ProjectMetricsServiceImpl implements ProjectMetricsService { | ||
private final @NonNull TransactionTemplateAsync template; | ||
private final @NonNull ProjectMetricsDAO projectMetricsDAO; | ||
|
||
public static final String NAME_TRACES = "traces"; | ||
|
||
@Override | ||
public Mono<ProjectMetricResponse> getProjectMetrics(UUID projectId, ProjectMetricRequest request) { | ||
validate(request); | ||
|
||
return template.nonTransaction(connection -> projectMetricsDAO.getTraceCount(projectId, request, | ||
connection) | ||
.map(dataPoints -> ProjectMetricResponse.builder() | ||
.projectId(projectId) | ||
.metricType(request.metricType()) | ||
.interval(request.interval()) | ||
.results(List.of(ProjectMetricResponse.Results.builder() | ||
.name(NAME_TRACES) | ||
.data(dataPoints) | ||
.build())) | ||
.build())); | ||
} | ||
|
||
private void validate(ProjectMetricRequest request) { | ||
if (!request.intervalStart().isBefore(request.intervalEnd())) { | ||
throw new BadRequestException(ERR_START_BEFORE_END); | ||
} | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.