2015-09-25 34 views
17

Ich habe Schwierigkeiten mit einer Java-basierten Lambda-Funktion, um Nachrichten von SNS zu empfangen. Meine Funktion sieht wie folgt:AWS Lambda NoClassDefFoundError

package com.mycompany; 

import com.amazonaws.services.lambda.runtime.Context; 
import com.amazonaws.services.lambda.runtime.LambdaLogger; 
import com.amazonaws.services.lambda.runtime.events.SNSEvent; 

public class LambdaHandler { 
    public void Handler(SNSEvent event, Context context) { 
     //Process the event 
    } 
} 

Es kompiliert ganz gut, und ich habe keine Probleme die JAR-Datei zu Lambda Hochladen (über die Web-Konsole).

Wenn jedoch ich es (über SNS bis zur gezeichneten Lambda-Funktion) veröffentlichen mit JSON das SNSEvent Modell darstellt, löst die Lambda-Funktion die folgende Ausnahme:

Error loading method handler on class com.mycompany.LambdaHandler: class java.lang.NoClassDefFoundError java.lang.NoClassDefFoundError: com/amazonaws/services/lambda/runtime/events/SNSEvent at

java.lang.Class.getDeclaredMethods0(Native Method) at java.lang.Class.privateGetDeclaredMethods(Class.java:2701) at java.lang.Class.privateGetPublicMethods(Class.java:2902) at java.lang.Class.getMethods(Class.java:1615) Caused by: java.lang.ClassNotFoundException: com.amazonaws.services.lambda.runtime.events.SNSEvent at java.net.URLClassLoader.findClass(URLClassLoader.java:381) at java.lang.ClassLoader.loadClass(ClassLoader.java:424) at java.lang.ClassLoader.loadClass(ClassLoader.java:357)

I verwenden Maven + Netbeans und es ist ein Maven Java Anwendungsprojekt. Ich habe die Funktion von der Lambda-Konsole heruntergeladen und bestätigt, dass das jar über ein lib/-Verzeichnis mit allen Jars für die Importe verfügt, einschließlich aws-lambda-java-events-1.1.0.jar, das selbst die/com/amazonaws enthält /services/lambda/runtime/events/SNSEvent.class Datei.

Warum kann die Laufzeitumgebung die Klasse nicht finden, wenn sie definitiv in der JAR-Datei enthalten ist? Muss ich noch etwas anderes tun, irgendwelche Umgebungsvariablen setzen usw.?

Jede Hilfe wäre willkommen!

EDIT 1 Ich versuchte aws-Lambda-java-1.0.0 Ereignisse und es ist die Berichterstattung immer noch die gleiche Ausnahme Herabstufung. Wie unten angegeben, ist meine POM-Datei (mit nur geändertem Projektnamen). Ich weiß nicht, wie ich Maven sagen soll, dass er die Bibliotheken in eine Baumstruktur einfügen soll.

<?xml version="1.0" encoding="UTF-8"?> 
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> 
    <modelVersion>4.0.0</modelVersion> 
    <groupId>com.app</groupId> 
    <artifactId>Handler</artifactId> 
    <version>1.0-SNAPSHOT</version> 
    <packaging>jar</packaging> 
    <dependencies> 
     <dependency> 
      <groupId>com.amazonaws</groupId> 
      <artifactId>aws-java-sdk-lambda</artifactId> 
      <version>1.10.6</version> 
     </dependency> 
     <dependency> 
      <groupId>com.amazonaws</groupId> 
      <artifactId>aws-lambda-java-core</artifactId> 
      <version>1.0.0</version> 
     </dependency> 
     <dependency> 
      <groupId>com.amazonaws</groupId> 
      <artifactId>aws-lambda-java-events</artifactId> 
      <version>1.0.0</version> 
     </dependency> 
    </dependencies> 
    <properties> 
     <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding> 
     <maven.compiler.source>1.8</maven.compiler.source> 
     <maven.compiler.target>1.8</maven.compiler.target> 
    </properties> 
</project> 
+0

Können Sie den relevanten Abschnitt Ihrer Maven-Datei teilen? Meine vergleichbare funktionierende Lambda-Funktion bündelte die aws-lambda-java-events-1.1.0.jar nicht unter lib, sondern hatte die SNSEvent.class-Datei als com/amazonaws/services/lambda/runtime/events/SNSEvents.class. – James

+0

