Datei aus Multipart parsen und speichern, seltsam codiert?

larsemann

Mitglied
Hallo,

ich habe ein echt seltsames Problem, dass ich einfach nicht gelöst bekomme.

Ein Webservice-API gibt Multipart-Nachrichten zurück und wenn ich den entsprechenden _GET-Aufruf im Browser mache, erhalte ich eine "völlig" andere Datei, als wenn ich den _GET-Aufruf mit Java mache.

Das Framework, was ich benutze ist Spring. Hat jemand von euch schon mal diesen "Effekt" gehabt?

Hier der Ausschnitt, den der Browser liefert:
----------------------------525641684547342045109852
Content-Disposition: form-data; name="name"

bild1.jpg
----------------------------525641684547342045109852
Content-Disposition: form-data; name="bild1"; filename="bild1.jpg"
Content-Type: image/jpeg

ˇÿˇ‡JFIFˇ˛;CREATOR: gd-jpeg v1.0 (using IJG JPEG v62), quality = 95
ˇ€C   
[..]
----------------------------525641684547342045109852

Das speichert mein Java-Programm:
----------------------------525641684547342045109852
Content-Disposition: form-data; name="name"

bild1.jpg
----------------------------525641684547342045109852
Content-Disposition: form-data; name="bild1"; filename="bild1.jpg"
Content-Type: image/jpeg

ÿØÿàJFIFÿþ;CREATOR: gd-jpeg v1.0 (using IJG JPEG v62), quality = 95
[..]
----------------------------525641684547342045109852


Dabei geht es mir natürlich auch um das Bild. Ich kann per Java das Bild einfach nicht aus dem Multipart extrahieren, die erzeugte .jpg-Datei ist immer "corrupt".

Das Browser-Ergebnis funktioniert immer...


Vielleicht kann mir hier jemand auf die Sprünge helfen :/
 

larsemann

Mitglied
Guten Morgen,

Code:
         String url = "localhost:8888/getImage?id=1";
         URL navUrl = new URL(url);
         URLConnection con = (URLConnection)navUrl.openConnection();
         result = getContent(con);
         [..]

Update:

Wenn ich mit einem Rest-Client die Seite aufrufe, zeigt er mir auch immer das zuvor gepostete untere Ergebnis an :/

Update 2: ein Direktaufruf m. Curl erzeugt das korrekte Format wie erwartet
 
Zuletzt bearbeitet:

DrZoidberg

Top Contributor
Das Problem muss in der getContent Methode liegen. Wahrscheinlich interpretiert sie die Antwort als Text. Das kann passieren wenn z.B. ein InputStreamReader anstelle eines InputStream verwendet wird.
 

larsemann

Mitglied
Du hast Recht, ich habe wirklich String benutzt.

Kann ich den Content nicht auch einfach in einen ByteArray speichern und dann per ByteArrayInputStream weiterbenutzen?

Grüße,
 

larsemann

Mitglied
Code:
			String requestUrl = "http://localhost:8888/getImage?id=1";
			RestTemplate restTemplate = new RestTemplate();


			restTemplate.getMessageConverters().add(new ByteArrayHttpMessageConverter());
			HttpEntity<String> entity = new HttpEntity<String>("test");
			ResponseEntity<byte[]> response = restTemplate.exchange(requestUrl, HttpMethod.GET, entity, byte[].class);

			Path path = Paths.get("/tmp/tmp_file");
			Files.write(path, response.getBody());

Jetzt enthält tmp_file wieder den falschen Output .... Langsam dreh ich noch durch.

Das Parsen ist auch eine eher lästige Aufgabe. Gibt es da nichts "Fertiges"?
 
Zuletzt bearbeitet:

larsemann

Mitglied
Hi,

genau den Ansatz habe ich gewählt!

Hier mein Quelltext:

