arize-phoenix 11.23.1__py3-none-any.whl → 12.28.1__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- {arize_phoenix-11.23.1.dist-info → arize_phoenix-12.28.1.dist-info}/METADATA +61 -36
- {arize_phoenix-11.23.1.dist-info → arize_phoenix-12.28.1.dist-info}/RECORD +212 -162
- {arize_phoenix-11.23.1.dist-info → arize_phoenix-12.28.1.dist-info}/WHEEL +1 -1
- {arize_phoenix-11.23.1.dist-info → arize_phoenix-12.28.1.dist-info}/licenses/IP_NOTICE +1 -1
- phoenix/__generated__/__init__.py +0 -0
- phoenix/__generated__/classification_evaluator_configs/__init__.py +20 -0
- phoenix/__generated__/classification_evaluator_configs/_document_relevance_classification_evaluator_config.py +17 -0
- phoenix/__generated__/classification_evaluator_configs/_hallucination_classification_evaluator_config.py +17 -0
- phoenix/__generated__/classification_evaluator_configs/_models.py +18 -0
- phoenix/__generated__/classification_evaluator_configs/_tool_selection_classification_evaluator_config.py +17 -0
- phoenix/__init__.py +2 -1
- phoenix/auth.py +27 -2
- phoenix/config.py +1594 -81
- phoenix/db/README.md +546 -28
- phoenix/db/bulk_inserter.py +119 -116
- phoenix/db/engines.py +140 -33
- phoenix/db/facilitator.py +22 -1
- phoenix/db/helpers.py +818 -65
- phoenix/db/iam_auth.py +64 -0
- phoenix/db/insertion/dataset.py +133 -1
- phoenix/db/insertion/document_annotation.py +9 -6
- phoenix/db/insertion/evaluation.py +2 -3
- phoenix/db/insertion/helpers.py +2 -2
- phoenix/db/insertion/session_annotation.py +176 -0
- phoenix/db/insertion/span_annotation.py +3 -4
- phoenix/db/insertion/trace_annotation.py +3 -4
- phoenix/db/insertion/types.py +41 -18
- phoenix/db/migrations/versions/01a8342c9cdf_add_user_id_on_datasets.py +40 -0
- phoenix/db/migrations/versions/0df286449799_add_session_annotations_table.py +105 -0
- phoenix/db/migrations/versions/272b66ff50f8_drop_single_indices.py +119 -0
- phoenix/db/migrations/versions/58228d933c91_dataset_labels.py +67 -0
- phoenix/db/migrations/versions/699f655af132_experiment_tags.py +57 -0
- phoenix/db/migrations/versions/735d3d93c33e_add_composite_indices.py +41 -0
- phoenix/db/migrations/versions/ab513d89518b_add_user_id_on_dataset_versions.py +40 -0
- phoenix/db/migrations/versions/d0690a79ea51_users_on_experiments.py +40 -0
- phoenix/db/migrations/versions/deb2c81c0bb2_dataset_splits.py +139 -0
- phoenix/db/migrations/versions/e76cbd66ffc3_add_experiments_dataset_examples.py +87 -0
- phoenix/db/models.py +364 -56
- phoenix/db/pg_config.py +10 -0
- phoenix/db/types/trace_retention.py +7 -6
- phoenix/experiments/functions.py +69 -19
- phoenix/inferences/inferences.py +1 -2
- phoenix/server/api/auth.py +9 -0
- phoenix/server/api/auth_messages.py +46 -0
- phoenix/server/api/context.py +60 -0
- phoenix/server/api/dataloaders/__init__.py +36 -0
- phoenix/server/api/dataloaders/annotation_summaries.py +60 -8
- phoenix/server/api/dataloaders/average_experiment_repeated_run_group_latency.py +50 -0
- phoenix/server/api/dataloaders/average_experiment_run_latency.py +17 -24
- phoenix/server/api/dataloaders/cache/two_tier_cache.py +1 -2
- phoenix/server/api/dataloaders/dataset_dataset_splits.py +52 -0
- phoenix/server/api/dataloaders/dataset_example_revisions.py +0 -1
- phoenix/server/api/dataloaders/dataset_example_splits.py +40 -0
- phoenix/server/api/dataloaders/dataset_examples_and_versions_by_experiment_run.py +47 -0
- phoenix/server/api/dataloaders/dataset_labels.py +36 -0
- phoenix/server/api/dataloaders/document_evaluation_summaries.py +2 -2
- phoenix/server/api/dataloaders/document_evaluations.py +6 -9
- phoenix/server/api/dataloaders/experiment_annotation_summaries.py +88 -34
- phoenix/server/api/dataloaders/experiment_dataset_splits.py +43 -0
- phoenix/server/api/dataloaders/experiment_error_rates.py +21 -28
- phoenix/server/api/dataloaders/experiment_repeated_run_group_annotation_summaries.py +77 -0
- phoenix/server/api/dataloaders/experiment_repeated_run_groups.py +57 -0
- phoenix/server/api/dataloaders/experiment_runs_by_experiment_and_example.py +44 -0
- phoenix/server/api/dataloaders/latency_ms_quantile.py +40 -8
- phoenix/server/api/dataloaders/record_counts.py +37 -10
- phoenix/server/api/dataloaders/session_annotations_by_session.py +29 -0
- phoenix/server/api/dataloaders/span_cost_summary_by_experiment_repeated_run_group.py +64 -0
- phoenix/server/api/dataloaders/span_cost_summary_by_project.py +28 -14
- phoenix/server/api/dataloaders/span_costs.py +3 -9
- phoenix/server/api/dataloaders/table_fields.py +2 -2
- phoenix/server/api/dataloaders/token_prices_by_model.py +30 -0
- phoenix/server/api/dataloaders/trace_annotations_by_trace.py +27 -0
- phoenix/server/api/exceptions.py +5 -1
- phoenix/server/api/helpers/playground_clients.py +263 -83
- phoenix/server/api/helpers/playground_spans.py +2 -1
- phoenix/server/api/helpers/playground_users.py +26 -0
- phoenix/server/api/helpers/prompts/conversions/google.py +103 -0
- phoenix/server/api/helpers/prompts/models.py +61 -19
- phoenix/server/api/input_types/{SpanAnnotationFilter.py → AnnotationFilter.py} +22 -14
- phoenix/server/api/input_types/ChatCompletionInput.py +3 -0
- phoenix/server/api/input_types/CreateProjectSessionAnnotationInput.py +37 -0
- phoenix/server/api/input_types/DatasetFilter.py +5 -2
- phoenix/server/api/input_types/ExperimentRunSort.py +237 -0
- phoenix/server/api/input_types/GenerativeModelInput.py +3 -0
- phoenix/server/api/input_types/ProjectSessionSort.py +158 -1
- phoenix/server/api/input_types/PromptVersionInput.py +47 -1
- phoenix/server/api/input_types/SpanSort.py +3 -2
- phoenix/server/api/input_types/UpdateAnnotationInput.py +34 -0
- phoenix/server/api/input_types/UserRoleInput.py +1 -0
- phoenix/server/api/mutations/__init__.py +8 -0
- phoenix/server/api/mutations/annotation_config_mutations.py +8 -8
- phoenix/server/api/mutations/api_key_mutations.py +15 -20
- phoenix/server/api/mutations/chat_mutations.py +106 -37
- phoenix/server/api/mutations/dataset_label_mutations.py +243 -0
- phoenix/server/api/mutations/dataset_mutations.py +21 -16
- phoenix/server/api/mutations/dataset_split_mutations.py +351 -0
- phoenix/server/api/mutations/experiment_mutations.py +2 -2
- phoenix/server/api/mutations/export_events_mutations.py +3 -3
- phoenix/server/api/mutations/model_mutations.py +11 -9
- phoenix/server/api/mutations/project_mutations.py +4 -4
- phoenix/server/api/mutations/project_session_annotations_mutations.py +158 -0
- phoenix/server/api/mutations/project_trace_retention_policy_mutations.py +8 -4
- phoenix/server/api/mutations/prompt_label_mutations.py +74 -65
- phoenix/server/api/mutations/prompt_mutations.py +65 -129
- phoenix/server/api/mutations/prompt_version_tag_mutations.py +11 -8
- phoenix/server/api/mutations/span_annotations_mutations.py +15 -10
- phoenix/server/api/mutations/trace_annotations_mutations.py +13 -8
- phoenix/server/api/mutations/trace_mutations.py +3 -3
- phoenix/server/api/mutations/user_mutations.py +55 -26
- phoenix/server/api/queries.py +501 -617
- phoenix/server/api/routers/__init__.py +2 -2
- phoenix/server/api/routers/auth.py +141 -87
- phoenix/server/api/routers/ldap.py +229 -0
- phoenix/server/api/routers/oauth2.py +349 -101
- phoenix/server/api/routers/v1/__init__.py +22 -4
- phoenix/server/api/routers/v1/annotation_configs.py +19 -30
- phoenix/server/api/routers/v1/annotations.py +455 -13
- phoenix/server/api/routers/v1/datasets.py +355 -68
- phoenix/server/api/routers/v1/documents.py +142 -0
- phoenix/server/api/routers/v1/evaluations.py +20 -28
- phoenix/server/api/routers/v1/experiment_evaluations.py +16 -6
- phoenix/server/api/routers/v1/experiment_runs.py +335 -59
- phoenix/server/api/routers/v1/experiments.py +475 -47
- phoenix/server/api/routers/v1/projects.py +16 -50
- phoenix/server/api/routers/v1/prompts.py +50 -39
- phoenix/server/api/routers/v1/sessions.py +108 -0
- phoenix/server/api/routers/v1/spans.py +156 -96
- phoenix/server/api/routers/v1/traces.py +51 -77
- phoenix/server/api/routers/v1/users.py +64 -24
- phoenix/server/api/routers/v1/utils.py +3 -7
- phoenix/server/api/subscriptions.py +257 -93
- phoenix/server/api/types/Annotation.py +90 -23
- phoenix/server/api/types/ApiKey.py +13 -17
- phoenix/server/api/types/AuthMethod.py +1 -0
- phoenix/server/api/types/ChatCompletionSubscriptionPayload.py +1 -0
- phoenix/server/api/types/Dataset.py +199 -72
- phoenix/server/api/types/DatasetExample.py +88 -18
- phoenix/server/api/types/DatasetExperimentAnnotationSummary.py +10 -0
- phoenix/server/api/types/DatasetLabel.py +57 -0
- phoenix/server/api/types/DatasetSplit.py +98 -0
- phoenix/server/api/types/DatasetVersion.py +49 -4
- phoenix/server/api/types/DocumentAnnotation.py +212 -0
- phoenix/server/api/types/Experiment.py +215 -68
- phoenix/server/api/types/ExperimentComparison.py +3 -9
- phoenix/server/api/types/ExperimentRepeatedRunGroup.py +155 -0
- phoenix/server/api/types/ExperimentRepeatedRunGroupAnnotationSummary.py +9 -0
- phoenix/server/api/types/ExperimentRun.py +120 -70
- phoenix/server/api/types/ExperimentRunAnnotation.py +158 -39
- phoenix/server/api/types/GenerativeModel.py +95 -42
- phoenix/server/api/types/GenerativeProvider.py +1 -1
- phoenix/server/api/types/ModelInterface.py +7 -2
- phoenix/server/api/types/PlaygroundModel.py +12 -2
- phoenix/server/api/types/Project.py +218 -185
- phoenix/server/api/types/ProjectSession.py +146 -29
- phoenix/server/api/types/ProjectSessionAnnotation.py +187 -0
- phoenix/server/api/types/ProjectTraceRetentionPolicy.py +1 -1
- phoenix/server/api/types/Prompt.py +119 -39
- phoenix/server/api/types/PromptLabel.py +42 -25
- phoenix/server/api/types/PromptVersion.py +11 -8
- phoenix/server/api/types/PromptVersionTag.py +65 -25
- phoenix/server/api/types/Span.py +130 -123
- phoenix/server/api/types/SpanAnnotation.py +189 -42
- phoenix/server/api/types/SystemApiKey.py +65 -1
- phoenix/server/api/types/Trace.py +184 -53
- phoenix/server/api/types/TraceAnnotation.py +149 -50
- phoenix/server/api/types/User.py +128 -33
- phoenix/server/api/types/UserApiKey.py +73 -26
- phoenix/server/api/types/node.py +10 -0
- phoenix/server/api/types/pagination.py +11 -2
- phoenix/server/app.py +154 -36
- phoenix/server/authorization.py +5 -4
- phoenix/server/bearer_auth.py +13 -5
- phoenix/server/cost_tracking/cost_model_lookup.py +42 -14
- phoenix/server/cost_tracking/model_cost_manifest.json +1085 -194
- phoenix/server/daemons/generative_model_store.py +61 -9
- phoenix/server/daemons/span_cost_calculator.py +10 -8
- phoenix/server/dml_event.py +13 -0
- phoenix/server/email/sender.py +29 -2
- phoenix/server/grpc_server.py +9 -9
- phoenix/server/jwt_store.py +8 -6
- phoenix/server/ldap.py +1449 -0
- phoenix/server/main.py +9 -3
- phoenix/server/oauth2.py +330 -12
- phoenix/server/prometheus.py +43 -6
- phoenix/server/rate_limiters.py +4 -9
- phoenix/server/retention.py +33 -20
- phoenix/server/session_filters.py +49 -0
- phoenix/server/static/.vite/manifest.json +51 -53
- phoenix/server/static/assets/components-BreFUQQa.js +6702 -0
- phoenix/server/static/assets/{index-BPCwGQr8.js → index-CTQoemZv.js} +42 -35
- phoenix/server/static/assets/pages-DBE5iYM3.js +9524 -0
- phoenix/server/static/assets/vendor-BGzfc4EU.css +1 -0
- phoenix/server/static/assets/vendor-DCE4v-Ot.js +920 -0
- phoenix/server/static/assets/vendor-codemirror-D5f205eT.js +25 -0
- phoenix/server/static/assets/{vendor-recharts-Bw30oz1A.js → vendor-recharts-V9cwpXsm.js} +7 -7
- phoenix/server/static/assets/{vendor-shiki-DZajAPeq.js → vendor-shiki-Do--csgv.js} +1 -1
- phoenix/server/static/assets/vendor-three-CmB8bl_y.js +3840 -0
- phoenix/server/templates/index.html +7 -1
- phoenix/server/thread_server.py +1 -2
- phoenix/server/utils.py +74 -0
- phoenix/session/client.py +55 -1
- phoenix/session/data_extractor.py +5 -0
- phoenix/session/evaluation.py +8 -4
- phoenix/session/session.py +44 -8
- phoenix/settings.py +2 -0
- phoenix/trace/attributes.py +80 -13
- phoenix/trace/dsl/query.py +2 -0
- phoenix/trace/projects.py +5 -0
- phoenix/utilities/template_formatters.py +1 -1
- phoenix/version.py +1 -1
- phoenix/server/api/types/Evaluation.py +0 -39
- phoenix/server/static/assets/components-D0DWAf0l.js +0 -5650
- phoenix/server/static/assets/pages-Creyamao.js +0 -8612
- phoenix/server/static/assets/vendor-CU36oj8y.js +0 -905
- phoenix/server/static/assets/vendor-CqDb5u4o.css +0 -1
- phoenix/server/static/assets/vendor-arizeai-Ctgw0e1G.js +0 -168
- phoenix/server/static/assets/vendor-codemirror-Cojjzqb9.js +0 -25
- phoenix/server/static/assets/vendor-three-BLWp5bic.js +0 -2998
- phoenix/utilities/deprecation.py +0 -31
- {arize_phoenix-11.23.1.dist-info → arize_phoenix-12.28.1.dist-info}/entry_points.txt +0 -0
- {arize_phoenix-11.23.1.dist-info → arize_phoenix-12.28.1.dist-info}/licenses/LICENSE +0 -0
|
@@ -0,0 +1,139 @@
|
|
|
1
|
+
"""dataset_splits
|
|
2
|
+
|
|
3
|
+
Revision ID: deb2c81c0bb2
|
|
4
|
+
Revises: 58228d933c91
|
|
5
|
+
Create Date: 2025-09-08 15:50:12.066217
|
|
6
|
+
|
|
7
|
+
"""
|
|
8
|
+
|
|
9
|
+
from typing import Any, Sequence, Union
|
|
10
|
+
|
|
11
|
+
import sqlalchemy as sa
|
|
12
|
+
from alembic import op
|
|
13
|
+
from sqlalchemy import JSON
|
|
14
|
+
from sqlalchemy.dialects import postgresql
|
|
15
|
+
from sqlalchemy.ext.compiler import compiles
|
|
16
|
+
|
|
17
|
+
|
|
18
|
+
class JSONB(JSON):
|
|
19
|
+
# See https://docs.sqlalchemy.org/en/20/core/custom_types.html
|
|
20
|
+
__visit_name__ = "JSONB"
|
|
21
|
+
|
|
22
|
+
|
|
23
|
+
@compiles(JSONB, "sqlite")
|
|
24
|
+
def _(*args: Any, **kwargs: Any) -> str:
|
|
25
|
+
# See https://docs.sqlalchemy.org/en/20/core/custom_types.html
|
|
26
|
+
return "JSONB"
|
|
27
|
+
|
|
28
|
+
|
|
29
|
+
JSON_ = (
|
|
30
|
+
JSON()
|
|
31
|
+
.with_variant(
|
|
32
|
+
postgresql.JSONB(),
|
|
33
|
+
"postgresql",
|
|
34
|
+
)
|
|
35
|
+
.with_variant(
|
|
36
|
+
JSONB(),
|
|
37
|
+
"sqlite",
|
|
38
|
+
)
|
|
39
|
+
)
|
|
40
|
+
|
|
41
|
+
_Integer = sa.Integer().with_variant(
|
|
42
|
+
sa.BigInteger(),
|
|
43
|
+
"postgresql",
|
|
44
|
+
)
|
|
45
|
+
|
|
46
|
+
|
|
47
|
+
# revision identifiers, used by Alembic.
|
|
48
|
+
revision: str = "deb2c81c0bb2"
|
|
49
|
+
down_revision: Union[str, None] = "e76cbd66ffc3"
|
|
50
|
+
branch_labels: Union[str, Sequence[str], None] = None
|
|
51
|
+
depends_on: Union[str, Sequence[str], None] = None
|
|
52
|
+
|
|
53
|
+
|
|
54
|
+
def upgrade() -> None:
|
|
55
|
+
# Create dataset_splits table
|
|
56
|
+
op.create_table(
|
|
57
|
+
"dataset_splits",
|
|
58
|
+
sa.Column("id", _Integer, primary_key=True),
|
|
59
|
+
sa.Column(
|
|
60
|
+
"user_id",
|
|
61
|
+
_Integer,
|
|
62
|
+
sa.ForeignKey("users.id", ondelete="SET NULL"),
|
|
63
|
+
nullable=True,
|
|
64
|
+
index=True,
|
|
65
|
+
),
|
|
66
|
+
sa.Column("name", sa.String, nullable=False, unique=True),
|
|
67
|
+
sa.Column("description", sa.String, nullable=True),
|
|
68
|
+
sa.Column("color", sa.String, nullable=False),
|
|
69
|
+
sa.Column("metadata", JSON_, nullable=False),
|
|
70
|
+
sa.Column(
|
|
71
|
+
"created_at",
|
|
72
|
+
sa.TIMESTAMP(timezone=True),
|
|
73
|
+
nullable=False,
|
|
74
|
+
server_default=sa.func.now(),
|
|
75
|
+
),
|
|
76
|
+
sa.Column(
|
|
77
|
+
"updated_at",
|
|
78
|
+
sa.TIMESTAMP(timezone=True),
|
|
79
|
+
nullable=False,
|
|
80
|
+
server_default=sa.func.now(),
|
|
81
|
+
onupdate=sa.func.now(),
|
|
82
|
+
),
|
|
83
|
+
)
|
|
84
|
+
|
|
85
|
+
# Create crosswalk table: dataset_splits_dataset_examples
|
|
86
|
+
op.create_table(
|
|
87
|
+
"dataset_splits_dataset_examples",
|
|
88
|
+
sa.Column(
|
|
89
|
+
"dataset_split_id",
|
|
90
|
+
_Integer,
|
|
91
|
+
sa.ForeignKey("dataset_splits.id", ondelete="CASCADE"),
|
|
92
|
+
nullable=False,
|
|
93
|
+
),
|
|
94
|
+
sa.Column(
|
|
95
|
+
"dataset_example_id",
|
|
96
|
+
_Integer,
|
|
97
|
+
sa.ForeignKey("dataset_examples.id", ondelete="CASCADE"),
|
|
98
|
+
nullable=False,
|
|
99
|
+
# index on the second element of the composite primary key
|
|
100
|
+
index=True,
|
|
101
|
+
),
|
|
102
|
+
sa.PrimaryKeyConstraint(
|
|
103
|
+
"dataset_split_id",
|
|
104
|
+
"dataset_example_id",
|
|
105
|
+
),
|
|
106
|
+
)
|
|
107
|
+
|
|
108
|
+
# Create experiments_dataset_splits table The rational of this table is to
|
|
109
|
+
# gather examples for a specific dataset split for a specific experiment.
|
|
110
|
+
# Select all dataset examples where examples belong to a dataset split and
|
|
111
|
+
# examples belong to a experiment.
|
|
112
|
+
|
|
113
|
+
op.create_table(
|
|
114
|
+
"experiments_dataset_splits",
|
|
115
|
+
sa.Column(
|
|
116
|
+
"experiment_id",
|
|
117
|
+
_Integer,
|
|
118
|
+
sa.ForeignKey("experiments.id", ondelete="CASCADE"),
|
|
119
|
+
nullable=False,
|
|
120
|
+
),
|
|
121
|
+
sa.Column(
|
|
122
|
+
"dataset_split_id",
|
|
123
|
+
_Integer,
|
|
124
|
+
sa.ForeignKey("dataset_splits.id", ondelete="CASCADE"),
|
|
125
|
+
nullable=False,
|
|
126
|
+
# index on the second element of the composite primary key
|
|
127
|
+
index=True,
|
|
128
|
+
),
|
|
129
|
+
sa.PrimaryKeyConstraint(
|
|
130
|
+
"experiment_id",
|
|
131
|
+
"dataset_split_id",
|
|
132
|
+
),
|
|
133
|
+
)
|
|
134
|
+
|
|
135
|
+
|
|
136
|
+
def downgrade() -> None:
|
|
137
|
+
op.drop_table("experiments_dataset_splits")
|
|
138
|
+
op.drop_table("dataset_splits_dataset_examples")
|
|
139
|
+
op.drop_table("dataset_splits")
|
|
@@ -0,0 +1,87 @@
|
|
|
1
|
+
"""add experiments_dataset_examples junction table
|
|
2
|
+
|
|
3
|
+
Revision ID: e76cbd66ffc3
|
|
4
|
+
Revises: deb2c81c0bb2
|
|
5
|
+
Create Date: 2025-09-23 12:33:13.554164
|
|
6
|
+
|
|
7
|
+
"""
|
|
8
|
+
|
|
9
|
+
from typing import Sequence, Union
|
|
10
|
+
|
|
11
|
+
import sqlalchemy as sa
|
|
12
|
+
from alembic import op
|
|
13
|
+
|
|
14
|
+
_Integer = sa.Integer().with_variant(
|
|
15
|
+
sa.BigInteger(),
|
|
16
|
+
"postgresql",
|
|
17
|
+
)
|
|
18
|
+
|
|
19
|
+
# revision identifiers, used by Alembic.
|
|
20
|
+
revision: str = "e76cbd66ffc3"
|
|
21
|
+
down_revision: Union[str, None] = "58228d933c91"
|
|
22
|
+
branch_labels: Union[str, Sequence[str], None] = None
|
|
23
|
+
depends_on: Union[str, Sequence[str], None] = None
|
|
24
|
+
|
|
25
|
+
BACKFILL = """\
|
|
26
|
+
INSERT INTO experiments_dataset_examples (
|
|
27
|
+
experiment_id,
|
|
28
|
+
dataset_example_id,
|
|
29
|
+
dataset_example_revision_id
|
|
30
|
+
)
|
|
31
|
+
SELECT
|
|
32
|
+
ranked.experiment_id,
|
|
33
|
+
ranked.dataset_example_id,
|
|
34
|
+
ranked.dataset_example_revision_id
|
|
35
|
+
FROM (
|
|
36
|
+
SELECT
|
|
37
|
+
e.id as experiment_id,
|
|
38
|
+
der.dataset_example_id,
|
|
39
|
+
der.id as dataset_example_revision_id,
|
|
40
|
+
der.revision_kind,
|
|
41
|
+
ROW_NUMBER() OVER (
|
|
42
|
+
PARTITION BY e.id, der.dataset_example_id
|
|
43
|
+
ORDER BY der.dataset_version_id DESC
|
|
44
|
+
) as rn
|
|
45
|
+
FROM experiments e
|
|
46
|
+
JOIN dataset_examples de ON de.dataset_id = e.dataset_id
|
|
47
|
+
JOIN dataset_example_revisions der ON der.dataset_example_id = de.id
|
|
48
|
+
WHERE der.dataset_version_id <= e.dataset_version_id
|
|
49
|
+
) ranked
|
|
50
|
+
WHERE ranked.rn = 1
|
|
51
|
+
AND ranked.revision_kind != 'DELETE'
|
|
52
|
+
"""
|
|
53
|
+
|
|
54
|
+
|
|
55
|
+
def upgrade() -> None:
|
|
56
|
+
op.create_table(
|
|
57
|
+
"experiments_dataset_examples",
|
|
58
|
+
sa.Column(
|
|
59
|
+
"experiment_id",
|
|
60
|
+
_Integer,
|
|
61
|
+
sa.ForeignKey("experiments.id", ondelete="CASCADE"),
|
|
62
|
+
nullable=False,
|
|
63
|
+
),
|
|
64
|
+
sa.Column(
|
|
65
|
+
"dataset_example_id",
|
|
66
|
+
_Integer,
|
|
67
|
+
sa.ForeignKey("dataset_examples.id", ondelete="CASCADE"),
|
|
68
|
+
nullable=False,
|
|
69
|
+
index=True,
|
|
70
|
+
),
|
|
71
|
+
sa.Column(
|
|
72
|
+
"dataset_example_revision_id",
|
|
73
|
+
_Integer,
|
|
74
|
+
sa.ForeignKey("dataset_example_revisions.id", ondelete="CASCADE"),
|
|
75
|
+
nullable=False,
|
|
76
|
+
index=True,
|
|
77
|
+
),
|
|
78
|
+
sa.PrimaryKeyConstraint(
|
|
79
|
+
"experiment_id",
|
|
80
|
+
"dataset_example_id",
|
|
81
|
+
),
|
|
82
|
+
)
|
|
83
|
+
op.execute(BACKFILL)
|
|
84
|
+
|
|
85
|
+
|
|
86
|
+
def downgrade() -> None:
|
|
87
|
+
op.drop_table("experiments_dataset_examples")
|