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

Download Sample Datasets #3725

Merged
merged 29 commits into from
Mar 6, 2019
Merged
Show file tree
Hide file tree
Changes from 13 commits
Commits
Show all changes
29 commits
Select commit Hold shift + click to select a range
567ab8f
[WIP] download demo dataset
fm3 Feb 4, 2019
ab27fcd
prepare download + guard
fm3 Feb 11, 2019
0c95bcd
Merge branch 'master' into demo-dataset
fm3 Feb 11, 2019
0df51af
list available sample datasets, enable download
fm3 Feb 12, 2019
d7a243e
Merge branch 'master' of github.com:scalableminds/webknossos into dem…
daniel-wer Feb 14, 2019
e2ee248
add sample datasets modal, add temp button to datasets view to open m…
daniel-wer Feb 14, 2019
7c1e3ac
fix typo in downloading status string
daniel-wer Feb 14, 2019
16fdaa5
fix reporting of downloading state
fm3 Feb 18, 2019
8fba42d
Merge branch 'master' of github.com:scalableminds/webknossos into dem…
daniel-wer Feb 18, 2019
d2a57be
use all datastores for sample dataset downloads, fix error case
daniel-wer Feb 18, 2019
7f4884b
deterministic sorting for datastore list (by name)
fm3 Feb 19, 2019
1be1b42
use single datastore, detect failed datasets, refactoring
daniel-wer Feb 19, 2019
06609aa
rework onboarding dataset step and dataset list placeholder, add link…
daniel-wer Feb 20, 2019
a268dc9
Update messages
fm3 Feb 21, 2019
756d4fb
merge master into demo-dataset
fm3 Feb 22, 2019
c49628d
Merge branch 'master' of github.com:scalableminds/webknossos into dem…
daniel-wer Feb 25, 2019
9bf5e61
refactor sample dataset modal state management according to PR feedback
daniel-wer Feb 25, 2019
054e84b
create generic useFetch method, make sure opening the modal with down…
daniel-wer Feb 26, 2019
155825a
Merge branch 'master' of github.com:scalableminds/webknossos into dem…
daniel-wer Feb 26, 2019
23caa4c
small style tweaks for onboarding dataset view
daniel-wer Feb 26, 2019
75caaea
insert actual sample datasets
fm3 Feb 27, 2019
93ac035
add description to sample datasets modal - untested
daniel-wer Feb 28, 2019
b0d5b50
preserve whitespace in sample datasets modal
daniel-wer Feb 28, 2019
2c7564e
Merge branch 'master' of github.com:scalableminds/webknossos into dem…
daniel-wer Feb 28, 2019
c9cf39f
Merge branch 'master' into demo-dataset
fm3 Mar 4, 2019
171b681
update docs + changelog
fm3 Mar 5, 2019
835748d
merge
fm3 Mar 5, 2019
96d046b
merge master into demo-dataset
fm3 Mar 5, 2019
5300199
Merge branch 'master' into demo-dataset
fm3 Mar 6, 2019
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
6 changes: 6 additions & 0 deletions app/models/binary/DataStore.scala
Original file line number Diff line number Diff line change
Expand Up @@ -82,6 +82,12 @@ class DataStoreDAO @Inject()(sqlClient: SQLClient)(implicit ec: ExecutionContext
parsed
}

override def findAll(implicit ctx: DBAccessContext): Fox[List[DataStore]] =
for {
r <- run(sql"select #${columns} from webknossos.datastores_ order by name".as[DatastoresRow])
parsed <- Fox.combined(r.toList.map(parse))
} yield parsed

