Amazon
Detect mitotic figures in whole slide images with Amazon Rekognition
Even after more than a hundred years after its introduction, histology remains the gold standard in tumor diagnosis and prognosis. Anatomic pathologists evaluate histology to stratify cancer patients into different groups depending on their tumor genotypes and phenotypes, and their clinical outcome [1,2]. However, human evaluation of histological slides is subjective and not repeatable [3].…
Published
3 years agoon
By
admin[]Even after more than a hundred years after its introduction, histology remains the gold standard in tumor diagnosis and prognosis. Anatomic pathologists evaluate histology to stratify cancer patients into different groups depending on their tumor genotypes and phenotypes, and their clinical outcome [1,2]. However, human evaluation of histological slides is subjective and not repeatable [3]. Furthermore, histological assessment is a time-consuming process that requires highly trained professionals.
[]With significant technological advances in the last decade, techniques such as whole slide imaging (WSI) and deep learning (DL) are now widely available. WSI is the scanning of conventional microscopy glass slides to produce a single, high-resolution image from those slides. This allows for the digitization and collection of large sets of pathology images, which would have been prohibitively time-consuming and expensive. The availability of such datasets creates new and innovative ways of accelerating diagnosis by using techniques such as machine learning (ML) to aid pathologists to accelerate diagnoses by quickly identifying features of interest.
[]In this post, we will explore how developers without previous ML experience can use Amazon Rekognition Custom Labels to train a model that classifies cellular features. Amazon Rekognition Custom Labels is a feature of Amazon Rekognition that enables you to build your own specialized ML-based image analysis capabilities to detect unique objects and scenes integral to your specific use case. In particular, we use a dataset containing whole slide images of canine mammary carcinoma [1] to demonstrate how to process these images and train a model that detects mitotic figures. This dataset has been used with permission from Prof. Dr. Marc Aubreville, who has kindly agreed to allow us to use it for this post. For more information, see the Acknowledgments section at the end of this post.
Solution Overview
[]The solution consists of two components:
- An Amazon Rekognition Custom Labels model — To enable Amazon Rekognition to detect mitotic figures, we complete the following steps:
- Sample the WSI dataset to produce adequately sized images using Amazon SageMaker Studio and a Python code running on a Jupyter notebook. Studio is a web-based, integrated development environment (IDE) for ML that provides all the tools you need to take your models from experimentation to production while boosting your productivity. We will use Studio to split the images into smaller ones to train our model.
- Train a Amazon Rekognition Custom Labels model to recognize mitotic figures in hematoxylin-eosin samples using the data prepared in the previous step.
- A frontend application — To demonstrate how to use a model like the one we trained in the previous step, we complete the following steps:
[]The following diagram illustrates the solution architecture.
[]All the necessary resources to deploy the implementation discussed in this post and the code for the whole section are available on GitHub. You can clone or fork the repository, make any changes you desire, and run it yourself.
[]In the next steps, we walk through the code to understand the different steps involved in obtaining and preparing the data, training the model, and using it from a sample application.
Costs
[]When running the steps in this walkthrough, you incur small costs from using the following AWS services:
- Amazon Rekognition
- AWS Fargate
- Application Load Balancer
- AWS Secrets Manager
[]Additionally, if no longer within the Free Tier period or conditions, you may incur costs from the following services:
- CodePipeline
- CodeBuild
- Amazon ECR
- Amazon SageMaker
[]If you complete the cleanup steps correctly after finishing this walkthrough, you may expect costs to be less than 10 USD, if the Amazon Rekognition Custom Labels model and the web application run for one hour or less.
Prerequisites
[]To complete all steps, you need the following:
Training the mitotic figure classification model
[]We run all the steps required to train the model from a Studio notebook. If you have never used Studio before, you may need to onboard first. For more information, see Onboard Quickly to Amazon SageMaker Studio.
[]Some of the following steps require more RAM than what is available in a standard ml.t3.medium notebook. Make sure that you have selected an ml.m5.large notebook. You should see a 2 vCPU + 8 GiB indication on the upper right corner of the page.
[]The code for this section is available as a Jupyter notebook file.
[]After onboarding to Studio, follow these instructions to grant Studio the necessary permissions to call Amazon Rekognition on your behalf.
Dependencies
[]To begin with, we need to complete the following steps:
- Update Linux packages and install the required dependencies, such as OpenSlide: !apt update > /dev/null && apt dist-upgrade -y > /dev/null !apt install -y build-essential openslide-tools python-openslide libgl1-mesa-glx > /dev/null
- Install the fastai and SlideRunner libraries using pip: !pip install SlideRunner SlideRunner_dataAccess fastai==1.0.61 > /dev/null
- Download the dataset (we provide a script to do this automatically): from dataset import download_dataset download_dataset()
Process the dataset
[]We will begin by importing some of the packages that we use throughout the data preparation stage. Then, we download and load the annotation database for this dataset. This database contains the positions in the whole slide images of the mitotic figures (the features we want to classify). See the following code:
%reload_ext autoreload %autoreload 2 import os from typing import List import urllib import numpy as np from SlideRunner.dataAccess.database import Database from pathlib import Path DATABASE_URL = ‘https://github.com/DeepPathology/MITOS_WSI_CMC/raw/master/databases/MITOS_WSI_CMC_MEL.sqlite’ DATABASE_FILENAME = ‘MITOS_WSI_CMC_MEL.sqlite’ Path(“./databases”).mkdir(parents=True, exist_ok=True) local_filename, headers = urllib.request.urlretrieve( DATABASE_URL, filename=os.path.join(‘databases’, DATABASE_FILENAME), ) []Because we’re using SageMaker, we create a new SageMaker session object to ease tasks such as uploading our dataset to an Amazon Simple Storage Service (Amazon S3) bucket. We also use the S3 bucket that SageMaker creates by default to upload our processed image files.
[]The slidelist_test array contains the IDs of the slides that we use as part of the test dataset to evaluate the performance of the trained model. See the following code:
import sagemaker sm_session = sagemaker.Session() size=512 bucket_name = sm_session.default_bucket() database = Database() database.open(os.path.join(‘databases’, DATABASE_FILENAME)) slidelist_test = [’14’,’18’,’3′,’22’,’10’,’15’,’21’] []The next step is to obtain a set of areas of training and test slides, along with the labels in them, from which we can take smaller areas to use to train our model. The code for get_slides is in the sampling.py file in GitHub.
from sampling import get_slides image_size = 512 lbl_bbox, training_slides, test_slides, files = get_slides(database, slidelist_test, negative_class=1, size=image_size) []We want to randomly sample from the training and test slides. We use the lists of training and test slides and randomly select n_training_images times a file for training, and n_test_images times a file for test:
n_training_images = 500 n_test_images = int(0.2 * n_training_images) training_files = list([ (y, files[y]) for y in np.random.choice( [x for x in training_slides], n_training_images) ]) test_files = list([ (y, files[y]) for y in np.random.choice( [x for x in test_slides], n_test_images) ]) []Next, we create a directory for training images and one for test images:
Path(“rek_slides/training”).mkdir(parents=True, exist_ok=True) Path(“rek_slides/test”).mkdir(parents=True, exist_ok=True) []Before we produce the smaller images needed to train the model, we need some helper code that produces the metadata needed to describe the training and test data. The following code makes sure that a given bounding box surrounding the features of interest (mitotic figures) are well within the zone we’re cutting, and produces a line of JSON that describes the image and the features in it in Amazon SageMaker Ground Truth format, which is the format Amazon Rekognition Custom Labels requires. For more information about this manifest file for object detection, see Object localization in manifest files.
def check_bbox(x_start: int, y_start: int, bbox) -> bool: return (bbox._left > x_start and bbox._right < x_start + image_size and bbox._top > y_start and bbox._bottom < y_start + image_size) def get_annotation_json_line(filename, channel, annotations, labels): objects = list([{'confidence' : 1} for i in range(0, len(annotations))]) return json.dumps({ 'source-ref': f's3://{bucket_name}/data/{channel}/{filename}', 'bounding-box': { 'image_size': [{ 'width': size, 'height': size, 'depth': 3 }], 'annotations': annotations, }, 'bounding-box-metadata': { 'objects': objects, 'class-map': dict({ x: str(x) for x in labels }), 'type': 'groundtruth/object-detection', 'human-annotated': 'yes', 'creation-date': datetime.datetime.now().isoformat(), 'job-name': 'rek-pathology', } }) def generate_annotations(x_start: int, y_start: int, bboxes, labels, filename: str, channel: str): annotations = [] for bbox in bboxes: if check_bbox(x_start, y_start, bbox): # Get coordinates relative to this slide. x0 = bbox.left - x_start y0 = bbox.top - y_start annotation = { 'class_id': 1, 'top': y0, 'left': x0, 'width': bbox.right - bbox.left, 'height': bbox.bottom - bbox.top } annotations.append(annotation) return get_annotation_json_line(filename, channel, annotations, labels) []With the generate_annotations function in place, we can write the code to produce the training and test images:
import datetime import json import random from fastai import * from fastai.vision import * from tqdm.notebook import tqdm # Margin size, in pixels, for training images. This is the space we leave on # each side for the bounding box(es) to be well into the image. margin_size = 64 training_annotations = [] test_annotations = [] def check_bbox(x_start: int, y_start: int, bbox) -> bool: return (bbox._left > x_start and bbox._right < x_start + image_size and bbox._top > y_start and bbox._bottom < y_start + image_size) def generate_images(file_list) -> None: for f_idx in tqdm(range(0, len(file_list)), desc=’Writing training images…’): slide_idx, f = file_list[f_idx] bboxes = lbl_bbox[slide_idx][0] labels = lbl_bbox[slide_idx][1] # Calculate the minimum and maximum horizontal and vertical positions # that bounding boxes should have within the image. x_min = min(map(lambda x: x.left, bboxes)) – margin_size y_min = min(map(lambda x: x.top, bboxes)) – margin_size x_max = max(map(lambda x: x.right, bboxes)) + margin_size y_max = max(map(lambda x: x.bottom, bboxes)) + margin_size result = False while not result: x_start = random.randint(x_min, x_max – image_size) y_start = random.randint(y_min, y_max – image_size) for bbox in bboxes: if check_bbox(x_start, y_start, bbox): result = True break filename = f’slide_{f_idx}.png’ channel = ‘test’ if slide_idx in test_slides else ‘training’ annotation = generate_annotations(x_start, y_start, bboxes, labels, filename, channel) if channel == ‘training’: training_annotations.append(annotation) else: test_annotations.append(annotation) img = Image(pil2tensor(f.get_patch(x_start, y_start) / 255., np.float32)) img.save(f’rek_slides/{channel}/{filename}’) generate_images(training_files) generate_images(test_files) []The last step towards having all of the required data is to write a manifest.json file for each of the datasets:
with open(‘rek_slides/training/manifest.json’, ‘w’) as mf: mf.write(“n”.join(training_annotations)) with open(‘rek_slides/test/manifest.json’, ‘w’) as mf: mf.write(“n”.join(test_annotations))
Transfer the files to S3
[]We use the upload_data method that the SageMaker session object exposes to upload the images and manifest files to the default SageMaker S3 bucket:
import sagemaker sm_session = sagemaker.Session() data_location = sm_session.upload_data( ‘./rek_slides’, bucket=bucket_name, )
Train an Amazon Rekognition Custom Labels model
[]With the data already in Amazon S3, we can get to training a custom model. We use the Boto3 library to create an Amazon Rekognition client and create a project:
import boto3 project_name = ‘rek-mitotic-figures-workshop’ rek = boto3.client(‘rekognition’) response = rek.create_project(ProjectName=project_name) # If you have already created the project, use the describe_projects call to # retrieve the project ARN. # response = rek.describe_projects()[‘ProjectDescriptions’][0] project_arn = response[‘ProjectArn’] []With the project ready to use, now you need a project version that points to the training and test datasets in Amazon S3. Each version ideally points to different datasets (or different versions of it). This enables us to have different versions of a model, compare their performance, and switch between them as needed. See the following code:
version_name = ‘1’ output_config = { ‘S3Bucket’: bucket_name, ‘S3KeyPrefix’: ‘output’, } training_dataset = { ‘Assets’: [ { ‘GroundTruthManifest’: { ‘S3Object’: { ‘Bucket’: bucket_name, ‘Name’: ‘data/training/manifest.json’ } }, }, ] } testing_dataset = { ‘Assets’: [ { ‘GroundTruthManifest’: { ‘S3Object’: { ‘Bucket’: bucket_name, ‘Name’: ‘data/test/manifest.json’ } }, }, ] } def describe_project_versions(): describe_response = rek.describe_project_versions( ProjectArn=project_arn, VersionNames=[version_name], ) for model in describe_response[‘ProjectVersionDescriptions’]: print(f”Status: {model[‘Status’]}”) print(f”Message: {model[‘StatusMessage’]}”) return describe_response response = rek.create_project_version( VersionName=version_name, ProjectArn=project_arn, OutputConfig=output_config, TrainingData=training_dataset, TestingData=testing_dataset, ) waiter = rek.get_waiter(‘project_version_training_completed’) waiter.wait( ProjectArn=project_arn, VersionNames=[version_name], ) describe_response = describe_project_versions() []After we create the project version, Amazon Rekognition automatically starts the training process. The training time depends on several features, such as the size of the images and the number of them, the number of classes, and so on. In this case, for 500 images, the training takes about 90 minutes to finish.
Test the model
[]After training, every model in Amazon Rekognition Custom Labels is in the STOPPED state. To use it for inference, you need to start it. We get the project version ARN from the project version description and pass it over to the start_project_version. Notice the MinInferenceUnits parameter — we start with one inference unit. The actual maximum number of transactions per second (TPS) that this inference unit supports depends on the complexity of your model. To learn more about TPS, refer to this blog post.
model_arn = describe_response[‘ProjectVersionDescriptions’][0][‘ProjectVersionArn’] response = rek.start_project_version( ProjectVersionArn=model_arn, MinInferenceUnits=1, ) waiter = rek.get_waiter(‘project_version_running’) waiter.wait( ProjectArn=project_arn, VersionNames=[version_name], ) []When your project version is listed as RUNNING, you can start sending images to Amazon Rekognition for inference.
[]We use one of the files in the test dataset to test the newly started model. You can use any suitable PNG or JPEG file instead.
from matplotlib import pyplot as plt from PIL import Image, ImageDraw # We’ll use one of our test images to try out our model. with open(‘./rek_slides/test/slide_0.png’, ‘rb’) as image_file: image_bytes=image_file.read() # Send the image data to the model. response = rek.detect_custom_labels( ProjectVersionArn=model_arn, Image={ ‘Bytes’: image_bytes } ) img = Image.open(io.BytesIO(image_bytes)) draw = ImageDraw.Draw(img) for custom_label in response[‘CustomLabels’]: geometry = custom_label[‘Geometry’][‘BoundingBox’] w = geometry[‘Width’] * img.width h = geometry[‘Height’] * img.height l = geometry[‘Left’] * img.width t = geometry[‘Top’] * img.height draw.rectangle([l, t, l + w, t + h], outline=(0, 0, 255, 255), width=5) plt.imshow(np.asarray(img))
Streamlit application
[]To demonstrate the integration with Amazon Rekognition, we use a very simple Python application. We use the Streamlit library to build a spartan user interface, where we prompt the user to upload an image file.
[]We use the Boto3 library and the detect_custom_labels method, together with the project version ARN, to invoke the inference endpoint. The response is a JSON document that contains the positions and classes of the different objects detected in the image. In our case, these are the mitotic figures that the algorithm has found in the image we sent to the endpoint. See the following code:
import os import boto3 import io import streamlit as st from PIL import Image, ImageDraw rek_client = boto3.client(‘rekognition’) uploaded_file = st.file_uploader(‘Image file’) if uploaded_file is not None: image_bytes = uploaded_file.read() result = rek_client.detect_custom_labels( ProjectVersionArn=’
Deploy the application to AWS
[]To deploy the application, we use an AWS CDK script. The whole project can be found on GitHub . Let’s look at the different resources deployed by the script.
Create an Amazon ECR repository
[]As the first step towards setting up our deployment, we create an Amazon ECR repository, where we can store our application container images:
aws ecr create-repository –repository-name rek-wsi
Create and store your GitHub token in AWS Secrets Manager
[]CodePipeline needs a GitHub Personal Access Token to monitor your GitHub repository for changes and pull code. To create the token, follow the instructions in the GitHub documentation. The token requires the following GitHub scopes:
- The repo scope, which is used for full control to read and pull artifacts from public and private repositories into a pipeline.
- The admin:repo_hook scope, which is used for full control of repository hooks.
[]After creating the token, store it in a new secret in AWS Secrets Manager as follows:
aws secretsmanager create-secret –name rek-wsi/github –secret-string “{“oauthToken”:”YOUR-TOKEN-VALUE-HERE”}”
Write configuration parameters to AWS Systems Manager Parameter Store
[]The AWS CDK script reads some configuration parameters from AWS Systems Manager Parameter Store, such as the name and owner of the GitHub repository, and target account and Region. Before launching the AWS CDK script, you need to create these parameters in your own account.
[]You can do that by using the AWS CLI. Simply invoke the put-parameter command with a name, a value, and the type of the parameter:
aws ssm put-parameter –name
- /rek_wsi/prod/accountId — The ID of the account where we deploy the application.
- /rek_wsi/prod/ecr_repo_name — The name of the Amazon ECR repository where the container images are stored.
- /rek_wsi/prod/github/branch — The branch in the GitHub repository from which CodePipeline needs to pull the code.
- /rek_wsi/prod/github/owner — The owner of the GitHub repository.
- /rek_wsi/prod/github/repo — The name of the GitHub repository where our code is stored.
- /rek_wsi/prod/github/token — The name or ARN of the secret in Secrets Manager that contains your GitHub authentication token. This is necessary for CodePipeline to be able to communicate with GitHub.
- /rek_wsi/prod/region — The region where we will deploy the application.
[]Notice the prod segment in all parameter names. Although we do not need this level of detail for such a simple example, it will enable to reuse this approach with other projects where different environments may be necessary.
Resources created by the AWS CDK script
[]We need our application, running in a Fargate task, to have permissions to invoke Amazon Rekognition. So we first create an AWS Identity and Access Management (IAM) Task Role with the RekognitionReadOnlyPolicy policy attached to it. Notice that the assumed_by parameter in the following code takes the ecs-tasks.amazonaws.com service principal. This is because we’re using Amazon ECS as the orchestrator, so we need Amazon ECS to assume the role and pass the credentials to the Fargate task.
streamlit_task_role = iam.Role( self, ‘StreamlitTaskRole’, assumed_by=iam.ServicePrincipal(‘ecs-tasks.amazonaws.com’), description=’ECS Task Role assumed by the Streamlit task deployed to ECS+Fargate’, managed_policies=[ iam.ManagedPolicy.from_managed_policy_arn( self, ‘RekognitionReadOnlyPolicy’, managed_policy_arn=’arn:aws:iam::aws:policy/AmazonRekognitionReadOnlyAccess’ ), ], ) []Once built, our application container image sits in a private Amazon ECR repository. We need an object that describes it that we can pass when creating the Fargate service:
ecs_container_image = ecs.ContainerImage.from_ecr_repository( repository=ecr.Repository.from_repository_name(self, ‘ECRRepo’, ‘rek-wsi’), tag=’latest’ ) []We create a new VPC and cluster for this application. You can modify this part to use your own VPC by using the from_lookup method of the Vpc class:
vpc = ec2.Vpc(self, ‘RekWSI’, max_azs=3) cluster = ecs.Cluster(self, ‘RekWSICluster’, vpc=vpc) []Now that we have a VPC and cluster to deploy to, we create the Fargate service. We use 0.25 vCPU and 512 MB RAM for this task, and we place a public Application Load Balancer (ALB) in front of it. Once deployed, we use the ALB CNAME to access the application. See the following code:
fargate_service = ecs_patterns.ApplicationLoadBalancedFargateService( self, ‘RekWSIECSApp’, cluster=cluster, cpu=256, memory_limit_mib=512, desired_count=1, task_image_options=ecs_patterns.ApplicationLoadBalancedTaskImageOptions( image=ecs_container_image, container_port=8501, task_role=streamlit_task_role, ), public_load_balancer=True, ) []To automatically build and deploy a new container image every time we push code to our main branch, we create a simple pipeline consisting of a GitHub source action and a build step. Here is where we use the secrets we stored in AWS Secrets Manager and AWS Systems Manager Parameter Store in the previous steps.
pipeline = codepipeline.Pipeline(self, ‘RekWSIPipeline’) # Create an artifact that points at the code pulled from GitHub. source_output = codepipeline.Artifact() # Create a source stage that pulls the code from GitHub. The repo parameters are # stored in SSM, and the OAuth token in Secrets Manager. source_action = codepipeline_actions.GitHubSourceAction( action_name=’GitHub’, output=source_output, oauth_token=SecretValue.secrets_manager( ssm.StringParameter.value_from_lookup(self, ‘/rek_wsi/prod/github/token’), json_field=’oauthToken’), trigger=codepipeline_actions.GitHubTrigger.WEBHOOK, owner=ssm.StringParameter.value_from_lookup(self, ‘/rek_wsi/prod/github/owner’), repo=ssm.StringParameter.value_from_lookup(self, ‘/rek_wsi/prod/github/repo’), branch=ssm.StringParameter.value_from_lookup(self, ‘/rek_wsi/prod/github/branch’), ) # Add the source stage to the pipeline. pipeline.add_stage( stage_name=’GitHub’, actions=[source_action] ) []CodeBuild needs permissions to push container images to Amazon ECR. To grant these permissions, we add the AmazonEC2ContainerRegistryFullAccess policy to a bespoke IAM role that the CodeBuild service principal can assume:
# Create an IAM role that grants CodeBuild access to Amazon ECR to push containers. build_role = iam.Role( self, ‘RekWsiCodeBuildAccessRole’, assumed_by=iam.ServicePrincipal(‘codebuild.amazonaws.com’), ) # Permissions are granted through an AWS managed policy, AmazonEC2ContainerRegistryFullAccess. managed_ecr_policy = iam.ManagedPolicy.from_managed_policy_arn( self, ‘cb_ecr_policy’, managed_policy_arn=’arn:aws:iam::aws:policy/AmazonEC2ContainerRegistryFullAccess’, ) build_role.add_managed_policy(policy=managed_ecr_policy) []The CodeBuild project logs into the private Amazon ECR repository, builds the Docker image with the Streamlit application, and pushes the image into the repository together with an appspec.yaml and an imagedefinitions.json file.
[]The appspec.yaml file describes the task (port, Fargate platform version, and so on), while the imagedefinitions.json file maps the names of the container images to their corresponding Amazon ECR URI. See the following code:
container_name = fargate_service.task_definition.default_container.container_name build_project = codebuild.PipelineProject( self, ‘RekWSIProject’, build_spec=codebuild.BuildSpec.from_object({ ‘version’: ‘0.2’, ‘phases’: { ‘pre_build’: { ‘commands’: [ ‘env’, ‘COMMIT_HASH=$(echo $CODEBUILD_RESOLVED_SOURCE_VERSION | cut -c 1-7)’, ‘export TAG=${COMMIT_HASH:=latest}’, ‘aws ecr get-login-password –region $AWS_DEFAULT_REGION | ‘ ‘docker login –username AWS ‘ ‘–password-stdin $AWS_ACCOUNT_ID.dkr.ecr.$AWS_DEFAULT_REGION.amazonaws.com’, ] }, ‘build’: { ‘commands’: [ # Build the Docker image ‘cd streamlit_app && docker build -t $IMAGE_REPO_NAME:$IMAGE_TAG .’, # Tag the image ‘docker tag $IMAGE_REPO_NAME:$IMAGE_TAG ‘ ‘$AWS_ACCOUNT_ID.dkr.ecr.$AWS_DEFAULT_REGION.amazonaws.com/$IMAGE_REPO_NAME:$IMAGE_TAG’, ] }, ‘post_build’: { ‘commands’: [ # Push the container into ECR. ‘docker push ‘ ‘$AWS_ACCOUNT_ID.dkr.ecr.$AWS_DEFAULT_REGION.amazonaws.com/$IMAGE_REPO_NAME:$IMAGE_TAG’, # Generate imagedefinitions.json ‘cd ..’, “printf ‘[{“name”:”%s”,”imageUri”:”%s”}]’ ” f”{container_name} ” “$AWS_ACCOUNT_ID.dkr.ecr.$AWS_DEFAULT_REGION.amazonaws.com/$IMAGE_REPO_NAME:$IMAGE_TAG ” “> imagedefinitions.json”, ‘ls -l’, ‘pwd’, ‘sed -i s”|REGION_NAME|$AWS_DEFAULT_REGION|g” appspec.yaml’, ‘sed -i s”|ACCOUNT_ID|$AWS_ACCOUNT_ID|g” appspec.yaml’, ‘sed -i s”|TASK_NAME|$IMAGE_REPO_NAME|g” appspec.yaml’, f’sed -i s”|CONTAINER_NAME|{container_name}|g” appspec.yaml’, ] } }, ‘artifacts’: { ‘files’: [ ‘imagedefinitions.json’, ‘appspec.yaml’, ], }, }), environment=codebuild.BuildEnvironment( build_image=codebuild.LinuxBuildImage.STANDARD_5_0, privileged=True, ), environment_variables={ ‘AWS_ACCOUNT_ID’: codebuild.BuildEnvironmentVariable(value=self.account), ‘IMAGE_REPO_NAME’: codebuild.BuildEnvironmentVariable( value=ssm.StringParameter.value_from_lookup(self, ‘/rek_wsi/prod/ecr_repo_name’)), ‘IMAGE_TAG’: codebuild.BuildEnvironmentVariable(value=’latest’), }, role=build_role, ) []Finally, we put the different pipeline stages together. The last action is the EcsDeployAction, which takes the container image built in the previous stage and does a rolling update of the tasks in our ECS cluster:
# Create an artifact to store the build output. build_output = codepipeline.Artifact() # Create a build action that ties the build project, the source artifact from the # previous stage, and the output artifact together. build_action = codepipeline_actions.CodeBuildAction( action_name=’Build’, project=build_project, input=source_output, outputs=[build_output], ) # Add the build stage to the pipeline. pipeline.add_stage( stage_name=’Build’, actions=[build_action] ) deploy_action = codepipeline_actions.EcsDeployAction( action_name=’Deploy’, service=fargate_service.service, # image_file=build_output input=build_output, ) pipeline.add_stage( stage_name=’Deploy’, actions=[deploy_action], )
Cleanup
[]To avoid incurring future costs, clean up the resources you created as part of this solution.
Amazon Rekognition Custom Labels model
[]Before you shut down your Studio notebook, make sure you stop the Amazon Rekognition Custom Labels model. If you don’t, it continues to incur costs.
rek.stop_project_version( ProjectVersionArn=model_arn, ) []Alternatively, you can use the Amazon Rekognition console to stop the service:
- On the Amazon Rekognition console, choose Use Custom Labels in the navigation pane.
- Choose Projects in the navigation pane.
- Choose version 1 of the rek-mitotic-figures-workshop project.
- On the Use Model tab, choose Stop.
Streamlit application
[]To destroy all resources associated to the Streamlit application, run the following code from the AWS CDK application directory:
AWS Secrets Manager
[]To delete the GitHub token, follow the instructions in the documentation.
Conclusion
[]In this post, we walked through the necessary steps to train a Amazon Rekognition Custom Labels model for a digital pathology application using real-world data. We then learned how to use the model from a simple application deployed from a CI/CD pipeline to Fargate.
[]Amazon Rekognition Custom Labels enables you to build ML-enabled healthcare applications that you can easily build and deploy using services like Fargate, CodeBuild, and CodePipeline.
[]Can you think of any applications to help researchers, doctors, or their patients to make their lives easier? If so, use the code in this walkthrough to build your next application. And if you have any questions, please share them in the comments section.
Acknowledgments
[]We would like to thank Prof. Dr. Marc Aubreville for kindly giving us permission to use the MITOS_WSI_CMC dataset for this blog post. The dataset can be found on GitHub.
References
[][1] Aubreville, M., Bertram, C.A., Donovan, T.A. et al. A completely annotated whole slide image dataset of canine breast cancer to aid human breast cancer research. Sci Data 7, 417 (2020). https://doi.org/10.1038/s41597-020-00756-z
[][2] Khened, M., Kori, A., Rajkumar, H. et al. A generalized deep learning framework for whole-slide image segmentation and analysis. Sci Rep 11, 11579 (2021). https://doi.org/10.1038/s41598-021-90444-8
[][3] PNAS March 27, 2018 115 (13) E2970-E2979; first published March 12, 2018; https://doi.org/10.1073/pnas.1717139115
About the Author
[]Pablo Nuñez Pölcher, MSc, is a Senior Solutions Architect working for the Public Sector team with Amazon Web Services. Pablo focuses on helping healthcare public sector customers build new, innovative products on AWS in accordance with best practices. He received his M.Sc. in Biological Sciences from Universidad de Buenos Aires. In his spare time, he enjoys cycling and tinkering with ML-enabled embedded devices.
[]Razvan Ionasec, PhD, MBA, is the technical leader for healthcare at Amazon Web Services in Europe, Middle East, and Africa. His work focuses on helping healthcare customers solve business problems by leveraging technology. Previously, Razvan was the global head of artificial intelligence (AI) products at Siemens Healthineers in charge of AI-Rad Companion, the family of AI-powered and cloud-based digital health solutions for imaging. He holds 30+ patents in AI/ML for medical imaging and has published 70+ international peer-reviewed technical and clinical publications on computer vision, computational modeling, and medical image analysis. Razvan received his PhD in Computer Science from the Technical University Munich and MBA from University of Cambridge, Judge Business School.
You may like
Amazon
Amazon EC2 P5e instances are generally available
In this post, we discuss the core capabilities of Amazon Elastic Compute Cloud (Amazon EC2) P5e instances and the use cases they’re well-suited for. We walk you through an example of how to get started with these instances and carry out inference deployment of Meta Llama 3.1 70B and 405B models on them. Source
Published
37 mins agoon
September 10, 2024By
adminIn this post, we discuss the core capabilities of Amazon Elastic Compute Cloud (Amazon EC2) P5e instances and the use cases they’re well-suited for. We walk you through an example of how to get started with these instances and carry out inference deployment of Meta Llama 3.1 70B and 405B models on them.
Source
Amazon
AWS Weekly Roundup: Amazon DynamoDB, AWS AppSync, Storage Browser for Amazon S3, and more (September 9, 2024)
Last week, the latest AWS Heroes arrived! AWS Heroes are amazing technical experts who generously share their insights, best practices, and innovative solutions to help others. The AWS GenAI Lofts are in full swing with San Francisco and São Paulo open now, and London, Paris, and Seoul coming in the next couple of months. Here’s…
Published
2 hours agoon
September 10, 2024By
adminLast week, the latest AWS Heroes arrived! AWS Heroes are amazing technical experts who generously share their insights, best practices, and innovative solutions to help others. The AWS GenAI Lofts are in full swing with San Francisco and São Paulo open now, and London, Paris, and Seoul coming in the next couple of months. Here’s […]
Source
Amazon
Align Meta Llama 3 to human preferences with DPO, Amazon SageMaker Studio, and Amazon SageMaker Ground Truth
In this post, we show you how to enhance the performance of Meta Llama 3 8B Instruct by fine-tuning it using direct preference optimization (DPO) on data collected with SageMaker Ground Truth. Source
Published
2 hours agoon
September 10, 2024By
adminIn this post, we show you how to enhance the performance of Meta Llama 3 8B Instruct by fine-tuning it using direct preference optimization (DPO) on data collected with SageMaker Ground Truth.
Source