Code:
		try {


			String requestUrl = "http://localhost:8888/getImage?id=1";
			RestTemplate restTemplate = new RestTemplate();

			restTemplate.getMessageConverters().add(new ByteArrayHttpMessageConverter());
			HttpEntity<String> entity = new HttpEntity<String>("test");
			ResponseEntity<byte[]> response = restTemplate.exchange(requestUrl, HttpMethod.GET, entity, byte[].class);

                        // zwischenspeichern ....
			Path path = Paths.get("/tmp/tmp-file");
			Files.write(path, response.getBody());
			Scanner scanner = new Scanner(path);
			byte[] boundary = scanner.nextLine().getBytes();
			System.out.println("boundary: "+ boundary);

			MultipartStream multipartStream = new MultipartStream(new ByteArrayInputStream(response.getBody()), boundary);

			boolean nextPart = multipartStream.skipPreamble();


			while(nextPart) {
				String headers = multipartStream.readHeaders();
				System.out.println("Headers:"+ headers);

				nextPart = multipartStream.readBoundary();
			}

		} catch (Exception e) {
			log.info("exception ", e);
		}

führt zu:

boundary: [45, 45, 45, 45, 45, 45, 45, 45, 45, 45, 45, 45, 45, 45, 45, 45, 45, 45, 45, 55, 50, 57, 51, 48, 49, 53, 52, 53, 51, 54, 54, 55, 54, 52, 51, 53, 49, 48, 49, 57, 49, 55, 54]
Headers: Content-Disposition: form-data; name="my_value"
exception
org.apache.commons.fileupload.MultipartStream$MalformedStreamException: Stream ended unexpectedly
at org.apache.commons.fileupload.MultipartStream.readBoundary(MultipartStream.java:477) ~[commons-fileupload-1.3.1.jar:1.3.1]
at com.neavis.vrac.HomeController.index(Main.java:63) ~[Main.class:na]
[..]

Main.java.:63 ist
nextPart = multipartStream.readBoundary();
 

Barista

Top Contributor
Ich kann Dir da keine besondere Hilfe sein, die FileUpload-Lib hatten wir mal in einem Projekt.

Ich weiss jetzt auch nicht was die Klasse Scanner in diesem Zusammenhang macht.

Ich würde das zwischenspeichern erst mal weglassen.

Eventuell ist auch das boundary falsch.
 

larsemann

Mitglied
Hi,

danke trotzdem!

Habe den Quelltext noch etwas optimiert.

Java:
                        // .. bekannt

			ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
			ByteArrayInputStream byteArrayInputStream = new ByteArrayInputStream(response.getBody());

			String boundaryExample = "----------------------------495985807863582227169678";



				for(int i = 0; i<boundaryExample.length(); i++) {
					byte b = (byte) byteArrayInputStream.read();


					byteArrayOutputStream.write(b);
				}
			System.out.println("boundary: " + byteArrayOutputStream.toString() );

			byteArrayInputStream.reset();

			MultipartStream multipartStream = new MultipartStream(byteArrayInputStream, byteArrayOutputStream.toByteArray());

			boolean nextPart = multipartStream.skipPreamble();


			while(nextPart) {
				String headers = multipartStream.readHeaders();
				System.out.println("Headers: {}", headers);

				nextPart = multipartStream.readBoundary();
			}

Leider nurnoch die Ausgabe des Boundaries.

Naja, ich mache mal morgen weiter.

Grüße,
 
Zuletzt bearbeitet:

DrZoidberg

Top Contributor
Ich bin mir nicht sicher was du falsch machst, aber das hier funktioniert.

Java:
static byte[] getData(String url) throws IOException {
    ByteArrayOutputStream out = new ByteArrayOutputStream();
    InputStream in = new URL(url).openStream();
    int b;
    while((b = in.read()) >= 0) {
        out.write(b);
    }
    in.close();
    return out.toByteArray();
}

static void writeDataToFile(String fileName, byte[] data) throws IOException {
    Files.write(Paths.get(fileName), data);
}