Ich kann heute Abend, wenn ich nach Hause komme, aber was meinst du damit hat die SNSEvent.class-Datei "loose"? Wie funktioniert es, wenn es die Laufzeitumgebung nicht mit dem jar der Bibliothek bereitstellt? – Brooks

+0

Mein Projekt bündelt das Glas nicht in meinem Glas. Es bündelt die kompilierten Klassen in meinem Jar in der Klassenpfad-Ordnerhierarchie neben den kompilierten Klassen für meinen Code. – James

Antwort

11

Verwenden der maven-shade plugin so dass der JAR enthält die Abhängigkeiten in einem uber-jar.

So fügen Sie diese zu Ihrer pom.xml

<build> 
<plugins> 
    <plugin> 
    <groupId>org.apache.maven.plugins</groupId> 
    <artifactId>maven-shade-plugin</artifactId> 
    <version>2.3</version> 
    <configuration> 
     <createDependencyReducedPom>false</createDependencyReducedPom> 
    </configuration> 
    <executions> 
     <execution> 
     <phase>package</phase> 
     <goals> 
      <goal>shade</goal> 
     </goals> 
     </execution> 
    </executions> 
    </plugin> 
</plugins> 

Quelle: http://docs.aws.amazon.com/lambda/latest/dg/java-create-jar-pkg-maven-no-ide.html

Potenziell können Sie dieses Problem haben https://github.com/aws/aws-lambda-java-libs/issues/2, die eine Herabstufung auf aws-Lambda-java-Ereignisse erfordert -1.0.0.jar

+2

Leider funktionierte das nicht für mich ... immer noch die gleiche Ausnahme. – Brooks

+1

Ah, habe gerade deine pom.xml gesehen. Versuchen Sie es mit dem maven-shade Plugin wie hier beschrieben: http://docs.aws.amazon.com/lambda/latest/dg/java-create-jar-pkg-maven-no-ide.html – Mike76

+0

Arbeitete! Ich akzeptiere, aber du solltest die URL in deine Antwort schreiben, damit sie noch mehr upvoted werden kann. – Brooks

0

Fügen Sie im Plugins-Abschnitt Ihrer pom.xml das Apache Maven Shade Plugin hinzu. Es wird während des Baus verwendet d Prozess. Dieses Plugin wird zum Verpacken von Jars verwendet, um ein eigenständiges .jar zu erstellen. Das maven-shade-plugin nimmt Artefakte (jars), die vom Paketziel erzeugt werden, und erstellt eine eigenständige .jar, die den kompilierten Code und die aufgelösten Abhängigkeiten von pom.xml.

<dependency> 
<groupId>org.apache.maven.plugins</groupId> 
<artifactId>maven-shade-plugin</artifactId> 
<version>3.0.0</version> 

1

Manchmal müssen Sie Ihre Lambda wieder hochladen. Auch ich habe das gleiche Problem, das ich mit dieser pom.xml behoben:

<properties> 
    <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding> 
    </properties> 

    <dependencyManagement> 
    <dependencies> 
    <dependency> 
     <groupId>com.amazonaws</groupId> 
     <artifactId>aws-java-sdk-bom</artifactId> 
     <version>1.11.83</version> 
     <type>pom</type> 
     <scope>import</scope> 
    </dependency> 
    </dependencies> 

</dependencyManagement> 

    <dependencies> 
    <dependency> 
     <groupId>junit</groupId> 
     <artifactId>junit</artifactId> 
     <version>3.8.1</version> 
     <scope>test</scope> 
    </dependency> 
    </dependencies> 

    <build> 
<plugins> 
    <plugin> 
    <groupId>org.apache.maven.plugins</groupId> 
    <artifactId>maven-shade-plugin</artifactId> 
    <version>2.3</version> 
    <configuration> 
     <createDependencyReducedPom>false</createDependencyReducedPom> 
    </configuration> 
    <executions> 
     <execution> 
     <phase>package</phase> 
     <goals> 
      <goal>shade</goal> 
     </goals> 
     </execution> 
    </executions> 
    </plugin> 
</plugins> 
</build> 
</project> 
+1

Denken Sie daran, Ihre eigenen aws-Abhängigkeiten einzuschließen! –

+0

Ha, "lade einfach das Glas wieder hoch" funktionierte für mich. Vielleicht habe ich zufällig das "Original ... jar" hochgeladen. –