Spring MultipartFile проверка и преобразование - PullRequest
0 голосов
/ 23 августа 2011

У меня сейчас есть контроллер Spring MVC, который принимает MultipartFile

@RequestMapping(method = RequestMethod.POST)
public String doUpload(@RequestParam("file") final MultipartFile file) {
    /* ... */
}

Файл содержит данные CSV, которые будут использоваться, по одному на строку, для создания списка объектов домена. Это работает.

Я написал конвертер для строковых данных:

class MyObjectConverter implements org.springframework...Converter<String[], MyObject> {
    /* ... */
}

И валидатор для файла

class UploadFileValidator implements org.springframework.validation.Validator { 
    /* ... */
}

И у меня есть форма для загрузки:

<form method="post" 
    action="<@spring.url '/upload'/>" 
    enctype="multipart/form-data">
        <input id="upload" type="file" name="file"/>
        <input type="submit" id="uploadButton"/>
    </form

Но то, что я действительно хочу сделать, это связать все это вместе, чтобы у моего контроллера мог быть метод, подобный

@RequestMapping(method = RequestMethod.POST)
public String doUpload(
    @Valid final List<MyObject> objList, 
    final BindingResult result) { ...}

Я знаю, что среда Spring MVC поддерживает конвертеры и валидаторы, но я не понимаю, как заставить их работать вместе.

Ответы [ 2 ]

4 голосов
/ 31 августа 2011

Сначала я обернул MultipartFile в объект поддержки формы:

public class UploadBackingForm {
    private MultipartFile multipartFile;
    /* ... getter/setter */
}

Который я затем привязал к своей форме:

<form method="post" enctype="multipart/form-data">
<@spring.bind "backingform.*"/>
<tr>
    <td><@spring.formInput 'backingform.multipartFile' '' 'file' /></td>
    <td> <button type="submit">Upload</button> </td>
</tr>
</form>

В контроллере я назначил валидатор:

@InitBinder
public void initBinder(final DataBinder binder) {
    binder.setValidator(new UploadValidator());
}

И это валидатор:

public class UploadValidator implements Validator {
    private final Converter<String[], MyObject> converter 
        = new MyObjectConverter();

    @Override
    public boolean supports(final Class<?> clazz) {
        return UploadBackingForm.class.equals(clazz);
    }

    @Override
    public void validate(final Object target, final Errors errors) {
        final UploadBackingForm backingForm = (UploadBackingForm) target;
        final MultipartFile multipartFile = backingForm.getMultipartFile();
        final List<String[]> uploadData = /* parse file */
        for (final String[] uploadDataRow : uploadData){
            try {
                converter.convert(uploadDataRow);
            } catch (IllegalArgumentException e) {
                errors.rejectValue("multipartFile", "line.invalid", ...);
            }
        }
    }
}

В валидаторе используется конвертер для преобразования отдельных позиций в MyObj.

Метод doPost теперь выглядит следующим образом:

@RequestMapping(method = RequestMethod.POST)
public String doUpload(
    @Valid @ModelAttribute("backingform") UploadBackingForm backingForm, 
    final BindingResult result, 
    final HttpSession session) throws IOException {

    final UploadConverter converter = new UploadConverter();
    final List<MyObj> imports = 
        converter.convert(backingForm.getMultipartFile().getInputStream());
 }

UploadConverter во многом совпадает с UploadValidator:

public class UploadConverter implements Converter<InputStream, List<MyObject>> {
    private final Converter<String[], MyObject> converter = new MyObjectConverter();

    @Override
    public List<MyObject> convert(final InputStream source) {
        final List<String[]> detailLines = /* ... getDetailLines */
        final List<MyObject> importList = 
            new ArrayList<MyObject>(detailLines.size());

        for (final String[] row : detailLines) {
            importList.add(converter.convert(row));
        }
        return importList;
    }
}

Единственная проблема состоит в том, что процессы проверки и преобразования во многом совпадают.К счастью, загружаемые файлы не будут очень большими, поэтому дублирование усилий не является большой проблемой.

0 голосов
/ 23 августа 2011

Вам нужен конвертер, который может конвертировать MultipartFile в MyObjects, но не конвертировать String[] в MyObjectsContainer. - MyObjectsContainer - не более чем оболочка списка MyObjects.

Но я действительно не знаю, будет ли работать этот конвертер, потому что MultipartFile - это какой-то особенный параметр.

Для валидации я настоятельно рекомендую аннотации JSR 303 Bean Validation в MyObjectsContainer

Тогда вы можете написать:

@RequestMapping(method = RequestMethod.POST)
public String doUpload(@RequestParam("file") final MyObjectsContainer container) {}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...