diff options
author | Jordan Bracco <href@random.sh> | 2023-03-05 10:48:26 +0100 |
---|---|---|
committer | Jordan Bracco <href@random.sh> | 2023-03-05 10:48:26 +0100 |
commit | cf5fecca3c8fc325798ca3858b2ea5e6681251af (patch) | |
tree | 66a2e8c1100d3896c54e2e0156b708c7e3e1a6ef | |
parent | More fixes (diff) |
plugin: link/image: commit py node
Diffstat (limited to '')
-rw-r--r-- | priv/python/.gitignore | 2 | ||||
-rw-r--r-- | priv/python/Makefile | 15 | ||||
-rw-r--r-- | priv/python/pyerlai/__init__.py | 0 | ||||
-rw-r--r-- | priv/python/pyerlai/genservers/__init__.py | 0 | ||||
-rw-r--r-- | priv/python/pyerlai/genservers/clip_ask.py | 65 | ||||
-rw-r--r-- | priv/python/pyerlai/genservers/image_to_text.py | 43 | ||||
-rw-r--r-- | priv/python/pyerlai/genservers/pil.py | 21 | ||||
-rw-r--r-- | priv/python/pyerlai/node.py | 42 | ||||
-rw-r--r-- | priv/python/requirements.txt | 4 |
9 files changed, 192 insertions, 0 deletions
diff --git a/priv/python/.gitignore b/priv/python/.gitignore new file mode 100644 index 0000000..3b72af0 --- /dev/null +++ b/priv/python/.gitignore @@ -0,0 +1,2 @@ +.env +__pycache__ diff --git a/priv/python/Makefile b/priv/python/Makefile new file mode 100644 index 0000000..a9ed39a --- /dev/null +++ b/priv/python/Makefile @@ -0,0 +1,15 @@ +all: .env/done + +.env/done: + python3 -m venv .env + .env/bin/pip3 install -r requirements.txt + touch .env/done + +upgrade: + make clean all + +clean: + rm -rf .env + +node: + .env/bin/python3 pyerlai/node.py diff --git a/priv/python/pyerlai/__init__.py b/priv/python/pyerlai/__init__.py new file mode 100644 index 0000000..e69de29 --- /dev/null +++ b/priv/python/pyerlai/__init__.py diff --git a/priv/python/pyerlai/genservers/__init__.py b/priv/python/pyerlai/genservers/__init__.py new file mode 100644 index 0000000..e69de29 --- /dev/null +++ b/priv/python/pyerlai/genservers/__init__.py diff --git a/priv/python/pyerlai/genservers/clip_ask.py b/priv/python/pyerlai/genservers/clip_ask.py new file mode 100644 index 0000000..0cb40dc --- /dev/null +++ b/priv/python/pyerlai/genservers/clip_ask.py @@ -0,0 +1,65 @@ +from term import Atom +from pyrlang.gen.server import GenServer +from pyrlang.gen.decorators import call, cast, info +from PIL import Image +import io +import sys + +from transformers import CLIPProcessor, CLIPModel + +PROMPTS=[ +"photo", +"dog photo", +"cat photo", +"food photo", +"meme", +"painting", +"drawing", +"selfie", +"portrait photography", +"tv capture", +"screenshot", +"terminal/ssh/console screenshot", +"twitter screenshot", +"chat log", +"4chan screenshot", +"scanned document", +"book picture"] + +class ClipAsk(GenServer): + def __init__(self, node, logger) -> None: + super().__init__() + node.register_name(self, Atom('clip_ask')) + self.logger = logger + self.model = None + self.processor = None + self.ready = False + print("clipask: starting") + mypid = self.pid_ + node.send_nowait(mypid, mypid, "register") + self.logger.info("initialized process: clip_ask.") + + @info(0, lambda msg: msg == 'register') + def setup(self, msg): + print("clipask: doing setup") + self.logger.info("image_to_text_vit_gpt2: setup...") + self.model = CLIPModel.from_pretrained("openai/clip-vit-base-patch32") + self.processor = CLIPProcessor.from_pretrained("openai/clip-vit-base-patch32") + self.logger.info("clip_ask: setup finished.") + self.ready = True + print("clipask: ready") + + @call(1, lambda msg: type(msg) == tuple and msg[0] == Atom("run")) + def run(self, msg): + if self.ready: + self.logger.info("clip_ask: inference") + image = Image.open(io.BytesIO(msg[1])) + inputs = self.processor(text=PROMPTS, images=image, return_tensors="pt", padding=True) + outputs = self.model(**inputs) + logits_per_image = outputs.logits_per_image + probs = logits_per_image.softmax(dim=1) + labels_with_probs = dict(zip(PROMPTS, probs.detach().numpy()[0])) + results = dict(sorted(labels_with_probs.items(), key=lambda item: item[1], reverse=True)) + return (Atom('ok'), {k: v.item() for k, v in results.items()}) + else: + return (Atom('error'), Atom('not_ready')) diff --git a/priv/python/pyerlai/genservers/image_to_text.py b/priv/python/pyerlai/genservers/image_to_text.py new file mode 100644 index 0000000..a1abc74 --- /dev/null +++ b/priv/python/pyerlai/genservers/image_to_text.py @@ -0,0 +1,43 @@ +from term import Atom +from pyrlang.gen.server import GenServer +from pyrlang.gen.decorators import call, cast, info +from PIL import Image +from transformers import GPT2TokenizerFast, ViTImageProcessor, VisionEncoderDecoderModel +import io + +class ImageToTextViTGPT2(GenServer): + def __init__(self, node, logger) -> None: + super().__init__() + node.register_name(self, Atom('image_to_text_vit_gpt2')) + self.logger = logger + self.model = None + self.tokenizer = None + self.image_processor = None + self.ready = False + print("lol") + mypid = self.pid_ + node.send_nowait(mypid, mypid, "register") + self.logger.info("initialized process: text_to_image_vit_gpt2.") + + @info(0, lambda msg: msg == 'register') + def setup(self, msg): + print("doing setup") + self.logger.info("image_to_text_vit_gpt2: setup...") + self.model = VisionEncoderDecoderModel.from_pretrained("nlpconnect/vit-gpt2-image-captioning") + self.tokenizer = GPT2TokenizerFast.from_pretrained("nlpconnect/vit-gpt2-image-captioning") + self.image_processor = ViTImageProcessor.from_pretrained("nlpconnect/vit-gpt2-image-captioning") + self.logger.info("text_to_image_vit_gpt2: setup finished.") + self.ready = True + print("ready") + + @call(1, lambda msg: type(msg) == tuple and msg[0] == Atom("run")) + def run(self, msg): + if self.ready: + self.logger.info("image_to_text_vit_gpt2: inference") + image = Image.open(io.BytesIO(msg[1])).convert('RGB') + pixel_values = self.image_processor(image, return_tensors="pt").pixel_values + generated_ids = self.model.generate(pixel_values, max_new_tokens=40) + generated_text = self.tokenizer.batch_decode(generated_ids, skip_special_tokens=True)[0] + return (Atom('ok'), generated_text) + else: + return (Atom('error'), Atom('not_ready')) diff --git a/priv/python/pyerlai/genservers/pil.py b/priv/python/pyerlai/genservers/pil.py new file mode 100644 index 0000000..13db32f --- /dev/null +++ b/priv/python/pyerlai/genservers/pil.py @@ -0,0 +1,21 @@ +from term import Atom +from pyrlang.gen.server import GenServer +from pyrlang.gen.decorators import call, cast, info +from PIL import Image +import io +import sys + +class Pil(GenServer): + def __init__(self, node, logger) -> None: + super().__init__() + node.register_name(self, Atom('pil')) + self.logger = logger + print("clipask: starting") + mypid = self.pid_ + self.logger.info("initialized process: pil.") + + @call(1, lambda msg: type(msg) == tuple and msg[0] == Atom("run")) + def run(self, msg): + self.logger.info("clip_ask: inference") + image = Image.open(io.BytesIO(msg[1])) + return (Atom('ok'), {"width": image.width, "height": image.height, "animated": getattr(image, "is_animated", False)}) diff --git a/priv/python/pyerlai/node.py b/priv/python/pyerlai/node.py new file mode 100644 index 0000000..b79b526 --- /dev/null +++ b/priv/python/pyerlai/node.py @@ -0,0 +1,42 @@ +import logging +import argparse +import os +import sys +from pyrlang import Node + +from genservers.image_to_text import ImageToTextViTGPT2 +from genservers.clip_ask import ClipAsk +from genservers.pil import Pil + +def main(): + parser = argparse.ArgumentParser() + + parser.add_argument("host_node", help="host node") + parser.add_argument("node", help="python node name") + + parser.add_argument("cookie", help="node cookie. prefer use NODE_COOKIE environment variable.") + parser.add_argument("process", help="process name (pyerlai)", default="pyerlai") + + args = parser.parse_args() + + cookie = os.environ.get('NODE_COOKIE') + if cookie is None: + if args.cookie is None: + sys.exit("pyerlai-python: missing NODE_COOKIE or --cookie") + else: + cookie = args.cookie + + logger = logging.getLogger("pyerlai-python:" + args.node) + logging.getLogger("").setLevel(logging.DEBUG) + + logger.info("starting node") + node = Node(node_name=args.node, cookie=cookie, hidden=True) + + ImageToTextViTGPT2(node, logger) + ClipAsk(node, logger) + Pil(node, logger) + + node.run() + +if __name__ == "__main__": + main() diff --git a/priv/python/requirements.txt b/priv/python/requirements.txt new file mode 100644 index 0000000..a3c9e39 --- /dev/null +++ b/priv/python/requirements.txt @@ -0,0 +1,4 @@ +transformers[torch]>=4.25.1 +pillow>=9.3.0 +git+https://github.com/Pyrlang/Pyrlang@2a65c22#egg=Pyrlang +git+https://github.com/Pyrlang/Term@ee1d770#egg=pyrlang-term |