def updateUrlByName(name: String, url: String)(implicit ctx: DBAccessContext): Fox[Unit] = {
val q = for { row <- Datastores if notdel(row) && row.name === name } yield row.url
for { _ <- run(q.update(url)) } yield ()
Expand Down
3 changes: 3 additions & 0 deletions conf/messages
Original file line number Diff line number Diff line change
Expand Up @@ -82,6 +82,7 @@ dataSet.notImported=Dataset {0} is not imported
dataSet.name.invalid=A dataset name can only contain letters, digits and underscores
dataSet.import.impossible.name=Import impossible. Dataset name can only consist of a-Z, 0-9, "-" and "_".
dataSet.name.alreadyTaken=This dataset name is already in use.
dataSet.name.notInSamples=This dataset name is not one of the available sample datasets.
dataSet.source.notFound=The data source for the data source couldn’t be found
dataSet.import.success=The import of the dataset was successful
dataSet.import.failed=Failed to import the dataset
Expand All @@ -93,6 +94,8 @@ dataSet.list.failed=Failed to retrieve list of data sets.
dataSet.url.missing=URL missing in the supplied json
dataSet.dataStore.missing=dataStore missing in the supplied json
dataSet.dataSet.missing=dataSet missing in the supplied json
dataSet.downloadAlreadyRunning=Sample dataset download already running.
daniel-wer marked this conversation as resolved.
Show resolved Hide resolved
dataSet.alreadyPresent=Sample dataset is already present.

dataSource.notFound=Datasource not found on datastore server

Expand Down
23 changes: 23 additions & 0 deletions frontend/javascripts/admin/admin_rest_api.js
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@ import {
type APIProjectProgressReport,
type APIProjectUpdater,
type APIProjectWithAssignments,
type APISampleDataset,
type APIScript,
type APIScriptCreator,
type APIScriptUpdater,
Expand Down Expand Up @@ -882,6 +883,28 @@ export async function getMappingsForDatasetLayer(
);
}

export function getSampleDatasets(
datastoreUrl: string,
organizationName: string,
): Promise<Array<APISampleDataset>> {
return doWithToken(token =>
Request.receiveJSON(`${datastoreUrl}/data/datasets/sample/${organizationName}?token=${token}`),
);
}

export async function triggerSampleDatasetDownload(
datastoreUrl: string,
organizationName: string,
datasetName: string,
) {
await doWithToken(token =>
Request.triggerRequest(
`${datastoreUrl}/data/datasets/sample/${organizationName}/${datasetName}/download?token=${token}`,
{ method: "POST" },
),
);
}

// #### Datastores
export async function getDatastores(): Promise<Array<APIDataStore>> {
const datastores = await Request.receiveJSON("/api/datastores");
Expand Down
5 changes: 5 additions & 0 deletions frontend/javascripts/admin/api_flow_types.js
Original file line number Diff line number Diff line change
Expand Up @@ -135,6 +135,11 @@ export type APIDataset = APIDatasetBase & {
+isActive: true,
};

export type APISampleDataset = {
name: string,
status: "available" | "downloading" | "present",
};

export type APIDataSourceWithMessages = {
+dataSource?: APIDataSource,
+messages: Array<APIMessage>,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -78,7 +78,7 @@ class DatasetAddForeignView extends React.PureComponent<Props> {

return (
<div className="container" style={{ paddingTop: 20 }}>
<Card title={<h3>Add Dataset</h3>}>
<Card title={<h3>Add Dataset</h3>} bordered={false}>
<p>
Specify the Dataset in the following format:
<br />
Expand Down Expand Up @@ -112,7 +112,7 @@ class DatasetAddForeignView extends React.PureComponent<Props> {
)}
</FormItem>
<FormItem>
<Button type="primary" htmlType="submit">
<Button size="large" type="primary" htmlType="submit" style={{ width: "100%" }}>
Add Dataset
</Button>
</FormItem>
Expand Down
89 changes: 62 additions & 27 deletions frontend/javascripts/admin/dataset/dataset_add_view.js
Original file line number Diff line number Diff line change
Expand Up @@ -2,49 +2,84 @@
import { type RouterHistory, withRouter } from "react-router-dom";
import { Tabs, Icon } from "antd";
import React from "react";
import { connect } from "react-redux";
import _ from "lodash";

import type { APIUser } from "admin/api_flow_types";
import type { OxalisState } from "oxalis/store";
import { enforceActiveUser } from "oxalis/model/accessors/user_accessor";
import DatasetAddForeignView from "admin/dataset/dataset_add_foreign_view";
import DatasetUploadView from "admin/dataset/dataset_upload_view";
import SampleDatasetsModal from "dashboard/dataset/sample_datasets_modal";
import features from "features";
import renderIndependently from "libs/render_independently";

const { TabPane } = Tabs;

type Props = {
type Props = {|
activeUser: APIUser,
|};
type PropsWithRouter = {|
...Props,
history: RouterHistory,
|};

const renderSampleDatasetsModal = (user: APIUser, history: RouterHistory) => {
renderIndependently(destroy => (
<SampleDatasetsModal
organizationName={user.organization}
destroy={destroy}
onOk={() => history.push("/dashboard/datasets")}
/>
));
};

const DatasetAddView = ({ history }: Props) => (
<Tabs defaultActiveKey="1" className="container">
<TabPane
tab={
<span>
<Icon type="upload" />
Upload Dataset
</span>
}
key="1"
>
<DatasetUploadView
onUploaded={(organization: string, datasetName: string) => {
const url = `/datasets/${organization}/${datasetName}/import`;
history.push(url);
}}
/>
</TabPane>
{features().addForeignDataset ? (
const DatasetAddView = ({ history, activeUser }: PropsWithRouter) => (
<React.Fragment>
<Tabs defaultActiveKey="1" className="container">
<TabPane
tab={
<span>
<Icon type="bars" />
Add foreign Dataset
<Icon type="upload" />
Upload Dataset
</span>
}
key="2"
key="1"
>
<DatasetAddForeignView onAdded={() => history.push("/dashboard")} />
<DatasetUploadView
onUploaded={(organization: string, datasetName: string) => {
const url = `/datasets/${organization}/${datasetName}/import`;
history.push(url);
}}
/>
</TabPane>
) : null}
</Tabs>
{features().addForeignDataset ? (
<TabPane
tab={
<span>
<Icon type="bars" />
Add foreign Dataset
</span>
}
key="2"
>
<DatasetAddForeignView onAdded={() => history.push("/dashboard")} />
</TabPane>
) : null}
</Tabs>
<div style={{ textAlign: "center" }}>
<p>or</p>
<p>
<a href="#" onClick={() => renderSampleDatasetsModal(activeUser, history)}>
Add a Sample Dataset
</a>
</p>
</div>
</React.Fragment>
);

export default withRouter(DatasetAddView);
const mapStateToProps = (state: OxalisState) => ({
activeUser: enforceActiveUser(state.activeUser),
});

export default connect<Props, {||}, _, _, _, _>(mapStateToProps)(withRouter(DatasetAddView));
Loading