iTextPDF Documentファイルをmemoryに作成した後、byte []に変換する必要があります。 PDFを正しく作成することに問題がないことをすでにテストしました。問題は、DBに格納するためにバイト配列に変換する方法です。
これが私のコードです:
Document generatedDocument = reportService.generateRequestForm(scdUser, jsonObject, 0, null);
reportService.generateRequestForm(scdUser, jsonObject, 0, null);
ByteArrayOutputStream baos = new ByteArrayOutputStream();
PdfWriter pdfWriter = PdfWriter.getInstance(generatedDocument, baos);
generatedDocument.open();
document.setDocument(baos.toByteArray()); // stores as blob
データベースblob列でnullの値を取得しました。
これが私のドキュメントドメインオブジェクトです:
ドキュメントドメインオブジェクト
@Entity
@Table(name = "document")
public class Document implements Java.io.Serializable {
@Id
@GeneratedValue(strategy = GenerationType.AUTO)
@Column(name = "document_id", nullable = false)
private int documentId;
@Column(name = "document_name", nullable = false, length = 65535)
private String documentName;
@Column(name = "document_type", nullable = false)
private int documentType;
@Temporal(TemporalType.TIMESTAMP)
@Column(name = "upload_date", nullable = false, length = 19)
private Date uploadDate = new Date();
@Column(name = "document", nullable = false)
private byte[] document; // BLOB COLUMN
@Column(name = "document_size", nullable = false)
private long documentSize;
@Column(name = "title", nullable = true, insertable = true, updatable = true, length = 65535, precision = 0)
private String title;
@Column(name = "tag", nullable = true, insertable = true, updatable = true, length = 65535, precision = 0)
private String tag;
@Column(name = "description", nullable = true, insertable = true, updatable = true, length = 65535, precision = 0)
private String description;
@Column(name = "shared", nullable = false, insertable = true, updatable = true, length = 1, precision = 0)
private boolean shared = false;
@Column(name = "status", nullable = false)
private int status = DocumentStatus.READY.getStatus();
public int getDocumentId() {
return this.documentId;
}
public void setDocumentId(int documentId) {
this.documentId = documentId;
}
public String getDocumentName() {
return this.documentName;
}
public void setDocumentName(String documentName) {
this.documentName = documentName;
}
public int getDocumentType() {
return this.documentType;
}
public void setDocumentType(int documentType) {
this.documentType = documentType;
}
public Date getUploadDate() {
return this.uploadDate;
}
public void setUploadDate(Date uploadDate) {
this.uploadDate = uploadDate;
}
public byte[] getDocument() {
return this.document;
}
public void setDocument(byte[] document) {
this.document = document;
}
public long getDocumentSize() {
return this.documentSize;
}
public void setDocumentSize(long documentSize) {
this.documentSize = documentSize;
}
public String getTag() {
return tag;
}
public void setTag(String tag) {
this.tag = tag;
}
public String getTitle() {
return title;
}
public void setTitle(String title) {
this.title = title;
}
public String getDescription() {
return description;
}
public void setDescription(String description) {
this.description = description;
}
public boolean getShared() {
return shared;
}
public void setShared(boolean shared) {
this.shared = shared;
}
public int getStatus() {
return status;
}
public void setStatus(int status) {
this.status = status;
}
}
同様の問題が発生しました...ドキュメントを作成し、それを作成したクラス内でファイルに保存できましたが、うまく機能しました。ただし、ストリームとして返そうとすると、null値が返されます。
問題は、ドキュメントが閉じられると(document.close())、ストリームも閉じることでした。
回避策は、ドキュメントを作成したときにByteArrayOutputStreamを作成し、それにPdfWriterを出力することでした。その後、PDFバイト...私の場合は、ユーザーに送信するためにそれらをStreamedContentに変換しました。
バイトを保持する変数を作成します。
private ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
PdfWriterに、ドキュメントの作成時にデータをbyte []に出力させます。
Document document = new Document(PageSize.LETTER, 0.75F, 0.75F, 0.75F, 0.75F);
PdfWriter.getInstance(document, byteArrayOutputStream); // Do this BEFORE document.open()
document.open();
createPDF(document); // Whatever function that you use to create your PDF
document.close();
PDFの生成が完了したら、バイトを取得して、必要に応じて処理します。
byte[] pdfBytes = byteArrayOutputStream.toByteArray();
ReportServiceクラスがどのように見えるかはわかりませんが、これを配置するのに適した場所である可能性があります。
お役に立てれば。
コメントから、PDFは実行時に生成される方法ではなく、DBに格納される方法とは関係がないようです。そのコードも提供する必要があります。
これが私が実際に持っているものです(私のテスト):
ホルダー:
import Java.io.Serializable;
import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
@Entity
public class PDFHolder implements Serializable {
@Id
@GeneratedValue(strategy=GenerationType.AUTO)
private long id;
@Column(columnDefinition = "LONGBLOB")
private byte[] documentPDF;
public byte[] getDocumentPDF() {
return documentPDF;
}
public void setDocumentPDF(byte[] documentPDF) {
this.documentPDF = documentPDF;
}
public long getId() {
return id;
}
public void setId(long id) {
this.id = id;
}
}
保存を行うリポジトリ:
@Repository
public interface PDFHolderRepository extends JpaRepository<PDFHolder, Long> {}
そして実際の挿入:
private void generateDatabaseRows() {
try{
String urlPDF = "http://cetatenie.just.ro/LinkClick.aspx?fileticket=K13DZkoIE2o%3d&tabid=57&mid=593";
URL url = new URL(urlPDF);
ByteBuffer byteBufferResponse = this.getAsByteArray(url.openConnection());
byte [] responseArray = byteBufferResponse.array();
System.out.println("Size of the PDF : " + responseArray.length);
// Save it to DB
PDFHolder pdfHolder = new PDFHolder();
pdfHolder.setDocumentPDF(responseArray);
pdfHolderRepository.save(pdfHolder);
} catch(Exception e){
e.printStackTrace();
}
}
private ByteBuffer getAsByteArray(final URLConnection urlConnection) throws IOException {
final ByteArrayOutputStream tmpOut = new ByteArrayOutputStream();
final InputStream inputStream = urlConnection.getInputStream();
final byte[] buf = new byte[1024];
int len;
while (true) {
len = inputStream.read(buf);
if (len == -1) {
break;
}
tmpOut.write(buf, 0, len);
}
tmpOut.close();
return ByteBuffer.wrap(tmpOut.toByteArray(), 0, tmpOut.size());
}
もちろん、このクラスには@Autowiredリポジトリがあります。
@Autowired
PDFHolderRepository pdfHolderRepository;