summaryrefslogtreecommitdiffstats
path: root/src/main/java/org/openslx/taskmanager/tasks/DownloadFile.java
blob: e7f4d800e5b1062df8e65b683b990099c46fb7a3 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
package org.openslx.taskmanager.tasks;

import java.io.BufferedInputStream;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.net.URL;
import java.net.URLConnection;

import org.apache.commons.io.FileUtils;
import org.apache.commons.io.FilenameUtils;
import org.openslx.satserver.util.Exec;
import org.openslx.satserver.util.Util;
import org.openslx.taskmanager.api.AbstractTask;

import com.google.gson.annotations.Expose;

public class DownloadFile extends AbstractTask
{

	@Expose
	private String url = null;
	@Expose
	private String destination = null;
	@Expose
	private String gpg = null;

	private Output status = new Output();

	private static final String[] ALLOWED_DIRS =
	{ "/srv/openslx/www/boot/", "/tmp/" };

	@Override
	protected boolean initTask()
	{
		this.setStatusObject( status );
		if ( this.url == null ) {
			status.error = "No URL given.";
			return false;
		}
		this.destination = FilenameUtils.normalize( this.destination );
		if ( this.destination == null || !Util.startsWith( this.destination, ALLOWED_DIRS ) || this.destination.endsWith( "/" ) ) {
			status.error = "File not in allowed directory";
			return false;
		}
		return true;
	}

	@Override
	protected boolean execute()
	{
		URLConnection connection = null;
		BufferedInputStream in = null;
		FileOutputStream fout = null;

		try {
			File tmpFile = File.createTempFile( "bwlp-", ".tmp", null );
			connection = new URL( this.url ).openConnection();
			in = new BufferedInputStream( connection.getInputStream() );
			fout = new FileOutputStream( tmpFile );
			status.size = connection.getContentLengthLong();
			if ( status.size <= 0 ) // If size is unknown, fake progress...
				status.progress = 10;

			final byte data[] = new byte[ 90000 ];
			int count;
			while ( ( count = in.read( data, 0, data.length ) ) != -1 ) {
				fout.write( data, 0, count );
				status.complete += count;
				if ( status.size > 0 )
					status.progress = (int) ( 100l * status.complete / status.size );
				else if ( status.progress < 99 && System.currentTimeMillis() % 20 == 0 )
					status.progress++;
			}
			fout.close();
			in.close();
			// If we have a gpg sig, validate
			if ( this.gpg != null && !this.gpg.isEmpty() ) {
				File gpgTempFile = null;
				try {
					gpgTempFile = File.createTempFile( "bwlp-", ".gpg", null );
					Util.writeStringToFile( gpgTempFile, this.gpg );
				} catch ( Exception e ) {
					status.error = "Could not create temporary file for gpg signature";
					return false;
				}
				if ( 0 != Exec.sync( 10, "gpg", "--verify", gpgTempFile.getAbsolutePath(), tmpFile.getAbsolutePath() ) ) {
					status.error = "GPG signature of downloaded file not valid!\n\n" + this.gpg;
					return false;
				}
				gpgTempFile.delete();
			}
			// Move file to destination
			File dest = new File( this.destination );
			FileUtils.forceMkdir( new File( dest.getParent() ) );
			if ( dest.exists() )
				dest.delete();
			if ( !tmpFile.renameTo( dest ) ) {
				status.error = "Could not move downloaded file to destination directory!";
				return false;
			}
			return true;
		} catch ( IOException e ) {
			status.error = "Download error: " + e.toString();
			return false;
		} finally {
			Util.multiClose( in, fout );
		}
	}

	/**
	 * Output - contains additional status data of this task
	 */
	@SuppressWarnings( "unused" )
	private static class Output
	{
		protected String error = null;
		protected long size = -1;
		protected long complete = 0;
		protected int progress = 0;
	}

}