SQLとバインド変数の値を両方ともログに出力するにはapplication.ymlに下記を追記する。
logging:
level:
org.hibernate:
SQL: debug
type:
EnumType: trace
descriptor.sql.BasicBinder: trace
Technical Notes
SQLとバインド変数の値を両方ともログに出力するにはapplication.ymlに下記を追記する。
logging:
level:
org.hibernate:
SQL: debug
type:
EnumType: trace
descriptor.sql.BasicBinder: trace
fetch = FetchType.LAZYとしている場合、データベースを参照するときにSessionが切れていてLazyInitializationExceptionが発生することがある。
Caused by:
org.hibernate.LazyInitializationException:
failed to lazily initialize a collection of role:
crawlerapi.entity.Novel.novelChapters, could not initialize proxy - no Session
そのような場合は、application.ymlに以下を追加する。
jpa:
properties:
hibernate:
enable_lazy_load_no_trans: true
Opencsvについては、リンク先を参照。
ここではアノテーションを使わずに、各列を各変数にマッピングする方法を紹介する。
・CSVを読み込む
try (InputStreamReader is = new InputStreamReader(multipartFile.getInputStream(), Constants.ENCODING);
CSVReader reader = new CSVReaderBuilder(is).withSkipLines(1).build()) {
ColumnPositionMappingStrategy strat = new ColumnPositionMappingStrategy<>();
strat.setType(User.class);
strat.setColumnMapping("username", "password", "firstName", "lastName", "email");
CsvToBean<User> csv = new CsvToBean<>();
csv.setCsvReader(reader);
csv.setMappingStrategy(strat);
return csv.parse();
} catch (IOException e) {
// 適宜例外処理
} catch (IllegalStateException e) {
// 適宜例外処理
}
InputStreamReaderについては、どこからファイルを取得するかによって変更する。
CSVReader reader = new CSVReaderBuilder(is).withSkipLines(1).build();
で、CSVReaderを作成する。1行目はヘッダー行のため、withSkipLines(1)で2行目から読み込むようにしている。
ColumnPositionMappingStrategyで、CSVファイルの各列をどのクラスのどの変数にマッピングするか指定出来る。
strat.setType(User.class)で、Userクラスにマッピングすることを指定している。
strat.setColumnMapping(“username”, “password”, “firstName”, “lastName”, “email”)で、1列目をUserクラスのusername変数へ、2列目をpasswordへのように指定していることになる。
CsvToBeanで、List<String[]>をList<User>に変換する。
・CSVに書き出す
try (OutputStreamWriter os = new OutputStreamWriter(response.getOutputStream(), Constants.ENCODING);
CSVWriter writer = new CSVWriter(os)) {
// ヘッダー行を追加
writer.writeNext(new String[] {"ユーザ名","パスワード","名字","名前","eメール"});
ColumnPositionMappingStrategy<User> strat = new ColumnPositionMappingStrategy<>();
strat.setType(User.class);
strat.setColumnMapping("username", "password", "firstName", "lastName", "email");
StatefulBeanToCsv<User> beanToCsv = new StatefulBeanToCsvBuilder<User>(writer)
.withMappingStrategy(strat)
.build();
beanToCsv.write(userList);
} catch (IOException e) {
// 適宜例外処理
} catch (CsvException e) {
// 適宜例外処理
}
CSVファイルへの書き出しは上記の通り。
testCompile 'org.springframework.boot:spring-boot-starter-test'
を以下の通り変更する。
testCompile('org.springframework.boot:spring-boot-starter-test') {
exclude module: 'junit'
}
さらに、以下の行を追加する。
testImplementation 'org.junit.jupiter:junit-jupiter-api'
testRuntimeOnly 'org.junit.jupiter:junit-jupiter-engine'
なお、Spring Boot 2.2からはJUnit 5.5が標準になったので、上記の対応は不要となる。
ビルド環境をopenjdk11に変更したら、coveralls-maven-pluginが落ちてしまって、Coverallsのカバレッジが更新されなくなっていた。
coveralls-maven-pluginに以下の依存関係を追加することで回避することが出来る。
<dependencies>
<dependency>
<groupId>javax.xml.bind</groupId>
<artifactId>jaxb-api</artifactId>
<version>2.3.1</version>
</dependency>
</dependencies>
本家がJava11に対応してくれるのが一番良いのだが、忙しいみたいでいつ対応されるかわからない。
Thymeleafはテンプレートの一種。
Springとの相性が良く簡単に使用出来る。
・top.html
<html xmlns="http://www.w3.org/1999/xhtml"
xmlns:th="http://www.thymeleaf.org"
xmlns:layout="http://www.ultraq.net.nz/thymeleaf/layout">
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8" />
</head>
<body>
<p th:text="Hello, Thymeleaf!">Hello, hide6644!</p>
</body>
</html>
View Resolverを設定する。
<!-- View Resolver for Thymeleaf -->
<bean id="templateResolver" class="org.thymeleaf.spring4.templateresolver.SpringResourceTemplateResolver">
<property name="prefix" value="/WEB-INF/pages/" />
<property name="suffix" value=".html" />
<property name="templateMode" value="HTML" />
<property name="cacheable" value="false" />
</bean>
<bean id="templateEngine" class="org.thymeleaf.spring4.SpringTemplateEngine">
<property name="templateResolver" ref="templateResolver" />
<property name="enableSpringELCompiler" value="true" />
</bean>
<bean class="org.thymeleaf.spring4.view.ThymeleafViewResolver">
<property name="templateEngine" ref="templateEngine" />
<property name="characterEncoding" value="UTF-8" />
<property name="order" value="4" />
</bean>
Veiwを呼び出す。
@RequestMapping(value = "top", method = RequestMethod.GET)
public String topRequest() {
return "top";
}
小説を読もう!の更新を自動でチェックしたかったのだが、自分の好みに合ったサービスがなかった。
じゃあ、自分で作っちゃえってことで以下の通り。
下記のインスタンスを生成する部分をMockにしたい。
NovelSource novelSource = new NovelSource(url);
そこで、PowerMockitoを使用してみる。
<!-- https://mvnrepository.com/artifact/org.powermock/powermock-api-mockito -->
<dependency>
<groupId>org.powermock</groupId>
<artifactId>powermock-api-mockito</artifactId>
<version>1.6.6</version>
</dependency>
使用方法はこんな感じ。
@RunWith(PowerMockRunner.class)
@PrepareForTest({ NovelManagerImpl.class })
@PowerMockIgnore("javax.management.*")
public class NovelManagerImplTest extends BaseManagerMockTestCase {
@Mock
private NovelSource novelSource;
@Mock
private Logger log;
@Mock
private NovelDao novelDao;
@Mock
private NovelInfoManager novelInfoManager;
@Mock
private NovelChapterManager novelChapterManager;
@InjectMocks
private NovelManagerImpl novelManager = new NovelManagerImpl();
@Test
public void testAdd() throws Exception {
String fileName = this.getClass().getClassLoader().getResource("novel/20160924/test.html").getPath();
File file = new File(fileName);
String url = "http://www.foo.bar/20160924/";
NovelSource novelSource = new NovelSource(new Source(file));
{
// 初期化
MockitoAnnotations.initMocks(this);
// NovelSourceをnewするとき、thenReturn(~)のインスタンスを返却する
PowerMockito.whenNew(NovelSource.class).withArguments(url).thenReturn(novelSource);
}
}
コンストラクタに引数を渡したいときは、withArguments()を使用する。
しかし、カバレッジが測定出来ないので注意。