...

byte[] data = getData("http://www.java-forum.org/images/misc/java_forum_org.gif");
writeDataToFile("java_forum.gif", data);
 

larsemann

Mitglied
Hi, danke erstmal!

Jetzt habe ich den Multipart-Teil selbst geparsed, aber immernoch keinen speicherbaren ByteStream, der mir ein Bilder anzeigt :/

Vielleicht kannst du mir hier auch nochmal auf die Sprünge helfen :/

Code:
ByteArrayInputStream byteArrayInputStream = new ByteArrayInputStream(response.getContent()); 
Scanner sc = new Scanner(byteArrayInputStream);
String boundary = sc.nextLine();
Part path;
ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
int part = 0;
while( sc.hasNextLine()) {
    String nextLine = sc.nextLine();
    if ( nextLine.contains(boundary) ) {
    switch(part) {
	case 0:
	     path = Paths.get("/tmp/part_0.json");
        break;
	default :
	     path = Paths.get("/tmp/part_"+part+".jpg");
	break;
     }
	Files.write(path, byteArrayOutputStream.toByteArray());

	byteArrayOutputStream.reset();
	part = part+1;

     } else {					
	byteArrayOutputStream.write(nextLine.getBytes());
    }
}
 
Zuletzt bearbeitet:

DrZoidberg

Top Contributor
Du könntest deine eigene InputStream Klasse definieren.

Java:
class ByteArray {
    byte[] array = new byte[10];
    int size = 0;
    
    void grow() {
        array = Arrays.copyOf(array, array.length*3/2);
    }
    
    void add(byte b) {
        if(size >= array.length) grow();
        array[size++] = b;
    }
    
    void removeLast(int n) {
        size -= n;
    }
    
    byte[] toArray() {
        return Arrays.copyOf(array, size);
    }
}

class DataInputStream {
    final InputStream in;
    
    DataInputStream(InputStream in) {
        this.in = in;
    }
    
    int read() throws IOException {
        return in.read();
    }
    
    byte[] readBytesUntil(String str) throws IOException {
        ByteArray bytes = new ByteArray();
        int matching = 0;
        while(true) {
            int b = in.read();
            if(b < 0) break;
            bytes.add((byte)b);
            if(str.charAt(matching) == (char)b) matching++;
            else matching = 0;
            if(matching == str.length()) {
                bytes.removeLast(str.length());
                break;
            }
        }
        return bytes.toArray();
    }
    
    String readLine() throws IOException {
        StringBuilder builder = new StringBuilder();
        while(true) {
            int b = in.read();
            if(b < 0 || b == '\n') break;
            if(b != '\r') builder.append((char)b);
        }
        return builder.toString();
    }
}

...
byte[] bytes = "abcdef01234".getBytes();    
DataInputStream in = new DataInputStream(new ByteArrayInputStream(bytes));
System.out.println(new String(in.readBytesUntil("23")));
 

larsemann

Mitglied
Danke, dass hat mir echt weitergeholfen.

Ich habe es jetzt etwas klassischer gelöst, glaube ich. Allerdings noch nicht ganz.
Das Problem ist eigentlich, dass meine Quelle kein standardkonformes Multipart erzeugt.
Code:
ByteArrayOutputStream current = new ByteArrayOutputStream();
ByteArrayOutputStream partContent = new ByteArrayOutputStream();
            int part = 0;
String boundary = null;
 
