タイトル: CSVファイルアップロード方法(Ajax)
SEOタイトル: Spring + Ajax CSV アップロード|FormData / MultipartFile 実装
| この記事の要点 |
|
概要
フロント側 JavaScript から Java (Spring Framework) で CSV ファイルを受け取る実装例です。Ajax + FormData を使うことで、画面遷移なしでアップロード進捗・完了を扱えます。
HTML
JavaScript (jQuery)
$("#upload").click(function() {
var formData = new FormData($('#fileuploadform').get(0));
$.ajax({
method: 'POST',
url: '/csvUpload',
data: formData,
processData: false, // ★ FormData を文字列化させない
contentType: false, // ★ multipart/form-data の boundary を維持
headers: { 'X-CSRF-TOKEN': csrfToken } // CSRF 対策(Spring Security)
}).done(function(res) {
$('#result').text('成功: ' + res.message);
}).fail(function(xhr) {
$('#result').text('失敗: ' + (xhr.responseJSON?.error ?? xhr.status));
});
});
JavaScript(fetch 版・モダン)
document.getElementById('upload').addEventListener('click', async () => {
const file = document.getElementById('file').files[0];
if (!file) { alert('ファイル未選択'); return; }
const formData = new FormData();
formData.append('file', file);
try {
const res = await fetch('/csvUpload', {
method: 'POST',
body: formData,
headers: { 'X-CSRF-TOKEN': csrfToken }
// ★ Content-Type は付けない(fetch が自動で multipart/form-data + boundary)
});
if (!res.ok) throw new Error('HTTP ' + res.status);
const data = await res.json();
document.getElementById('result').textContent = '成功: ' + data.message;
} catch (e) {
document.getElementById('result').textContent = '失敗: ' + e.message;
}
});
Spring Controller
@RestController
public class CsvUploadController {
@PostMapping("/csvUpload")
public ResponseEntity
Spring 側のサイズ上限設定
デフォルトでは Spring Boot のアップロード上限は 1MB / 1MB(ファイル / リクエスト全体)です。大きなファイルを扱う場合は application.properties で拡張:
spring.servlet.multipart.max-file-size=10MB
spring.servlet.multipart.max-request-size=10MB
# 一時ディレクトリ(メモリ閾値超え時の書き出し先)
spring.servlet.multipart.location=/tmp
# メモリに保持する閾値(これを超えるとディスクへ)
spring.servlet.multipart.file-size-threshold=2KB
processData / contentType を false にする理由
| オプション | デフォルト | FormData 送信時に必要な値 |
|---|---|---|
processData | true(クエリ文字列化) | false(FormData をそのまま送る) |
contentType | application/x-www-form-urlencoded | false(ブラウザに任せて boundary 付き multipart に) |
fetch API ではこの指定は不要 — body に FormData を入れれば自動的に正しいヘッダになります(むしろ手動で Content-Type を付けると boundary が消えて壊れます)。
よくあるエラー
- 413 Payload Too Large: ファイルサイズが上限超え →
spring.servlet.multipart.max-file-sizeを増やす。または Nginx のclient_max_body_size - 403 Forbidden: CSRF トークン未送信(Spring Security)→
X-CSRF-TOKENヘッダ追加 - 415 Unsupported Media Type:
contentType: falseを忘れている → jQuery が間違った Content-Type を送っている - 500 + FileNotFoundException tmp/...: Multipart の一時ディレクトリが書き込み不可 →
spring.servlet.multipart.locationを見直し - 日本語ファイル名が文字化け:
file.getOriginalFilename()がブラウザ依存。URL エンコードする運用が安全
セキュリティ注意
- 拡張子だけでなくマジックバイトもチェック(Polyglot ファイル対策)
- アップロードしたファイルは Web ルート外に保存(直接 URL でアクセスされないように)
- ファイル名は サーバ側で UUID 等にリネーム(パストラバーサル防止)
- 1 リクエストあたりの行数上限を設ける(メモリ枯渇 / DoS 対策)