Skip to content

Commit

Permalink
feat: Onboard DeepMind AlphaFold DB (#431)
Browse files Browse the repository at this point in the history
  • Loading branch information
adlersantos authored Jul 30, 2022
1 parent 0395c37 commit 02c887e
Show file tree
Hide file tree
Showing 9 changed files with 923 additions and 0 deletions.
70 changes: 70 additions & 0 deletions datasets/deepmind/infra/deepmind_dataset.tf
Original file line number Diff line number Diff line change
@@ -0,0 +1,70 @@
/**
* Copyright 2022 Google LLC
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/


resource "google_storage_bucket" "deepmind-alphafold" {
name = "${var.bucket_name_prefix}-deepmind-alphafold"
force_destroy = true
location = "US"
uniform_bucket_level_access = true
lifecycle {
ignore_changes = [
logging,
]
}
}

data "google_iam_policy" "storage_bucket__deepmind-alphafold" {
dynamic "binding" {
for_each = var.iam_policies["storage_buckets"]["deepmind-alphafold"]
content {
role = binding.value["role"]
members = binding.value["members"]
}
}
}

resource "google_storage_bucket_iam_policy" "deepmind-alphafold" {
bucket = google_storage_bucket.deepmind-alphafold.name
policy_data = data.google_iam_policy.storage_bucket__deepmind-alphafold.policy_data
}
output "storage_bucket-deepmind-alphafold-name" {
value = google_storage_bucket.deepmind-alphafold.name
}

resource "google_bigquery_dataset" "deepmind_alphafold" {
dataset_id = "deepmind_alphafold"
project = var.project_id
description = "Metadata for the AlphaFold Protein Structure Database"
}

data "google_iam_policy" "bq_ds__deepmind_alphafold" {
dynamic "binding" {
for_each = var.iam_policies["bigquery_datasets"]["deepmind_alphafold"]
content {
role = binding.value["role"]
members = binding.value["members"]
}
}
}

resource "google_bigquery_dataset_iam_policy" "deepmind_alphafold" {
dataset_id = google_bigquery_dataset.deepmind_alphafold.dataset_id
policy_data = data.google_iam_policy.bq_ds__deepmind_alphafold.policy_data
}
output "bigquery_dataset-deepmind_alphafold-dataset_id" {
value = google_bigquery_dataset.deepmind_alphafold.dataset_id
}
28 changes: 28 additions & 0 deletions datasets/deepmind/infra/provider.tf
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
/**
* Copyright 2022 Google LLC
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/


provider "google" {
project = var.project_id
impersonate_service_account = var.impersonating_acct
region = var.region
}

data "google_client_openid_userinfo" "me" {}

output "impersonating-account" {
value = data.google_client_openid_userinfo.me.email
}
26 changes: 26 additions & 0 deletions datasets/deepmind/infra/variables.tf
Original file line number Diff line number Diff line change
@@ -0,0 +1,26 @@
/**
* Copyright 2022 Google LLC
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/


variable "project_id" {}
variable "bucket_name_prefix" {}
variable "impersonating_acct" {}
variable "region" {}
variable "env" {}
variable "iam_policies" {
default = {}
}

21 changes: 21 additions & 0 deletions datasets/deepmind/pipelines/_images/sts_jobs_generator/Dockerfile
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
# Copyright 2022 Google LLC
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.

FROM python:3.8
ENV PYTHONUNBUFFERED True
COPY requirements.txt ./
RUN python3 -m pip install --no-cache-dir -r requirements.txt
WORKDIR /custom
COPY . .
CMD ["python3", "script.py"]
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
google-cloud-storage-transfer
google-cloud-storage
128 changes: 128 additions & 0 deletions datasets/deepmind/pipelines/_images/sts_jobs_generator/script.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,128 @@
# Copyright 2022 Google LLC
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.


import logging
import os
import time
import typing

from google.cloud import storage, storage_transfer

POLL_INTERVAL_INC = 60
POLL_INTERVAL_MAX = 600


def main(
manifest_bucket: str,
manifest_prefix: str,
source_bucket: str,
destination_bucket: str,
gcp_project: str,
) -> None:
manifest_files = get_manifest_files(manifest_bucket, manifest_prefix)
_operations = []
for manifest in manifest_files:
_operation = create_transfer_job(
manifest_bucket=manifest_bucket,
manifest=manifest,
source_bucket=source_bucket,
destination_bucket=destination_bucket,
gcp_project=gcp_project,
)
_operations.append(_operation)

poll_interval = 0
while True:
logging.info("Checking transfer job statuses..")

_done = [op.done() for op in _operations]
if all(_done):
# Break the loop when all transfers are done
logging.info(f"All {len(_done)} transfer jobs complete.")
break
else:
# If transfers are still running, wait some time and poll again
poll_interval = min(poll_interval + POLL_INTERVAL_INC, POLL_INTERVAL_MAX)
logging.info(
f"Transfer jobs completed: {_done.count(True)}/{len(_done)} (next check in {poll_interval} seconds)"
)

time.sleep(poll_interval)


def get_manifest_files(bucket: str, prefix: str) -> typing.List[storage.blob.Blob]:
storage_client = storage.Client()
blobs = storage_client.list_blobs(bucket, prefix=f"{prefix}/", delimiter="/")
blobs = list(blobs)
logging.info(f"Found {len(blobs)} manifest files:")
for blob in blobs:
logging.info(f"- {blob.name}")
return blobs


def create_transfer_job(
manifest_bucket: str,
manifest: storage.blob.Blob,
source_bucket: str,
destination_bucket: str,
gcp_project: str,
):
# Create a client
client = storage_transfer.StorageTransferServiceClient()

job_name = f"transferJobs/deepmind-afdb-{manifest.name.split('/')[-1].replace('.csv','')}-{time.time_ns() // 1_000_000}"
logging.info(f"Creating transfer job {job_name}")

# Initialize request argument(s)
request = {
"transfer_job": {
"name": job_name,
"project_id": gcp_project,
"status": storage_transfer.TransferJob.Status.ENABLED,
"transfer_spec": {
"gcs_data_source": {
"bucket_name": source_bucket,
},
"gcs_data_sink": {"bucket_name": destination_bucket},
"transfer_options": {"overwrite_when": "DIFFERENT"},
"transfer_manifest": {
"location": f"gs://{manifest_bucket}/{manifest.name}"
},
},
}
}

client.create_transfer_job(request=request)
logging.info(f"Created transfer job {job_name}")

logging.info(f"Running transfer job {job_name}")
request = storage_transfer.RunTransferJobRequest(
job_name=job_name, project_id=gcp_project
)

operation = client.run_transfer_job(request=request)
return operation


if __name__ == "__main__":
logging.getLogger().setLevel(logging.INFO)

main(
manifest_bucket=os.environ["MANIFEST_BUCKET"],
manifest_prefix=os.environ["MANIFEST_PREFIX"],
source_bucket=os.environ["SOURCE_BUCKET"],
destination_bucket=os.environ["DESTINATION_BUCKET"],
gcp_project=os.environ["GCP_PROJECT"],
)
Loading

0 comments on commit 02c887e

Please sign in to comment.