import java.io.FileOutputStream;
import java.io.FileWriter;
import java.io.IOException;
-import java.nio.file.Files;
-import java.nio.file.Paths;
-import java.nio.file.StandardCopyOption;
+import java.nio.file.*;
import java.text.DateFormat;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
private ChecksumAlgorithm[] algorithms = new ChecksumAlgorithm[]{ ChecksumAlgorithm.SHA1, ChecksumAlgorithm.MD5 };
+ private final String FS = FileSystems.getDefault().getSeparator();
+
@Inject
@Named(value = "archivaTaskScheduler#repository")
private ArchivaTaskScheduler scheduler;
//Content-Disposition: form-data; name="files[]"; filename="org.apache.karaf.features.command-2.2.2.jar"
String fileName = file.getContentDisposition().getParameter( "filename" );
+ Path fileNamePath = Paths.get(fileName);
+ if (!fileName.equals(fileNamePath.getFileName().toString())) {
+ ArchivaRestServiceException e = new ArchivaRestServiceException("Bad filename in upload content: " + fileName + " - File traversal chars (..|/) are not allowed"
+ , null);
+ e.setHttpErrorCode(422);
+ e.setErrorKey("error.upload.malformed.filename");
+ throw e;
+ }
File tmpFile = File.createTempFile( "upload-artifact", ".tmp" );
tmpFile.deleteOnExit();
return fileMetadatas == null ? Collections.<FileMetadata>emptyList() : fileMetadatas;
}
+ private boolean hasValidChars(String checkString) {
+ if (checkString.contains(FS)) {
+ return false;
+ }
+ if (checkString.contains("../")) {
+ return false;
+ }
+ if (checkString.contains("/..")) {
+ return false;
+ }
+ return true;
+ }
+
+ private void checkParamChars(String param, String value) throws ArchivaRestServiceException {
+ if (!hasValidChars(value)) {
+ ArchivaRestServiceException e = new ArchivaRestServiceException("Bad characters in " + param, null);
+ e.setHttpErrorCode(422);
+ e.setErrorKey("error.upload.malformed.param." + param);
+ e.setFieldName(param);
+ throw e;
+ }
+ }
+
@Override
public Boolean save( String repositoryId, String groupId, String artifactId, String version, String packaging,
boolean generatePom )
version = StringUtils.trim( version );
packaging = StringUtils.trim( packaging );
+ checkParamChars("repositoryId", repositoryId);
+ checkParamChars("groupId", groupId);
+ checkParamChars("artifactId", artifactId);
+ checkParamChars("packaging", packaging);
+
List<FileMetadata> fileMetadatas = getSessionFilesList();
if ( fileMetadatas == null || fileMetadatas.isEmpty() )
{
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
+import javax.ws.rs.ClientErrorException;
import java.io.File;
import java.io.IOException;
import java.net.URLEncoder;
service.post( body );
fail( "FileNames with path contents should not be allowed." );
}
- catch ( ArchivaRestServiceException e )
+ catch ( ClientErrorException e )
{
- // OK
+ assertEquals(422, e.getResponse().getStatus());
}
}
finally
log.debug( "Metadata {}", meta.toString( ) );
assertTrue( service.save( "internal", "org.archiva", "archiva-model", "1.2", "jar", true ) );
- fileAttachment = new AttachmentBuilder( ).object( Files.newInputStream( file ) ).contentDisposition( new ContentDisposition( "form-data; filename=\"/../TestFile.FileExt\"; name=\"files[]\"" ) ).build( );
+ fileAttachment = new AttachmentBuilder( ).object( Files.newInputStream( file ) ).contentDisposition( new ContentDisposition( "form-data; filename=\"TestFile.FileExt\"; name=\"files[]\"" ) ).build( );
body = new MultipartBody( fileAttachment );
meta = service.post( body );
log.debug( "Metadata {}", meta.toString( ) );
try {
service.save("internal", "org", URLEncoder.encode("../../../test", "UTF-8"), URLEncoder.encode("testSave", "UTF-8"), "4", true);
fail("Error expected, if the content contains bad characters.");
- } catch (ArchivaRestServiceException e) {
- // OK
+ } catch (ClientErrorException e) {
+ assertEquals(422, e.getResponse().getStatus());
}
assertFalse( Files.exists( Paths.get( "target/test-testSave.4" ) ) );
}