diff --git a/docs/modules/ROOT/nav.adoc b/docs/modules/ROOT/nav.adoc index 35629fc9b..162a9a634 100644 --- a/docs/modules/ROOT/nav.adoc +++ b/docs/modules/ROOT/nav.adoc @@ -62,6 +62,7 @@ * xref:cron-source.adoc[] * xref:data-type-action.adoc[] * xref:delay-action.adoc[] +* xref:djl-image-to-text-action.adoc[] * xref:dns-dig-action.adoc[] * xref:dns-ip-action.adoc[] * xref:dns-lookup-action.adoc[] diff --git a/kamelets/djl-image-to-text-action.kamelet.yaml b/kamelets/djl-image-to-text-action.kamelet.yaml new file mode 100644 index 000000000..7031d9f71 --- /dev/null +++ b/kamelets/djl-image-to-text-action.kamelet.yaml @@ -0,0 +1,75 @@ +# --------------------------------------------------------------------------- +# Licensed to the Apache Software Foundation (ASF) under one or more +# contributor license agreements. See the NOTICE file distributed with +# this work for additional information regarding copyright ownership. +# The ASF licenses this file to You 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. +# --------------------------------------------------------------------------- +apiVersion: camel.apache.org/v1 +kind: Kamelet +metadata: + name: djl-image-to-text-action + annotations: + camel.apache.org/kamelet.support.level: "Preview" + camel.apache.org/catalog.version: "4.8.0-SNAPSHOT" + camel.apache.org/kamelet.icon: "" + camel.apache.org/provider: "Apache Software Foundation" + camel.apache.org/kamelet.group: "Actions" + camel.apache.org/kamelet.namespace: "AI" + labels: + camel.apache.org/kamelet.type: "action" +spec: + definition: + title: "Image-to-Text Action" + description: Detect and classify objects in an image into texts using the SSD and ResNet models and the ImageNet dataset. + type: object + types: + out: + mediaType: application/json + dependencies: + - "mvn:ai.djl.pytorch:pytorch-engine:0.29.0" + - "mvn:ai.djl.pytorch:pytorch-model-zoo:0.29.0" + - "mvn:net.sf.extjwnl:extjwnl:2.0.5" + - "mvn:net.sf.extjwnl:extjwnl-data-wn31:1.2" + - "mvn:org.apache.camel.kamelets:camel-kamelets-utils:4.8.0-SNAPSHOT" + - "camel:core" + - "camel:kamelet" + - "camel:jackson" + - "camel:djl" + template: + beans: + - name: imageNetUtil + type: "#class:org.apache.camel.kamelets.utils.djl.ImageNetUtil" + from: + uri: "kamelet:source" + steps: + - to: "djl:cv/object_detection?artifactId=ssd" + - convertBodyTo: "ai.djl.modality.cv.Image[]" + - split: + expression: + simple: "${body}" + aggregationStrategy: "#class:org.apache.camel.processor.aggregate.GroupedBodyAggregationStrategy" + steps: + - to: "djl:cv/image_classification?artifactId=resnet" + # The output from the image classification model is classified + # as one of 1000 labels from WordNet. + # Since it's too fine-grained, we want to find the higher-level + # group (= hypernym) for the classification using the WordNet + # dictionary. + - bean: + ref: "{{imageNetUtil}}" + method: extractClassName + - bean: + ref: "{{imageNetUtil}}" + method: addHypernym + - marshal: + json: {} diff --git a/library/camel-kamelets-catalog/src/test/java/org/apache/camel/kamelets/catalog/KameletsCatalogTest.java b/library/camel-kamelets-catalog/src/test/java/org/apache/camel/kamelets/catalog/KameletsCatalogTest.java index 5056592c7..8307e227f 100644 --- a/library/camel-kamelets-catalog/src/test/java/org/apache/camel/kamelets/catalog/KameletsCatalogTest.java +++ b/library/camel-kamelets-catalog/src/test/java/org/apache/camel/kamelets/catalog/KameletsCatalogTest.java @@ -155,7 +155,7 @@ void testAllKameletDependencies() throws Exception { @Test void testSupportedHeaders() throws Exception { verifyHeaders("aws-s3-source", 20); - verifyHeaders("aws-s3-sink", 27); + verifyHeaders("aws-s3-sink", 29); verifyHeaders("aws-cloudtrail-source", 4); verifyHeaders("aws-redshift-source", 0); verifyHeaders("aws-not-exists", 0); diff --git a/library/camel-kamelets-utils/pom.xml b/library/camel-kamelets-utils/pom.xml index 704d05a9b..e217210ae 100644 --- a/library/camel-kamelets-utils/pom.xml +++ b/library/camel-kamelets-utils/pom.xml @@ -92,6 +92,24 @@ provided + + + org.apache.camel + camel-djl + provided + + + net.sf.extjwnl + extjwnl + 2.0.5 + provided + + + net.sf.extjwnl + extjwnl-data-wn31 + 1.2 + provided + diff --git a/library/camel-kamelets-utils/src/main/java/org/apache/camel/kamelets/utils/djl/ImageNetUtil.java b/library/camel-kamelets-utils/src/main/java/org/apache/camel/kamelets/utils/djl/ImageNetUtil.java new file mode 100644 index 000000000..26dad8808 --- /dev/null +++ b/library/camel-kamelets-utils/src/main/java/org/apache/camel/kamelets/utils/djl/ImageNetUtil.java @@ -0,0 +1,58 @@ +/* + * Licensed to the Apache Software Foundation (ASF) under one or more + * contributor license agreements. See the NOTICE file distributed with + * this work for additional information regarding copyright ownership. + * The ASF licenses this file to You 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. + */ +package org.apache.camel.kamelets.utils.djl; + +import java.util.List; + +import org.apache.camel.Exchange; +import org.apache.camel.RuntimeCamelException; + +import ai.djl.modality.Classifications; +import net.sf.extjwnl.data.IndexWord; +import net.sf.extjwnl.data.POS; +import net.sf.extjwnl.data.PointerUtils; +import net.sf.extjwnl.data.list.PointerTargetNodeList; +import net.sf.extjwnl.dictionary.Dictionary; + +/** + * A utility bean class for handling ImageNet (https://image-net.org/) classifications. + */ +public class ImageNetUtil { + + public ImageNetUtil() { + } + + public void extractClassName(Exchange exchange) { + Classifications body = exchange.getMessage().getBody(Classifications.class); + String className = body.best().getClassName().split(",")[0].split(" ", 2)[1]; + exchange.getMessage().setBody(className); + } + + public void addHypernym(Exchange exchange) throws Exception { + String className = exchange.getMessage().getBody(String.class); + Dictionary dic = Dictionary.getDefaultResourceInstance(); + IndexWord word = dic.getIndexWord(POS.NOUN, className); + if (word == null) { + throw new RuntimeCamelException("Word not found: " + className); + } + PointerTargetNodeList hypernyms = PointerUtils.getDirectHypernyms(word.getSenses().get(0)); + String hypernym = hypernyms.stream() + .map(h -> h.getSynset().getWords().get(0).getLemma()) + .findFirst().orElse(className); + exchange.getMessage().setBody(List.of(className, hypernym)); + } +} diff --git a/library/camel-kamelets/src/main/resources/kamelets/djl-image-to-text-action.kamelet.yaml b/library/camel-kamelets/src/main/resources/kamelets/djl-image-to-text-action.kamelet.yaml new file mode 100644 index 000000000..7031d9f71 --- /dev/null +++ b/library/camel-kamelets/src/main/resources/kamelets/djl-image-to-text-action.kamelet.yaml @@ -0,0 +1,75 @@ +# --------------------------------------------------------------------------- +# Licensed to the Apache Software Foundation (ASF) under one or more +# contributor license agreements. See the NOTICE file distributed with +# this work for additional information regarding copyright ownership. +# The ASF licenses this file to You 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. +# --------------------------------------------------------------------------- +apiVersion: camel.apache.org/v1 +kind: Kamelet +metadata: + name: djl-image-to-text-action + annotations: + camel.apache.org/kamelet.support.level: "Preview" + camel.apache.org/catalog.version: "4.8.0-SNAPSHOT" + camel.apache.org/kamelet.icon: "" + camel.apache.org/provider: "Apache Software Foundation" + camel.apache.org/kamelet.group: "Actions" + camel.apache.org/kamelet.namespace: "AI" + labels: + camel.apache.org/kamelet.type: "action" +spec: + definition: + title: "Image-to-Text Action" + description: Detect and classify objects in an image into texts using the SSD and ResNet models and the ImageNet dataset. + type: object + types: + out: + mediaType: application/json + dependencies: + - "mvn:ai.djl.pytorch:pytorch-engine:0.29.0" + - "mvn:ai.djl.pytorch:pytorch-model-zoo:0.29.0" + - "mvn:net.sf.extjwnl:extjwnl:2.0.5" + - "mvn:net.sf.extjwnl:extjwnl-data-wn31:1.2" + - "mvn:org.apache.camel.kamelets:camel-kamelets-utils:4.8.0-SNAPSHOT" + - "camel:core" + - "camel:kamelet" + - "camel:jackson" + - "camel:djl" + template: + beans: + - name: imageNetUtil + type: "#class:org.apache.camel.kamelets.utils.djl.ImageNetUtil" + from: + uri: "kamelet:source" + steps: + - to: "djl:cv/object_detection?artifactId=ssd" + - convertBodyTo: "ai.djl.modality.cv.Image[]" + - split: + expression: + simple: "${body}" + aggregationStrategy: "#class:org.apache.camel.processor.aggregate.GroupedBodyAggregationStrategy" + steps: + - to: "djl:cv/image_classification?artifactId=resnet" + # The output from the image classification model is classified + # as one of 1000 labels from WordNet. + # Since it's too fine-grained, we want to find the higher-level + # group (= hypernym) for the classification using the WordNet + # dictionary. + - bean: + ref: "{{imageNetUtil}}" + method: extractClassName + - bean: + ref: "{{imageNetUtil}}" + method: addHypernym + - marshal: + json: {} diff --git a/pom.xml b/pom.xml index 218db32e7..925171b55 100644 --- a/pom.xml +++ b/pom.xml @@ -23,7 +23,7 @@ org.apache.camel camel-dependencies - 4.7.0 + 4.8.0-SNAPSHOT org.apache.camel.kamelets @@ -62,7 +62,7 @@ 0.16.1 2.8.1 - 4.7.0 + 4.8.0-SNAPSHOT 2.4.0 4.2.0