user = $user; $this->filename = $filename; } public function execute() { // Guess charset $charset = $this->guessCharset($this->filename); // Open CSV $csv = Reader::createFromPath($this->filename, 'r'); $csv->setHeaderOffset(0); if ($charset === 'SJIS-win') { $csv->addStreamFilter('convert.mbstring.encoding.SJIS-win:UTF-8'); } // Import DB::transaction(function () use ($csv) { $errors = []; if (!in_array('日時', $csv->getHeader(), true)) { $errors[] = '日時列は必須です。'; throw new CsvImportException($errors); } foreach ($csv->getRecords() as $offset => $record) { $line = $offset + 1; $ejaculation = new Ejaculation(['user_id' => $this->user->id]); $validator = Validator::make($record, [ '日時' => ['required', new CsvDateTime()], 'ノート' => 'nullable|string|max:500', 'オカズリンク' => 'nullable|url|max:2000', ]); if ($validator->fails()) { foreach ($validator->errors()->all() as $message) { $errors[] = "{$line} 行 : {$message}"; } continue; } $ejaculation->ejaculated_date = Carbon::createFromFormat('!Y/m/d H:i+', $record['日時']); $ejaculation->note = str_replace(["\r\n", "\r"], "\n", $record['ノート'] ?? ''); $ejaculation->link = $record['オカズリンク'] ?? ''; $tagIds = []; for ($i = 1; $i <= 32; $i++) { $column = 'タグ' . $i; if (empty($record[$column])) { break; } else { $tag = trim($record[$column]); if (empty($tag)) { break; } if (mb_strlen($tag) > 255) { $errors[] = "{$line} 行 : {$column}列は255文字以内にしてください。"; continue 2; } $tag = Tag::firstOrCreate(['name' => $tag]); $tagIds[] = $tag->id; } } $ejaculation->tags()->sync($tagIds); $ejaculation->save(); } if (!empty($errors)) { throw new CsvImportException($errors); } }); } /** * 指定されたファイルを読み込み、文字コードの判定を行います。 * @param string $filename CSVファイル名 * @param int $samplingLength ファイルの先頭から何バイトを判定に使用するかを指定 * @return string 検出した文字コード (UTF-8, SJIS-win, ...) * @throws CsvImportException ファイルの読み込みに失敗した、文字コードを判定できなかった、または非対応文字コードを検出した場合にスロー */ private function guessCharset(string $filename, int $samplingLength = 1024): string { $fp = fopen($filename, 'rb'); if (!$fp) { throw new CsvImportException(['CSVファイルの読み込み中にエラーが発生しました。']); } try { $head = fread($fp, $samplingLength); if ($head === false) { throw new CsvImportException(['CSVファイルの読み込み中にエラーが発生しました。']); } for ($addition = 0; $addition < 4; $addition++) { $charset = mb_detect_encoding($head, ['ASCII', 'UTF-8', 'SJIS-win'], true); if ($charset) { if (array_search($charset, ['UTF-8', 'SJIS-win'], true) === false) { throw new CsvImportException(['文字コード判定に失敗しました。UTF-8 (BOM無し) または Shift_JIS をお使いください。']); } else { return $charset; } } // 1バイト追加で読み込んだら、文字境界に到達して上手く判定できるかもしれない if (feof($fp)) { break; } $next = fread($fp, 1); if ($next === false) { throw new CsvImportException(['CSVファイルの読み込み中にエラーが発生しました。']); } $head .= $next; } throw new CsvImportException(['文字コード判定に失敗しました。UTF-8 (BOM無し) または Shift_JIS をお使いください。']); } finally { fclose($fp); } } }