int b;
int prevB = 0;
while( byteArrayInputStream.available() > 0 ) { // byteArrayInputStream 
    current.reset();
    while( (b = byteArrayInputStream.read()) > 0 ) {
 
        if ( b == 10 && prevB == 13 ) { // check if b == LF and prevB == CR
            current.write(10);
            break; // break loop
        }
        else current.write(b);
 
        prevB = b;
    }
 
    if ( boundary == null) { // first line is boundary
        boundary = current.toString();
        boundary = boundary.substring(0, boundary.length()-2); // remove carriage return
    } else {
        if ( current.toString().contains(boundary) ) {
            switch (part) {
                case 0:
                    Files.write(Paths.get("/tmp/json.bin"), partContent.toByteArray()); // first part is a json value
                    break;
                default:
                    Files.write(Paths.get("/tmp/" + part+".bin"), partContent.toByteArray()); // write binary file content
                    break;
                }
 
            partContent.reset();
            part++;
        } else {
            partContent.write(current.toByteArray());
        }
 
    }
}

Das erzeugt ungefähr das, was ich benötige. Allerdings gibt es jetzt einen feinen Unterschied zwischen Quelle und meinem geparsten Ergebnis.
Ich weiss einfach nicht, wieso im Ergebnis alle "^@" verschwunden sind :/
 

Anhänge

  • Screen Shot 2014-02-23 at 20.33.10.jpg
    Screen Shot 2014-02-23 at 20.33.10.jpg
    147,5 KB · Aufrufe: 23
Ähnliche Java Themen
  Titel Forum Antworten Datum
