Skip to content

Commit

Permalink
Create an Image-to-Text action kamelet (#2171)
Browse files Browse the repository at this point in the history
* Switch to Camel 4.8.0-SNAPSHOT

* Create an Image-to-Text action kamelet
  • Loading branch information
tadayosi authored Aug 29, 2024
1 parent 0db335a commit 2ad4d45
Show file tree
Hide file tree
Showing 7 changed files with 230 additions and 3 deletions.
1 change: 1 addition & 0 deletions docs/modules/ROOT/nav.adoc
Original file line number Diff line number Diff line change
Expand Up @@ -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[]
Expand Down
75 changes: 75 additions & 0 deletions kamelets/djl-image-to-text-action.kamelet.yaml
Original file line number Diff line number Diff line change
@@ -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: {}
Original file line number Diff line number Diff line change
Expand Up @@ -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);
Expand Down
18 changes: 18 additions & 0 deletions library/camel-kamelets-utils/pom.xml
Original file line number Diff line number Diff line change
Expand Up @@ -92,6 +92,24 @@
<scope>provided</scope>
</dependency>

<!-- Dependencies for DJL and ImageNet util classes -->
<dependency>
<groupId>org.apache.camel</groupId>
<artifactId>camel-djl</artifactId>
<scope>provided</scope>
</dependency>
<dependency>
<groupId>net.sf.extjwnl</groupId>
<artifactId>extjwnl</artifactId>
<version>2.0.5</version>
<scope>provided</scope>
</dependency>
<dependency>
<groupId>net.sf.extjwnl</groupId>
<artifactId>extjwnl-data-wn31</artifactId>
<version>1.2</version>
<scope>provided</scope>
</dependency>

<!-- Test scoped dependencies -->
<dependency>
Expand Down
Original file line number Diff line number Diff line change
@@ -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));
}
}
Loading

0 comments on commit 2ad4d45

Please sign in to comment.