From 38dd390e08aafccf12b95ac3c5cf98e9f1ce1f0c Mon Sep 17 00:00:00 2001 From: Kye Date: Fri, 20 Oct 2023 13:06:11 -0400 Subject: [PATCH] bioclip --- pyproject.toml | 1 + requirements.txt | 1 + swarms/__init__.py | 2 +- swarms/models/__init__.py | 4 +- swarms/models/base.py | 4 +- swarms/models/bioclip.py | 170 ++++++++++++++++++++++++++ swarms/models/layoutlm_document_qa.py | 15 ++- swarms/models/nougat.py | 3 +- 8 files changed, 185 insertions(+), 15 deletions(-) create mode 100644 swarms/models/bioclip.py diff --git a/pyproject.toml b/pyproject.toml index a17c6fa0..b06d4b93 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -55,6 +55,7 @@ open-interpreter = "*" tabulate = "*" termcolor = "*" black = "*" +open_clip_torch = "*" dalle3 = "*" torchvision = "*" rich = "*" diff --git a/requirements.txt b/requirements.txt index b2c28df0..23331763 100644 --- a/requirements.txt +++ b/requirements.txt @@ -12,6 +12,7 @@ wget==3.2 simpleaichat httpx torch +open_clip_torch ggl beautifulsoup4 google-search-results==2.4.2 diff --git a/swarms/__init__.py b/swarms/__init__.py index a93fed25..2ecf2033 100644 --- a/swarms/__init__.py +++ b/swarms/__init__.py @@ -12,7 +12,7 @@ from swarms import workers from swarms.workers.worker import Worker # from swarms import chunkers -from swarms.models import * # import * only works when __all__ = [] is defined in __init__.py +from swarms.models import * # import * only works when __all__ = [] is defined in __init__.py from swarms import structs from swarms import swarms from swarms import agents diff --git a/swarms/models/__init__.py b/swarms/models/__init__.py index c5b787e5..20ff2dfe 100644 --- a/swarms/models/__init__.py +++ b/swarms/models/__init__.py @@ -12,6 +12,7 @@ from swarms.models.kosmos_two import Kosmos from swarms.models.vilt import Vilt from swarms.models.nougat import Nougat from swarms.models.layoutlm_document_qa import LayoutLMDocumentQA + # from swarms.models.fuyu import Fuyu # Not working, wait until they update @@ -29,6 +30,3 @@ __all__ = [ "Nougat", "LayoutLMDocumentQA", ] - - - diff --git a/swarms/models/base.py b/swarms/models/base.py index 6f0e9ecb..32a45c43 100644 --- a/swarms/models/base.py +++ b/swarms/models/base.py @@ -84,10 +84,10 @@ class AbstractModel(ABC): _tokens_per_second = self._tokens_per_second() _num_tokens = self._num_tokens(self.history) _time_for_generation = self._time_for_generation(self.history) - + return f""" SEC TO FIRST TOKEN: {_sec_to_first_token} TOKENS/SEC: {_tokens_per_second} TOKENS: {_num_tokens} Tokens/SEC: {_time_for_generation} - """ \ No newline at end of file + """ diff --git a/swarms/models/bioclip.py b/swarms/models/bioclip.py new file mode 100644 index 00000000..937634e3 --- /dev/null +++ b/swarms/models/bioclip.py @@ -0,0 +1,170 @@ +""" + + +BiomedCLIP-PubMedBERT_256-vit_base_patch16_224 +https://huggingface.co/microsoft/BiomedCLIP-PubMedBERT_256-vit_base_patch16_224 +BiomedCLIP is a biomedical vision-language foundation model that is pretrained on PMC-15M, +a dataset of 15 million figure-caption pairs extracted from biomedical research articles in PubMed Central, using contrastive learning. It uses PubMedBERT as the text encoder and Vision Transformer as the image encoder, with domain-specific adaptations. It can perform various vision-language processing (VLP) tasks such as cross-modal retrieval, image classification, and visual question answering. BiomedCLIP establishes new state of the art in a wide range of standard datasets, and substantially outperforms prior VLP approaches: + + + +Citation +@misc{https://doi.org/10.48550/arXiv.2303.00915, + doi = {10.48550/ARXIV.2303.00915}, + url = {https://arxiv.org/abs/2303.00915}, + author = {Zhang, Sheng and Xu, Yanbo and Usuyama, Naoto and Bagga, Jaspreet and Tinn, Robert and Preston, Sam and Rao, Rajesh and Wei, Mu and Valluri, Naveen and Wong, Cliff and Lungren, Matthew and Naumann, Tristan and Poon, Hoifung}, + title = {Large-Scale Domain-Specific Pretraining for Biomedical Vision-Language Processing}, + publisher = {arXiv}, + year = {2023}, +} + +Model Use +How to use +Please refer to this example notebook. + +Intended Use +This model is intended to be used solely for (I) future research on visual-language processing and (II) reproducibility of the experimental results reported in the reference paper. + +Primary Intended Use +The primary intended use is to support AI researchers building on top of this work. BiomedCLIP and its associated models should be helpful for exploring various biomedical VLP research questions, especially in the radiology domain. + +Out-of-Scope Use +Any deployed use case of the model --- commercial or otherwise --- is currently out of scope. Although we evaluated the models using a broad set of publicly-available research benchmarks, the models and evaluations are not intended for deployed use cases. Please refer to the associated paper for more details. + +Data +This model builds upon PMC-15M dataset, which is a large-scale parallel image-text dataset for biomedical vision-language processing. It contains 15 million figure-caption pairs extracted from biomedical research articles in PubMed Central. It covers a diverse range of biomedical image types, such as microscopy, radiography, histology, and more. + +Limitations +This model was developed using English corpora, and thus can be considered English-only. + +Further information +Please refer to the corresponding paper, "Large-Scale Domain-Specific Pretraining for Biomedical Vision-Language Processing" for additional details on the model training and evaluation. +""" + +import open_clip +import glob +import torch +from PIL import Image +import matplotlib.pyplot as plt + + +class BioClip: + """ + BioClip + + Args: + model_path (str): path to the model + + Attributes: + model_path (str): path to the model + model (torch.nn.Module): the model + preprocess_train (torchvision.transforms.Compose): the preprocessing pipeline for training + preprocess_val (torchvision.transforms.Compose): the preprocessing pipeline for validation + tokenizer (open_clip.Tokenizer): the tokenizer + device (torch.device): the device to run the model on + + Methods: + __call__(self, img_path: str, labels: list, template: str = 'this is a photo of ', context_length: int = 256): + returns a dictionary of labels and their probabilities + plot_image_with_metadata(img_path: str, metadata: dict): plots the image with the metadata + + Usage: + clip = BioClip('hf-hub:microsoft/BiomedCLIP-PubMedBERT_256-vit_base_patch16_224') + + labels = [ + 'adenocarcinoma histopathology', + 'brain MRI', + 'covid line chart', + 'squamous cell carcinoma histopathology', + 'immunohistochemistry histopathology', + 'bone X-ray', + 'chest X-ray', + 'pie chart', + 'hematoxylin and eosin histopathology' + ] + + result = clip("your_image_path.jpg", labels) + metadata = {'filename': "your_image_path.jpg".split('/')[-1], 'top_probs': result} + clip.plot_image_with_metadata("your_image_path.jpg", metadata) + + + """ + + def __init__(self, model_path: str): + self.model_path = model_path + ( + self.model, + self.preprocess_train, + self.preprocess_val, + ) = open_clip.create_model_and_transforms(model_path) + self.tokenizer = open_clip.get_tokenizer(model_path) + self.device = ( + torch.device("cuda") if torch.cuda.is_available() else torch.device("cpu") + ) + self.model.to(self.device) + self.model.eval() + + def __call__( + self, + img_path: str, + labels: list, + template: str = "this is a photo of ", + context_length: int = 256, + ): + image = torch.stack([self.preprocess_val(Image.open(img_path))]).to(self.device) + texts = self.tokenizer( + [template + l for l in labels], context_length=context_length + ).to(self.device) + + with torch.no_grad(): + image_features, text_features, logit_scale = self.model(image, texts) + logits = ( + (logit_scale * image_features @ text_features.t()) + .detach() + .softmax(dim=-1) + ) + sorted_indices = torch.argsort(logits, dim=-1, descending=True) + logits = logits.cpu().numpy() + sorted_indices = sorted_indices.cpu().numpy() + + results = {} + for idx in sorted_indices[0]: + label = labels[idx] + prob = logits[0][idx] + results[label] = prob + return results + + @staticmethod + def plot_image_with_metadata(img_path: str, metadata: dict): + img = Image.open(img_path) + fig, ax = plt.subplots(figsize=(5, 5)) + ax.imshow(img) + ax.axis("off") + title = ( + metadata["filename"] + + "\n" + + "\n".join([f"{k}: {v*100:.1f}" for k, v in metadata["top_probs"].items()]) + ) + ax.set_title(title, fontsize=14) + plt.tight_layout() + plt.show() + + +# Usage +# clip = BioClip('hf-hub:microsoft/BiomedCLIP-PubMedBERT_256-vit_base_patch16_224') + +# labels = [ +# 'adenocarcinoma histopathology', +# 'brain MRI', +# 'covid line chart', +# 'squamous cell carcinoma histopathology', +# 'immunohistochemistry histopathology', +# 'bone X-ray', +# 'chest X-ray', +# 'pie chart', +# 'hematoxylin and eosin histopathology' +# ] + +# result = clip("your_image_path.jpg", labels) +# metadata = {'filename': "your_image_path.jpg".split('/')[-1], 'top_probs': result} +# clip.plot_image_with_metadata("your_image_path.jpg", metadata) diff --git a/swarms/models/layoutlm_document_qa.py b/swarms/models/layoutlm_document_qa.py index a5f7a32c..6fe83210 100644 --- a/swarms/models/layoutlm_document_qa.py +++ b/swarms/models/layoutlm_document_qa.py @@ -5,6 +5,7 @@ visual question answering on real world docs lik invoice, pdfs, etc from transformers import pipeline from swarms.models.base import AbstractModel + class LayoutLMDocumentQA(AbstractModel): """ LayoutLMDocumentQA for document question answering: @@ -18,20 +19,18 @@ class LayoutLMDocumentQA(AbstractModel): >>> model = LayoutLMDocumentQA() >>> out = model("What is the total amount?", "path/to/img.png") >>> print(out) - + """ + def __init__( - self, + self, model_name: str = "impira/layoutlm-document-qa", task: str = "document-question-answering", ): - self.pipeline = pipeline( - self.task, - model=self.model_name - ) - + self.pipeline = pipeline(self.task, model=self.model_name) + def __call__(self, task: str, img_path: str): """Call for model""" out = self.pipeline(img_path, task) out = str(out) - return out \ No newline at end of file + return out diff --git a/swarms/models/nougat.py b/swarms/models/nougat.py index 4bd99f1a..cc154283 100644 --- a/swarms/models/nougat.py +++ b/swarms/models/nougat.py @@ -28,8 +28,9 @@ class Nougat: >>> nougat = Nougat() >>> nougat("path/to/image.png") - + """ + def __init__( self, model_name_or_path="facebook/nougat-base",