K Warum wird hier nur etwas in eine txt Datei geschrieben und nicht in alle drei (InputStream/OutputStream/Reader/Writer) Java Basics - Anfänger-Themen 1
farbenlos Csv Datei in Java einlesen Java Basics - Anfänger-Themen 18
E Audio Datei unter Bedingungen ausführen Java Basics - Anfänger-Themen 19
S Daten aus Import Datei auslesen und sortieren Java Basics - Anfänger-Themen 2
A exe Datei erstellen Java Basics - Anfänger-Themen 8
J .jar datei öffnen funktioniert nicht Java Basics - Anfänger-Themen 17
P Aus Text Datei nur Zahlen übernehmen Java Basics - Anfänger-Themen 13
P Welches SDK für das erstellen einer ausführbaren Datei? Java Basics - Anfänger-Themen 4
W Fehler in der Datei pom.xml Java Basics - Anfänger-Themen 19
M Verständnisfrage: Warum wird die Datei ohne Inhalt übertragen Java Basics - Anfänger-Themen 3
D Jar Datei startet unter Linux nicht Java Basics - Anfänger-Themen 3
P Probleme mit NetBeans: Wie lässt sich jar. Datei an einem MacBook öffnen Java Basics - Anfänger-Themen 21
N Programm Funktioniert mit .txt Datei aber nicht mit .rtf Datei Java Basics - Anfänger-Themen 2
A Wie führe ich eine Batch-Datei von meiner Java-Anwendung aus? Java Basics - Anfänger-Themen 18
D Java Programm mit Batch-Datei starten Java Basics - Anfänger-Themen 32
W Objekte einer ArrayList in txt-datei schreiben mit Paths? Java Basics - Anfänger-Themen 2
E TIF Datei auslesen Java Basics - Anfänger-Themen 2
B von Java/Eclipse verwendete Datei existiert gar nicht? Java Basics - Anfänger-Themen 6
M Spezifischen Wert einer Zeile aus .txt Datei entnehmen Java Basics - Anfänger-Themen 15
B Popups mit Klicksabfangen zumAusfüllen einer .ods Datei Java Basics - Anfänger-Themen 0
M Daten aus .txt Datei einlesen und weiterverarbeiten Java Basics - Anfänger-Themen 80
M RandomAccessFile int und String gleichzeitig in einer Datei Java Basics - Anfänger-Themen 49
I Datei (Bild) Drucken und wie Druckeinstellung speichern? Java Basics - Anfänger-Themen 3
A CSV-Datei Verarbeiten Java Basics - Anfänger-Themen 8
D Downloadfortschritt von Datei über Google Drive API v3 Java Basics - Anfänger-Themen 10
A CSv.Datei einlesen und die werte in zweidemosional Int Array speichern Java Basics - Anfänger-Themen 9
B Den Dateipfad einer Java Datei durch Code in Selbiger finden? Java Basics - Anfänger-Themen 10
S In Datei schreiben in Java? Java Basics - Anfänger-Themen 1
Saiko Zeilen einer Datei einlesen Java Basics - Anfänger-Themen 3
sserio TXT-Datei Auslesen und den Wert jedes Namen ausrechnen etc. Java Basics - Anfänger-Themen 37
sserio Txt Datei einlesen Java Basics - Anfänger-Themen 9
T Printwriter Datei nicht überschreiben Java Basics - Anfänger-Themen 10
berserkerdq2 An selbst ersteller txt Datei immer Text dranhängen, ohne den vorherign Text zu löschen Java Basics - Anfänger-Themen 8
berserkerdq2 Wie gebe ich den Pfad zu einer Datei an, die in einem Ordner in Eclipse ist? Java Basics - Anfänger-Themen 1
D Strings aus Excel-Datei einlesen Java Basics - Anfänger-Themen 2
M Text in Datei schreiben Java Basics - Anfänger-Themen 9
S Datei anlegen Problem! Groß- und Kleinschreibung wird nicht unterschieden Java Basics - Anfänger-Themen 4
J selbst erstellte Datei mit Programm öffnen Java Basics - Anfänger-Themen 10
J int innerhalb einer Datei ändern Java Basics - Anfänger-Themen 1
T208 Text Datei individuell benennen. Java Basics - Anfänger-Themen 5
julian112 Input/Output .gz bzw. .txt Datei Einlesen und Umgang mit Exceptions Java Basics - Anfänger-Themen 1
F Aus eingelesener Datei korrekt Objekte erzeugen Java Basics - Anfänger-Themen 5
E extern Datei von meinem Computer aufmachen Java Basics - Anfänger-Themen 5
H Scripte oder Programmcode aus Datei lesen? Java Basics - Anfänger-Themen 5
E PDF Datei im xfdf-Datei umwandeln und auf dem Laufwerk ablegen Java Basics - Anfänger-Themen 0
J CSV-Datei verarbeiten Java Basics - Anfänger-Themen 27
A Verarbeiten einer Excel Datei durch das java-Programm Java Basics - Anfänger-Themen 3
P Datei einlesen, nach Begriff filtern und in Datei ausgeben. Problem Standardausgabe über Konsole Java Basics - Anfänger-Themen 19
nbergmann Installation unter jdk.java.net: Keine ZIP-Datei zum entpacken Java Basics - Anfänger-Themen 2
J Datei aus Netzwerk auslesen Java Basics - Anfänger-Themen 9
EchtKeineAhnungManchmal hallo habe ein Problem mit einer Datei -> (Zugriff verweigert) Java Basics - Anfänger-Themen 4
EchtKeineAhnungManchmal Controller aus FXML Datei entfernen Java Basics - Anfänger-Themen 49
I Probleme mit OutputStream - Datei lässt sich nicht öffnen Java Basics - Anfänger-Themen 4
Kotelettklopfer Sqlite DB aus Java Datei ansprechen. Java Basics - Anfänger-Themen 147
C XML Datei speichern und laden Java Basics - Anfänger-Themen 18
M Von einem Menü Methode aus anderer Klasse ausführen, die errechnete Werte in Datei schreibt. Java Basics - Anfänger-Themen 8
C XML Datei schreiben Java Basics - Anfänger-Themen 14
S Zufällige ungerade Zeile aus Text-Datei lesen Java Basics - Anfänger-Themen 5
J Wert in einer json Datei ändern und speichern Java Basics - Anfänger-Themen 3
L Java erstellt leere Datei Java Basics - Anfänger-Themen 8
J Json Datei auslesen Java Basics - Anfänger-Themen 4
J In main() Datei geöffnet, von anderer Funktion beschreiben Java Basics - Anfänger-Themen 3
I JAX-WS.... Datei ".ical" zurückgeben.... Wie annotieren? Java Basics - Anfänger-Themen 1
O zweidimensionales array in eine csv-Datei Java Basics - Anfänger-Themen 1
CptK Richtigen Pfad beim einlesen von Datei finden Java Basics - Anfänger-Themen 2
E Pfadangaben bei Ausführbarer Jar Datei Java Basics - Anfänger-Themen 8
J Input/Output Konstruktor ergänzen, der zur Datei mit einem Objekt passt Java Basics - Anfänger-Themen 0
I Datei als Stream aus Ressource laden? Java Basics - Anfänger-Themen 2
LetsSebi Methode, die einen arry von objekten speichert in einer datei Java Basics - Anfänger-Themen 6
R Wie installiere ich Jdownloadersetup.sh datei mit debian Java Basics - Anfänger-Themen 2
J Csv-Datei einlesen Java Basics - Anfänger-Themen 52
H Daten aus einer Datei in eine Liste speichern Java Basics - Anfänger-Themen 23
W Java in Exe Datei umgewandelt, Ressourcen fehlen (Bilder und Audiodateien) Java Basics - Anfänger-Themen 1
A Fehler beim Ausführen einer class Datei Java Basics - Anfänger-Themen 6
L Konstruktor für eine Map-Datei/Map-Datei einlesen Java Basics - Anfänger-Themen 5
S CSV Datei auslesen und anders darstellen Java Basics - Anfänger-Themen 2
O zufälliges Wort aus einer Datei einlesen Java Basics - Anfänger-Themen 32
E Input/Output Datei anhand von Dateinamen einlesen Java Basics - Anfänger-Themen 5
C Datei über relative Pfade einlesen Java Basics - Anfänger-Themen 6
F Auto String mit Array Name aus Datei... oder so ähnlich Java Basics - Anfänger-Themen 4
O Best Practice Datei-Pfad zerlegen Java Basics - Anfänger-Themen 4
N Java CSV Datei auslesen Java Basics - Anfänger-Themen 6
J Ein Wort aus einer Datei zufällig ermitteln Java Basics - Anfänger-Themen 3
S Verbindung von einer Excel Datei zu Java-- Java Basics - Anfänger-Themen 4
G In Datei schreiben Java Basics - Anfänger-Themen 1
J Klassen .class Datei öffnen Java Basics - Anfänger-Themen 31
P Datei einlesen und zurückgeben Java Basics - Anfänger-Themen 5
G String wird nach Einlesen aus Datei nicht erkannt Java Basics - Anfänger-Themen 3
G Problem beim Speichern von Objekten in einer Datei Java Basics - Anfänger-Themen 7
J bundel - Datei wird nicht gefunden. Trotz STRG-SHIFT-J Java Basics - Anfänger-Themen 2
I ArrayList - Methode zum Speichern eines Eintrags in einer Datei Java Basics - Anfänger-Themen 17
C "HelloWorld" - Dateien erstellt, aber ist es eine class-Datei? Java Basics - Anfänger-Themen 2
A Klassen Datei als Kommandozeilenparameter übergeben Java Basics - Anfänger-Themen 8
J Datei auslesen (nur bestimmte Stellen) Java Basics - Anfänger-Themen 2
B Text-Datei nur in Stream speichern und dann ausgeben Java Basics - Anfänger-Themen 3
G Einbinden von Bildern in ausführbare Jar-Datei Java Basics - Anfänger-Themen 25
G If / While Programm (Datei auslesen) Java Basics - Anfänger-Themen 6
M Java: BufferedReader liest nur die Hälfte der Text-Datei. Java Basics - Anfänger-Themen 4
T PNG Datei wird nur manchmal geöffnet Java Basics - Anfänger-Themen 6
S Aus einer .bz2 Datei auslesen Java Basics - Anfänger-Themen 2

Ähnliche Java Themen

Neue Themen


Oben