Merge remote-tracking branch 'baeldung/master' into bael-2538
This commit is contained in:
commit
c030b8830b
|
@ -66,3 +66,10 @@ jmeter/src/main/resources/*-JMeter.csv
|
||||||
**/nb-configuration.xml
|
**/nb-configuration.xml
|
||||||
core-scala/.cache-main
|
core-scala/.cache-main
|
||||||
core-scala/.cache-tests
|
core-scala/.cache-tests
|
||||||
|
|
||||||
|
|
||||||
|
persistence-modules/hibernate5/transaction.log
|
||||||
|
apache-avro/src/main/java/com/baeldung/avro/model/
|
||||||
|
jta/transaction-logs/
|
||||||
|
software-security/sql-injection-samples/derby.log
|
||||||
|
spring-soap/src/main/java/com/baeldung/springsoap/gen/
|
|
@ -0,0 +1,85 @@
|
||||||
|
package com.baeldung.map
|
||||||
|
|
||||||
|
import static org.junit.Assert.*
|
||||||
|
import org.junit.Test
|
||||||
|
|
||||||
|
class MapUnitTest {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void whenUsingEach_thenMapIsIterated() {
|
||||||
|
def map = [
|
||||||
|
'FF0000' : 'Red',
|
||||||
|
'00FF00' : 'Lime',
|
||||||
|
'0000FF' : 'Blue',
|
||||||
|
'FFFF00' : 'Yellow'
|
||||||
|
]
|
||||||
|
|
||||||
|
map.each { println "Hex Code: $it.key = Color Name: $it.value" }
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void whenUsingEachWithEntry_thenMapIsIterated() {
|
||||||
|
def map = [
|
||||||
|
'E6E6FA' : 'Lavender',
|
||||||
|
'D8BFD8' : 'Thistle',
|
||||||
|
'DDA0DD' : 'Plum',
|
||||||
|
]
|
||||||
|
|
||||||
|
map.each { entry -> println "Hex Code: $entry.key = Color Name: $entry.value" }
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void whenUsingEachWithKeyAndValue_thenMapIsIterated() {
|
||||||
|
def map = [
|
||||||
|
'000000' : 'Black',
|
||||||
|
'FFFFFF' : 'White',
|
||||||
|
'808080' : 'Gray'
|
||||||
|
]
|
||||||
|
|
||||||
|
map.each { key, val ->
|
||||||
|
println "Hex Code: $key = Color Name $val"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void whenUsingEachWithIndexAndEntry_thenMapIsIterated() {
|
||||||
|
def map = [
|
||||||
|
'800080' : 'Purple',
|
||||||
|
'4B0082' : 'Indigo',
|
||||||
|
'6A5ACD' : 'Slate Blue'
|
||||||
|
]
|
||||||
|
|
||||||
|
map.eachWithIndex { entry, index ->
|
||||||
|
def indent = ((index == 0 || index % 2 == 0) ? " " : "")
|
||||||
|
println "$indent Hex Code: $entry.key = Color Name: $entry.value"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void whenUsingEachWithIndexAndKeyAndValue_thenMapIsIterated() {
|
||||||
|
def map = [
|
||||||
|
'FFA07A' : 'Light Salmon',
|
||||||
|
'FF7F50' : 'Coral',
|
||||||
|
'FF6347' : 'Tomato',
|
||||||
|
'FF4500' : 'Orange Red'
|
||||||
|
]
|
||||||
|
|
||||||
|
map.eachWithIndex { key, val, index ->
|
||||||
|
def indent = ((index == 0 || index % 2 == 0) ? " " : "")
|
||||||
|
println "$indent Hex Code: $key = Color Name: $val"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void whenUsingForLoop_thenMapIsIterated() {
|
||||||
|
def map = [
|
||||||
|
'2E8B57' : 'Seagreen',
|
||||||
|
'228B22' : 'Forest Green',
|
||||||
|
'008000' : 'Green'
|
||||||
|
]
|
||||||
|
|
||||||
|
for (entry in map) {
|
||||||
|
println "Hex Code: $entry.key = Color Name: $entry.value"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,16 @@
|
||||||
|
package com.baeldung.multireleaseapp;
|
||||||
|
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
|
public class App {
|
||||||
|
|
||||||
|
private static final Logger logger = LoggerFactory.getLogger(App.class);
|
||||||
|
|
||||||
|
public static void main(String[] args) throws Exception {
|
||||||
|
String dateToCheck = args[0];
|
||||||
|
boolean isLeapYear = DateHelper.checkIfLeapYear(dateToCheck);
|
||||||
|
logger.info("Date given " + dateToCheck + " is leap year: " + isLeapYear);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,22 @@
|
||||||
|
package com.baeldung.multireleaseapp;
|
||||||
|
|
||||||
|
import java.text.SimpleDateFormat;
|
||||||
|
import java.util.Calendar;
|
||||||
|
import java.util.GregorianCalendar;
|
||||||
|
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
|
public class DateHelper {
|
||||||
|
|
||||||
|
private static final Logger logger = LoggerFactory.getLogger(DateHelper.class);
|
||||||
|
|
||||||
|
public static boolean checkIfLeapYear(String dateStr) throws Exception {
|
||||||
|
logger.info("Checking for leap year using Java 1 calendar API");
|
||||||
|
Calendar cal = Calendar.getInstance();
|
||||||
|
cal.setTime(new SimpleDateFormat("yyyy-MM-dd").parse(dateStr));
|
||||||
|
int year = cal.get(Calendar.YEAR);
|
||||||
|
return (new GregorianCalendar()).isLeapYear(year);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,18 @@
|
||||||
|
package com.baeldung.multireleaseapp;
|
||||||
|
|
||||||
|
import java.time.LocalDate;
|
||||||
|
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
|
public class DateHelper {
|
||||||
|
|
||||||
|
private static final Logger logger = LoggerFactory.getLogger(DateHelper.class);
|
||||||
|
|
||||||
|
public static boolean checkIfLeapYear(String dateStr) throws Exception {
|
||||||
|
logger.info("Checking for leap year using Java 9 Date Api");
|
||||||
|
return LocalDate.parse(dateStr)
|
||||||
|
.isLeapYear();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -5,6 +5,7 @@ import static org.junit.Assert.assertTrue;
|
||||||
|
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
import java.io.FileNotFoundException;
|
import java.io.FileNotFoundException;
|
||||||
|
import java.io.IOException;
|
||||||
import java.io.PrintWriter;
|
import java.io.PrintWriter;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
@ -17,7 +18,6 @@ import org.slf4j.LoggerFactory;
|
||||||
public class WriteCsvFileExampleUnitTest {
|
public class WriteCsvFileExampleUnitTest {
|
||||||
private static final Logger LOG = LoggerFactory.getLogger(WriteCsvFileExampleUnitTest.class);
|
private static final Logger LOG = LoggerFactory.getLogger(WriteCsvFileExampleUnitTest.class);
|
||||||
|
|
||||||
private static final String CSV_FILE_NAME = "src/test/resources/exampleOutput.csv";
|
|
||||||
private WriteCsvFileExample csvExample;
|
private WriteCsvFileExample csvExample;
|
||||||
|
|
||||||
@Before
|
@Before
|
||||||
|
@ -65,12 +65,12 @@ public class WriteCsvFileExampleUnitTest {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void givenDataArray_whenConvertToCSV_thenOutputCreated() {
|
public void givenDataArray_whenConvertToCSV_thenOutputCreated() throws IOException {
|
||||||
List<String[]> dataLines = new ArrayList<String[]>();
|
List<String[]> dataLines = new ArrayList<String[]>();
|
||||||
dataLines.add(new String[] { "John", "Doe", "38", "Comment Data\nAnother line of comment data" });
|
dataLines.add(new String[] { "John", "Doe", "38", "Comment Data\nAnother line of comment data" });
|
||||||
dataLines.add(new String[] { "Jane", "Doe, Jr.", "19", "She said \"I'm being quoted\"" });
|
dataLines.add(new String[] { "Jane", "Doe, Jr.", "19", "She said \"I'm being quoted\"" });
|
||||||
|
|
||||||
File csvOutputFile = new File(CSV_FILE_NAME);
|
File csvOutputFile = File.createTempFile("exampleOutput", ".csv");
|
||||||
try (PrintWriter pw = new PrintWriter(csvOutputFile)) {
|
try (PrintWriter pw = new PrintWriter(csvOutputFile)) {
|
||||||
dataLines.stream()
|
dataLines.stream()
|
||||||
.map(csvExample::convertToCSV)
|
.map(csvExample::convertToCSV)
|
||||||
|
@ -80,5 +80,6 @@ public class WriteCsvFileExampleUnitTest {
|
||||||
}
|
}
|
||||||
|
|
||||||
assertTrue(csvOutputFile.exists());
|
assertTrue(csvOutputFile.exists());
|
||||||
|
csvOutputFile.deleteOnExit();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,2 +0,0 @@
|
||||||
John,Doe,38,Comment Data Another line of comment data
|
|
||||||
Jane,"Doe, Jr.",19,"She said ""I'm being quoted"""
|
|
|
|
@ -0,0 +1,40 @@
|
||||||
|
package com.baeldung.jackson.dtos;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.dataformat.xml.annotation.JacksonXmlProperty;
|
||||||
|
|
||||||
|
public class Address {
|
||||||
|
|
||||||
|
@JacksonXmlProperty(localName = "street_number")
|
||||||
|
String streetNumber;
|
||||||
|
|
||||||
|
@JacksonXmlProperty(localName = "street_name")
|
||||||
|
String streetName;
|
||||||
|
|
||||||
|
@JacksonXmlProperty(localName = "city")
|
||||||
|
String city;
|
||||||
|
|
||||||
|
public String getStreetNumber() {
|
||||||
|
return streetNumber;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setStreetNumber(String streetNumber) {
|
||||||
|
this.streetNumber = streetNumber;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getStreetName() {
|
||||||
|
return streetName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setStreetName(String streetName) {
|
||||||
|
this.streetName = streetName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getCity() {
|
||||||
|
return city;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setCity(String city) {
|
||||||
|
this.city = city;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,51 @@
|
||||||
|
package com.baeldung.jackson.dtos;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.dataformat.xml.annotation.JacksonXmlElementWrapper;
|
||||||
|
import com.fasterxml.jackson.dataformat.xml.annotation.JacksonXmlRootElement;
|
||||||
|
|
||||||
|
@JacksonXmlRootElement(localName = "person")
|
||||||
|
public final class Person {
|
||||||
|
private String firstName;
|
||||||
|
private String lastName;
|
||||||
|
@JacksonXmlElementWrapper(useWrapping = false)
|
||||||
|
private List<String> phoneNumbers = new ArrayList<>();
|
||||||
|
@JacksonXmlElementWrapper(localName = "addresses")
|
||||||
|
private List<Address> address = new ArrayList<>();
|
||||||
|
|
||||||
|
public List<Address> getAddress() {
|
||||||
|
return address;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setAddress(List<Address> address) {
|
||||||
|
this.address = address;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getFirstName() {
|
||||||
|
return firstName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setFirstName(String firstName) {
|
||||||
|
this.firstName = firstName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getLastName() {
|
||||||
|
return lastName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setLastName(String lastName) {
|
||||||
|
this.lastName = lastName;
|
||||||
|
}
|
||||||
|
public List<String> getPhoneNumbers() {
|
||||||
|
return phoneNumbers;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setPhoneNumbers(List<String> phoneNumbers) {
|
||||||
|
this.phoneNumbers = phoneNumbers;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
}
|
|
@ -9,12 +9,16 @@ import java.io.FileInputStream;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.InputStream;
|
import java.io.InputStream;
|
||||||
import java.io.InputStreamReader;
|
import java.io.InputStreamReader;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
|
|
||||||
|
import com.baeldung.jackson.dtos.Address;
|
||||||
|
import com.baeldung.jackson.dtos.Person;
|
||||||
|
import com.fasterxml.jackson.annotation.JsonProperty;
|
||||||
import com.fasterxml.jackson.core.JsonProcessingException;
|
import com.fasterxml.jackson.core.JsonProcessingException;
|
||||||
import com.fasterxml.jackson.dataformat.xml.XmlMapper;
|
import com.fasterxml.jackson.dataformat.xml.XmlMapper;
|
||||||
import com.fasterxml.jackson.annotation.JsonProperty;
|
|
||||||
|
|
||||||
public class XMLSerializeDeserializeUnitTest {
|
public class XMLSerializeDeserializeUnitTest {
|
||||||
|
|
||||||
|
@ -49,7 +53,7 @@ public class XMLSerializeDeserializeUnitTest {
|
||||||
assertTrue(value.getX() == 1 && value.getY() == 2);
|
assertTrue(value.getX() == 1 && value.getY() == 2);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void whenJavaGotFromXmlStrWithCapitalElem_thenCorrect() throws IOException {
|
public void whenJavaGotFromXmlStrWithCapitalElem_thenCorrect() throws IOException {
|
||||||
XmlMapper xmlMapper = new XmlMapper();
|
XmlMapper xmlMapper = new XmlMapper();
|
||||||
SimpleBeanForCapitalizedFields value = xmlMapper.
|
SimpleBeanForCapitalizedFields value = xmlMapper.
|
||||||
|
@ -67,6 +71,54 @@ public class XMLSerializeDeserializeUnitTest {
|
||||||
assertNotNull(file);
|
assertNotNull(file);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void whenJavaDeserializedFromXmlFile_thenCorrect() throws IOException {
|
||||||
|
XmlMapper xmlMapper = new XmlMapper();
|
||||||
|
Person value = xmlMapper.readValue(new File("src/test/resources/person.xml"), Person.class);
|
||||||
|
|
||||||
|
assertTrue(value.getAddress()
|
||||||
|
.get(0)
|
||||||
|
.getCity()
|
||||||
|
.equalsIgnoreCase("city1")
|
||||||
|
&& value.getAddress()
|
||||||
|
.get(1)
|
||||||
|
.getCity()
|
||||||
|
.equalsIgnoreCase("city2"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void whenJavaSerializedToXmlFile_thenSuccess() throws IOException {
|
||||||
|
XmlMapper xmlMapper = new XmlMapper();
|
||||||
|
|
||||||
|
Person person = new Person();
|
||||||
|
|
||||||
|
person.setFirstName("Rohan");
|
||||||
|
person.setLastName("Daye");
|
||||||
|
|
||||||
|
List<String> ph = new ArrayList<>();
|
||||||
|
ph.add("9911778981");
|
||||||
|
ph.add("9991111111");
|
||||||
|
person.setPhoneNumbers(ph);
|
||||||
|
|
||||||
|
List<Address> addresses = new ArrayList<>();
|
||||||
|
|
||||||
|
Address address1 = new Address();
|
||||||
|
address1.setStreetNumber("1");
|
||||||
|
address1.setStreetName("streetname1");
|
||||||
|
address1.setCity("city1");
|
||||||
|
|
||||||
|
Address address2 = new Address();
|
||||||
|
address2.setStreetNumber("2");
|
||||||
|
address2.setStreetName("streetname2");
|
||||||
|
address2.setCity("city2");
|
||||||
|
|
||||||
|
addresses.add(address1);
|
||||||
|
addresses.add(address2);
|
||||||
|
person.setAddress(addresses);
|
||||||
|
|
||||||
|
xmlMapper.writeValue(new File("src/test/resources/PersonGenerated.xml"), person);
|
||||||
|
}
|
||||||
|
|
||||||
private static String inputStreamToString(InputStream is) throws IOException {
|
private static String inputStreamToString(InputStream is) throws IOException {
|
||||||
BufferedReader br;
|
BufferedReader br;
|
||||||
StringBuilder sb = new StringBuilder();
|
StringBuilder sb = new StringBuilder();
|
||||||
|
@ -103,10 +155,10 @@ class SimpleBean {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
class SimpleBeanForCapitalizedFields {
|
class SimpleBeanForCapitalizedFields {
|
||||||
@JsonProperty("X")
|
@JsonProperty("X")
|
||||||
private int x = 1;
|
private int x = 1;
|
||||||
private int y = 2;
|
private int y = 2;
|
||||||
|
|
||||||
public int getX() {
|
public int getX() {
|
||||||
return x;
|
return x;
|
||||||
|
|
|
@ -0,0 +1,18 @@
|
||||||
|
<person>
|
||||||
|
<firstName></firstName>
|
||||||
|
<lastName></lastName>
|
||||||
|
<phoneNumbers>9911778981</phoneNumbers>
|
||||||
|
<phoneNumbers>9991111111</phoneNumbers>
|
||||||
|
<addresses>
|
||||||
|
<address>
|
||||||
|
<street_number>1</street_number>
|
||||||
|
<street_name>streetname1</street_name>
|
||||||
|
<city>city1</city>
|
||||||
|
</address>
|
||||||
|
<address>
|
||||||
|
<street_number>2</street_number>
|
||||||
|
<street_name>streetname2</street_name>
|
||||||
|
<city>city2</city>
|
||||||
|
</address>
|
||||||
|
</addresses>
|
||||||
|
</person>
|
|
@ -0,0 +1,19 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<person>
|
||||||
|
<firstName>Rohan</firstName>
|
||||||
|
<lastName>Daye</lastName>
|
||||||
|
<phoneNumbers>9911034731</phoneNumbers>
|
||||||
|
<phoneNumbers>9911033478</phoneNumbers>
|
||||||
|
<addresses>
|
||||||
|
<address>
|
||||||
|
<street_number>1</street_number>
|
||||||
|
<street_name>Name1</street_name>
|
||||||
|
<city>City1</city>
|
||||||
|
</address>
|
||||||
|
<address>
|
||||||
|
<street_number>2</street_number>
|
||||||
|
<street_name>Name2</street_name>
|
||||||
|
<city>City2</city>
|
||||||
|
</address>
|
||||||
|
</addresses>
|
||||||
|
</person>
|
|
@ -9,8 +9,10 @@ import java.util.Arrays;
|
||||||
import java.util.Collection;
|
import java.util.Collection;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.Map.Entry;
|
import java.util.Map.Entry;
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
import org.apache.commons.collections4.MultiMapUtils;
|
import org.apache.commons.collections4.MultiMapUtils;
|
||||||
|
import org.apache.commons.collections4.MultiSet;
|
||||||
import org.apache.commons.collections4.MultiValuedMap;
|
import org.apache.commons.collections4.MultiValuedMap;
|
||||||
import org.apache.commons.collections4.multimap.ArrayListValuedHashMap;
|
import org.apache.commons.collections4.multimap.ArrayListValuedHashMap;
|
||||||
import org.apache.commons.collections4.multimap.HashSetValuedHashMap;
|
import org.apache.commons.collections4.multimap.HashSetValuedHashMap;
|
||||||
|
@ -65,25 +67,28 @@ public class MultiValuedMapUnitTest {
|
||||||
@Test
|
@Test
|
||||||
public void givenMultiValuesMap_whenUsingKeysMethod_thenReturningAllKeys() {
|
public void givenMultiValuesMap_whenUsingKeysMethod_thenReturningAllKeys() {
|
||||||
MultiValuedMap<String, String> map = new ArrayListValuedHashMap<>();
|
MultiValuedMap<String, String> map = new ArrayListValuedHashMap<>();
|
||||||
|
|
||||||
map.put("fruits", "apple");
|
map.put("fruits", "apple");
|
||||||
map.put("fruits", "orange");
|
map.put("fruits", "orange");
|
||||||
map.put("vehicles", "car");
|
map.put("vehicles", "car");
|
||||||
map.put("vehicles", "bike");
|
map.put("vehicles", "bike");
|
||||||
|
|
||||||
assertThat(((Collection<String>) map.keys())).contains("fruits", "vehicles");
|
MultiSet<String> keys = map.keys();
|
||||||
|
|
||||||
|
assertThat((keys)).contains("fruits", "vehicles");
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void givenMultiValuesMap_whenUsingKeySetMethod_thenReturningAllKeys() {
|
public void givenMultiValuesMap_whenUsingKeySetMethod_thenReturningAllKeys() {
|
||||||
MultiValuedMap<String, String> map = new ArrayListValuedHashMap<>();
|
MultiValuedMap<String, String> map = new ArrayListValuedHashMap<>();
|
||||||
|
|
||||||
map.put("fruits", "apple");
|
map.put("fruits", "apple");
|
||||||
map.put("fruits", "orange");
|
map.put("fruits", "orange");
|
||||||
map.put("vehicles", "car");
|
map.put("vehicles", "car");
|
||||||
map.put("vehicles", "bike");
|
map.put("vehicles", "bike");
|
||||||
|
|
||||||
assertThat((Collection<String>) map.keySet()).contains("fruits", "vehicles");
|
Set<String> keys = map.keySet();
|
||||||
|
|
||||||
|
assertThat(keys).contains("fruits", "vehicles");
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -71,6 +71,10 @@ public interface UserRepository extends JpaRepository<User, Integer> , UserRepos
|
||||||
@Query(value = "UPDATE Users u SET u.status = ? WHERE u.name = ?", nativeQuery = true)
|
@Query(value = "UPDATE Users u SET u.status = ? WHERE u.name = ?", nativeQuery = true)
|
||||||
int updateUserSetStatusForNameNative(Integer status, String name);
|
int updateUserSetStatusForNameNative(Integer status, String name);
|
||||||
|
|
||||||
|
@Query(value = "INSERT INTO Users (name, age, email, status, active) VALUES (:name, :age, :email, :status, :active)", nativeQuery = true)
|
||||||
|
@Modifying
|
||||||
|
void insertUser(@Param("name") String name, @Param("age") Integer age, @Param("email") String email, @Param("status") Integer status, @Param("active") boolean active);
|
||||||
|
|
||||||
@Modifying
|
@Modifying
|
||||||
@Query(value = "UPDATE Users u SET status = ? WHERE u.name = ?", nativeQuery = true)
|
@Query(value = "UPDATE Users u SET status = ? WHERE u.name = ?", nativeQuery = true)
|
||||||
int updateUserSetStatusForNameNativePostgres(Integer status, String name);
|
int updateUserSetStatusForNameNativePostgres(Integer status, String name);
|
||||||
|
|
|
@ -30,10 +30,10 @@ class UserRepositoryCommon {
|
||||||
final String USER_EMAIL4 = "email4@example.com";
|
final String USER_EMAIL4 = "email4@example.com";
|
||||||
final Integer INACTIVE_STATUS = 0;
|
final Integer INACTIVE_STATUS = 0;
|
||||||
final Integer ACTIVE_STATUS = 1;
|
final Integer ACTIVE_STATUS = 1;
|
||||||
private final String USER_EMAIL5 = "email5@example.com";
|
final String USER_EMAIL5 = "email5@example.com";
|
||||||
private final String USER_EMAIL6 = "email6@example.com";
|
final String USER_EMAIL6 = "email6@example.com";
|
||||||
private final String USER_NAME_ADAM = "Adam";
|
final String USER_NAME_ADAM = "Adam";
|
||||||
private final String USER_NAME_PETER = "Peter";
|
final String USER_NAME_PETER = "Peter";
|
||||||
|
|
||||||
@Autowired
|
@Autowired
|
||||||
protected UserRepository userRepository;
|
protected UserRepository userRepository;
|
||||||
|
@ -389,6 +389,22 @@ class UserRepositoryCommon {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Test
|
||||||
|
@Transactional
|
||||||
|
public void whenInsertedWithQuery_ThenUserIsPersisted() {
|
||||||
|
userRepository.insertUser(USER_NAME_ADAM, 1, USER_EMAIL, ACTIVE_STATUS, true);
|
||||||
|
userRepository.insertUser(USER_NAME_PETER, 1, USER_EMAIL2, ACTIVE_STATUS, true);
|
||||||
|
|
||||||
|
User userAdam = userRepository.findUserByNameLike(USER_NAME_ADAM);
|
||||||
|
User userPeter = userRepository.findUserByNameLike(USER_NAME_PETER);
|
||||||
|
|
||||||
|
assertThat(userAdam).isNotNull();
|
||||||
|
assertThat(userAdam.getEmail()).isEqualTo(USER_EMAIL);
|
||||||
|
assertThat(userPeter).isNotNull();
|
||||||
|
assertThat(userPeter.getEmail()).isEqualTo(USER_EMAIL2);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
@Transactional
|
@Transactional
|
||||||
public void givenTwoUsers_whenFindByNameUsr01_ThenUserUsr01() {
|
public void givenTwoUsers_whenFindByNameUsr01_ThenUserUsr01() {
|
||||||
|
@ -520,7 +536,7 @@ class UserRepositoryCommon {
|
||||||
Query nativeQuery = entityManager.createNativeQuery("select deleted from USERS where NAME = 'usr01'");
|
Query nativeQuery = entityManager.createNativeQuery("select deleted from USERS where NAME = 'usr01'");
|
||||||
assertEquals(0, nativeQuery.getResultList().get(0));
|
assertEquals(0, nativeQuery.getResultList().get(0));
|
||||||
}
|
}
|
||||||
|
|
||||||
@After
|
@After
|
||||||
public void cleanUp() {
|
public void cleanUp() {
|
||||||
userRepository.deleteAll();
|
userRepository.deleteAll();
|
||||||
|
|
2
pom.xml
2
pom.xml
|
@ -1594,7 +1594,7 @@
|
||||||
<properties>
|
<properties>
|
||||||
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
|
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
|
||||||
<project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding>
|
<project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding>
|
||||||
<gib.referenceBranch>refs/heads/master</gib.referenceBranch>
|
<gib.referenceBranch>refs/remotes/origin/master</gib.referenceBranch>
|
||||||
<gib.skipTestsForNotImpactedModules>true</gib.skipTestsForNotImpactedModules>
|
<gib.skipTestsForNotImpactedModules>true</gib.skipTestsForNotImpactedModules>
|
||||||
<gib.failOnMissingGitDir>false</gib.failOnMissingGitDir>
|
<gib.failOnMissingGitDir>false</gib.failOnMissingGitDir>
|
||||||
<gib.failOnError>false</gib.failOnError>
|
<gib.failOnError>false</gib.failOnError>
|
||||||
|
|
|
@ -16,6 +16,8 @@
|
||||||
</parent>
|
</parent>
|
||||||
|
|
||||||
<dependencies>
|
<dependencies>
|
||||||
|
|
||||||
|
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>org.springframework.boot</groupId>
|
<groupId>org.springframework.boot</groupId>
|
||||||
<artifactId>spring-boot-starter-jdbc</artifactId>
|
<artifactId>spring-boot-starter-jdbc</artifactId>
|
||||||
|
@ -42,6 +44,17 @@
|
||||||
<scope>provided</scope>
|
<scope>provided</scope>
|
||||||
</dependency>
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.springframework.boot</groupId>
|
||||||
|
<artifactId>spring-boot-starter-data-jpa</artifactId>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.hibernate</groupId>
|
||||||
|
<artifactId>hibernate-jpamodelgen</artifactId>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
|
||||||
</dependencies>
|
</dependencies>
|
||||||
|
|
||||||
<build>
|
<build>
|
||||||
|
@ -57,4 +70,4 @@
|
||||||
<java.version>1.8</java.version>
|
<java.version>1.8</java.version>
|
||||||
</properties>
|
</properties>
|
||||||
|
|
||||||
</project>
|
</project>
|
|
@ -0,0 +1,34 @@
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
package com.baeldung.examples.security.sql;
|
||||||
|
|
||||||
|
import java.math.BigDecimal;
|
||||||
|
|
||||||
|
import javax.persistence.Entity;
|
||||||
|
import javax.persistence.GeneratedValue;
|
||||||
|
import javax.persistence.GenerationType;
|
||||||
|
import javax.persistence.Id;
|
||||||
|
import javax.persistence.Table;
|
||||||
|
|
||||||
|
import lombok.Data;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author Philippe
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
@Entity
|
||||||
|
@Table(name="Accounts")
|
||||||
|
@Data
|
||||||
|
public class Account {
|
||||||
|
|
||||||
|
@Id
|
||||||
|
@GeneratedValue(strategy=GenerationType.IDENTITY)
|
||||||
|
private Long id;
|
||||||
|
|
||||||
|
private String customerId;
|
||||||
|
private String accNumber;
|
||||||
|
private String branchId;
|
||||||
|
private BigDecimal balance;
|
||||||
|
|
||||||
|
}
|
|
@ -7,14 +7,24 @@ import java.sql.Connection;
|
||||||
import java.sql.PreparedStatement;
|
import java.sql.PreparedStatement;
|
||||||
import java.sql.ResultSet;
|
import java.sql.ResultSet;
|
||||||
import java.sql.SQLException;
|
import java.sql.SQLException;
|
||||||
|
import java.util.AbstractMap;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
import java.util.HashSet;
|
import java.util.HashSet;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
import java.util.stream.Collectors;
|
import java.util.stream.Collectors;
|
||||||
import java.util.stream.Stream;
|
import java.util.stream.Stream;
|
||||||
|
|
||||||
|
import javax.persistence.EntityManager;
|
||||||
|
import javax.persistence.Query;
|
||||||
|
import javax.persistence.TypedQuery;
|
||||||
|
import javax.persistence.criteria.CriteriaBuilder;
|
||||||
|
import javax.persistence.criteria.CriteriaQuery;
|
||||||
|
import javax.persistence.criteria.Order;
|
||||||
|
import javax.persistence.criteria.Root;
|
||||||
|
import javax.persistence.metamodel.SingularAttribute;
|
||||||
import javax.sql.DataSource;
|
import javax.sql.DataSource;
|
||||||
|
|
||||||
import org.springframework.stereotype.Component;
|
import org.springframework.stereotype.Component;
|
||||||
|
@ -27,9 +37,11 @@ import org.springframework.stereotype.Component;
|
||||||
public class AccountDAO {
|
public class AccountDAO {
|
||||||
|
|
||||||
private final DataSource dataSource;
|
private final DataSource dataSource;
|
||||||
|
private final EntityManager em;
|
||||||
|
|
||||||
public AccountDAO(DataSource dataSource) {
|
public AccountDAO(DataSource dataSource, EntityManager em) {
|
||||||
this.dataSource = dataSource;
|
this.dataSource = dataSource;
|
||||||
|
this.em = em;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -63,6 +75,26 @@ public class AccountDAO {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return all accounts owned by a given customer,given his/her external id - JPA version
|
||||||
|
*
|
||||||
|
* @param customerId
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
public List<AccountDTO> unsafeJpaFindAccountsByCustomerId(String customerId) {
|
||||||
|
String jql = "from Account where customerId = '" + customerId + "'";
|
||||||
|
TypedQuery<Account> q = em.createQuery(jql, Account.class);
|
||||||
|
return q.getResultList()
|
||||||
|
.stream()
|
||||||
|
.map(a -> AccountDTO.builder()
|
||||||
|
.accNumber(a.getAccNumber())
|
||||||
|
.balance(a.getBalance())
|
||||||
|
.branchId(a.getAccNumber())
|
||||||
|
.customerId(a.getCustomerId())
|
||||||
|
.build())
|
||||||
|
.collect(Collectors.toList());
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Return all accounts owned by a given customer,given his/her external id
|
* Return all accounts owned by a given customer,given his/her external id
|
||||||
*
|
*
|
||||||
|
@ -71,7 +103,7 @@ public class AccountDAO {
|
||||||
*/
|
*/
|
||||||
public List<AccountDTO> safeFindAccountsByCustomerId(String customerId) {
|
public List<AccountDTO> safeFindAccountsByCustomerId(String customerId) {
|
||||||
|
|
||||||
String sql = "select " + "customer_id,acc_number,branch_id,balance from Accounts where customer_id = ?";
|
String sql = "select customer_id, branch_id,acc_number,balance from Accounts where customer_id = ?";
|
||||||
|
|
||||||
try (Connection c = dataSource.getConnection(); PreparedStatement p = c.prepareStatement(sql)) {
|
try (Connection c = dataSource.getConnection(); PreparedStatement p = c.prepareStatement(sql)) {
|
||||||
p.setString(1, customerId);
|
p.setString(1, customerId);
|
||||||
|
@ -93,23 +125,73 @@ public class AccountDAO {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return all accounts owned by a given customer,given his/her external id - JPA version
|
||||||
|
*
|
||||||
|
* @param customerId
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
public List<AccountDTO> safeJpaFindAccountsByCustomerId(String customerId) {
|
||||||
|
|
||||||
|
String jql = "from Account where customerId = :customerId";
|
||||||
|
TypedQuery<Account> q = em.createQuery(jql, Account.class)
|
||||||
|
.setParameter("customerId", customerId);
|
||||||
|
|
||||||
|
return q.getResultList()
|
||||||
|
.stream()
|
||||||
|
.map(a -> AccountDTO.builder()
|
||||||
|
.accNumber(a.getAccNumber())
|
||||||
|
.balance(a.getBalance())
|
||||||
|
.branchId(a.getAccNumber())
|
||||||
|
.customerId(a.getCustomerId())
|
||||||
|
.build())
|
||||||
|
.collect(Collectors.toList());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return all accounts owned by a given customer,given his/her external id - JPA version
|
||||||
|
*
|
||||||
|
* @param customerId
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
public List<AccountDTO> safeJpaCriteriaFindAccountsByCustomerId(String customerId) {
|
||||||
|
|
||||||
|
CriteriaBuilder cb = em.getCriteriaBuilder();
|
||||||
|
CriteriaQuery<Account> cq = cb.createQuery(Account.class);
|
||||||
|
Root<Account> root = cq.from(Account.class);
|
||||||
|
cq.select(root)
|
||||||
|
.where(cb.equal(root.get(Account_.customerId), customerId));
|
||||||
|
|
||||||
|
TypedQuery<Account> q = em.createQuery(cq);
|
||||||
|
|
||||||
|
return q.getResultList()
|
||||||
|
.stream()
|
||||||
|
.map(a -> AccountDTO.builder()
|
||||||
|
.accNumber(a.getAccNumber())
|
||||||
|
.balance(a.getBalance())
|
||||||
|
.branchId(a.getAccNumber())
|
||||||
|
.customerId(a.getCustomerId())
|
||||||
|
.build())
|
||||||
|
.collect(Collectors.toList());
|
||||||
|
}
|
||||||
|
|
||||||
private static final Set<String> VALID_COLUMNS_FOR_ORDER_BY = Stream.of("acc_number", "branch_id", "balance")
|
private static final Set<String> VALID_COLUMNS_FOR_ORDER_BY = Stream.of("acc_number", "branch_id", "balance")
|
||||||
.collect(Collectors.toCollection(HashSet::new));
|
.collect(Collectors.toCollection(HashSet::new));
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Return all accounts owned by a given customer,given his/her external id
|
* Return all accounts owned by a given customer,given his/her external id
|
||||||
*
|
*
|
||||||
* @param customerId
|
* @param customerId
|
||||||
* @return
|
* @return
|
||||||
*/
|
*/
|
||||||
public List<AccountDTO> safeFindAccountsByCustomerId(String customerId, String orderBy) {
|
public List<AccountDTO> safeFindAccountsByCustomerId(String customerId, String orderBy) {
|
||||||
|
|
||||||
String sql = "select " + "customer_id,acc_number,branch_id,balance from Accounts where customer_id = ? ";
|
String sql = "select " + "customer_id,acc_number,branch_id,balance from Accounts where customer_id = ? ";
|
||||||
|
|
||||||
if (VALID_COLUMNS_FOR_ORDER_BY.contains(orderBy)) {
|
if (VALID_COLUMNS_FOR_ORDER_BY.contains(orderBy)) {
|
||||||
sql = sql + " order by " + orderBy;
|
sql = sql + " order by " + orderBy;
|
||||||
}
|
} else {
|
||||||
else {
|
|
||||||
throw new IllegalArgumentException("Nice try!");
|
throw new IllegalArgumentException("Nice try!");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -135,35 +217,82 @@ public class AccountDAO {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private static final Map<String,SingularAttribute<Account,?>> VALID_JPA_COLUMNS_FOR_ORDER_BY = Stream.of(
|
||||||
|
new AbstractMap.SimpleEntry<>(Account_.ACC_NUMBER, Account_.accNumber),
|
||||||
|
new AbstractMap.SimpleEntry<>(Account_.BRANCH_ID, Account_.branchId),
|
||||||
|
new AbstractMap.SimpleEntry<>(Account_.BALANCE, Account_.balance)
|
||||||
|
)
|
||||||
|
.collect(Collectors.toMap(Map.Entry::getKey, Map.Entry::getValue));
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return all accounts owned by a given customer,given his/her external id
|
||||||
|
*
|
||||||
|
* @param customerId
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
public List<AccountDTO> safeJpaFindAccountsByCustomerId(String customerId, String orderBy) {
|
||||||
|
|
||||||
|
SingularAttribute<Account,?> orderByAttribute = VALID_JPA_COLUMNS_FOR_ORDER_BY.get(orderBy);
|
||||||
|
if ( orderByAttribute == null) {
|
||||||
|
throw new IllegalArgumentException("Nice try!");
|
||||||
|
}
|
||||||
|
|
||||||
|
CriteriaBuilder cb = em.getCriteriaBuilder();
|
||||||
|
CriteriaQuery<Account> cq = cb.createQuery(Account.class);
|
||||||
|
Root<Account> root = cq.from(Account.class);
|
||||||
|
cq.select(root)
|
||||||
|
.where(cb.equal(root.get(Account_.customerId), customerId))
|
||||||
|
.orderBy(cb.asc(root.get(orderByAttribute)));
|
||||||
|
|
||||||
|
TypedQuery<Account> q = em.createQuery(cq);
|
||||||
|
|
||||||
|
return q.getResultList()
|
||||||
|
.stream()
|
||||||
|
.map(a -> AccountDTO.builder()
|
||||||
|
.accNumber(a.getAccNumber())
|
||||||
|
.balance(a.getBalance())
|
||||||
|
.branchId(a.getAccNumber())
|
||||||
|
.customerId(a.getCustomerId())
|
||||||
|
.build())
|
||||||
|
.collect(Collectors.toList());
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Invalid placeholder usage example
|
* Invalid placeholder usage example
|
||||||
*
|
*
|
||||||
* @param tableName
|
* @param tableName
|
||||||
* @return
|
* @return
|
||||||
*/
|
*/
|
||||||
public List<AccountDTO> wrongCountRecordsByTableName(String tableName) {
|
public Long wrongCountRecordsByTableName(String tableName) {
|
||||||
|
|
||||||
|
try (Connection c = dataSource.getConnection(); PreparedStatement p = c.prepareStatement("select count(*) from ?")) {
|
||||||
|
|
||||||
try (Connection c = dataSource.getConnection();
|
|
||||||
PreparedStatement p = c.prepareStatement("select count(*) from ?")) {
|
|
||||||
|
|
||||||
p.setString(1, tableName);
|
p.setString(1, tableName);
|
||||||
ResultSet rs = p.executeQuery();
|
ResultSet rs = p.executeQuery();
|
||||||
List<AccountDTO> accounts = new ArrayList<>();
|
rs.next();
|
||||||
while (rs.next()) {
|
return rs.getLong(1);
|
||||||
AccountDTO acc = AccountDTO.builder()
|
|
||||||
.customerId(rs.getString("customerId"))
|
|
||||||
.branchId(rs.getString("branch_id"))
|
|
||||||
.accNumber(rs.getString("acc_number"))
|
|
||||||
.balance(rs.getBigDecimal("balance"))
|
|
||||||
.build();
|
|
||||||
|
|
||||||
accounts.add(acc);
|
|
||||||
}
|
|
||||||
|
|
||||||
return accounts;
|
|
||||||
} catch (SQLException ex) {
|
} catch (SQLException ex) {
|
||||||
throw new RuntimeException(ex);
|
throw new RuntimeException(ex);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Invalid placeholder usage example - JPA
|
||||||
|
*
|
||||||
|
* @param tableName
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
public Long wrongJpaCountRecordsByTableName(String tableName) {
|
||||||
|
|
||||||
|
String jql = "select count(*) from :tableName";
|
||||||
|
TypedQuery<Long> q = em.createQuery(jql, Long.class)
|
||||||
|
.setParameter("tableName", tableName);
|
||||||
|
|
||||||
|
return q.getSingleResult();
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1 @@
|
||||||
|
|
|
@ -1,19 +0,0 @@
|
||||||
<databaseChangeLog
|
|
||||||
xmlns="http://www.liquibase.org/xml/ns/dbchangelog"
|
|
||||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
|
||||||
xsi:schemaLocation="http://www.liquibase.org/xml/ns/dbchangelog
|
|
||||||
http://www.liquibase.org/xml/ns/dbchangelog/dbchangelog-3.1.xsd">
|
|
||||||
|
|
||||||
<changeSet id="create-tables" author="baeldung">
|
|
||||||
<createTable tableName="Accounts" >
|
|
||||||
<column name="id" autoIncrement="true" type="BIGINT" remarks="Internal account PK" >
|
|
||||||
<constraints primaryKey="true"/>
|
|
||||||
</column>
|
|
||||||
<column name="customer_id" type="java.sql.Types.VARCHAR(32)" remarks="External Customer Id"></column>
|
|
||||||
<column name="acc_number" type="java.sql.Types.VARCHAR(128)" remarks="External Account Number"></column>
|
|
||||||
<column name="branch_id" type="java.sql.Types.VARCHAR(32)"></column>
|
|
||||||
<column name="balance" type="CURRENCY"></column>
|
|
||||||
|
|
||||||
</createTable>
|
|
||||||
</changeSet>
|
|
||||||
</databaseChangeLog>
|
|
|
@ -1,8 +0,0 @@
|
||||||
<databaseChangeLog
|
|
||||||
xmlns="http://www.liquibase.org/xml/ns/dbchangelog"
|
|
||||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
|
||||||
xsi:schemaLocation="http://www.liquibase.org/xml/ns/dbchangelog
|
|
||||||
http://www.liquibase.org/xml/ns/dbchangelog/dbchangelog-3.1.xsd">
|
|
||||||
|
|
||||||
<include file="changelog/create-tables.xml" relativeToChangelogFile="true"/>
|
|
||||||
</databaseChangeLog>
|
|
|
@ -40,6 +40,15 @@ public class SqlInjectionSamplesApplicationUnitTest {
|
||||||
assertThat(accounts).hasSize(3);
|
assertThat(accounts).hasSize(3);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void givenAVulnerableJpaMethod_whenHackedCustomerId_thenReturnAllAccounts() {
|
||||||
|
|
||||||
|
List<AccountDTO> accounts = target.unsafeJpaFindAccountsByCustomerId("C1' or '1'='1");
|
||||||
|
assertThat(accounts).isNotNull();
|
||||||
|
assertThat(accounts).isNotEmpty();
|
||||||
|
assertThat(accounts).hasSize(3);
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void givenASafeMethod_whenHackedCustomerId_thenReturnNoAccounts() {
|
public void givenASafeMethod_whenHackedCustomerId_thenReturnNoAccounts() {
|
||||||
|
|
||||||
|
@ -48,13 +57,36 @@ public class SqlInjectionSamplesApplicationUnitTest {
|
||||||
assertThat(accounts).isEmpty();
|
assertThat(accounts).isEmpty();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void givenASafeJpaMethod_whenHackedCustomerId_thenReturnNoAccounts() {
|
||||||
|
|
||||||
|
List<AccountDTO> accounts = target.safeJpaFindAccountsByCustomerId("C1' or '1'='1");
|
||||||
|
assertThat(accounts).isNotNull();
|
||||||
|
assertThat(accounts).isEmpty();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void givenASafeJpaCriteriaMethod_whenHackedCustomerId_thenReturnNoAccounts() {
|
||||||
|
|
||||||
|
List<AccountDTO> accounts = target.safeJpaCriteriaFindAccountsByCustomerId("C1' or '1'='1");
|
||||||
|
assertThat(accounts).isNotNull();
|
||||||
|
assertThat(accounts).isEmpty();
|
||||||
|
}
|
||||||
|
|
||||||
@Test(expected = IllegalArgumentException.class)
|
@Test(expected = IllegalArgumentException.class)
|
||||||
public void givenASafeMethod_whenInvalidOrderBy_thenThroweException() {
|
public void givenASafeMethod_whenInvalidOrderBy_thenThroweException() {
|
||||||
target.safeFindAccountsByCustomerId("C1", "INVALID");
|
target.safeFindAccountsByCustomerId("C1", "INVALID");
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test(expected = RuntimeException.class)
|
@Test(expected = Exception.class)
|
||||||
public void givenWrongPlaceholderUsageMethod_whenNormalCall_thenThrowsException() {
|
public void givenWrongPlaceholderUsageMethod_whenNormalCall_thenThrowsException() {
|
||||||
target.wrongCountRecordsByTableName("Accounts");
|
target.wrongCountRecordsByTableName("Accounts");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test(expected = Exception.class)
|
||||||
|
public void givenWrongJpaPlaceholderUsageMethod_whenNormalCall_thenThrowsException() {
|
||||||
|
target.wrongJpaCountRecordsByTableName("Accounts");
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,5 +2,17 @@
|
||||||
# Test profile configuration
|
# Test profile configuration
|
||||||
#
|
#
|
||||||
spring:
|
spring:
|
||||||
|
liquibase:
|
||||||
|
change-log: db/changelog/db.changelog-master.xml
|
||||||
|
|
||||||
|
jpa:
|
||||||
|
hibernate:
|
||||||
|
ddl-auto: none
|
||||||
|
|
||||||
datasource:
|
datasource:
|
||||||
initialization-mode: always
|
initialization-mode: embedded
|
||||||
|
|
||||||
|
logging:
|
||||||
|
level:
|
||||||
|
sql: DEBUG
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
create table Accounts (
|
create table Accounts (
|
||||||
|
id BIGINT NOT NULL GENERATED ALWAYS AS IDENTITY (START WITH 1, INCREMENT BY 1),
|
||||||
customer_id varchar(16) not null,
|
customer_id varchar(16) not null,
|
||||||
acc_number varchar(16) not null,
|
acc_number varchar(16) not null,
|
||||||
branch_id decimal(8,0),
|
branch_id decimal(8,0),
|
||||||
|
|
|
@ -3,15 +3,14 @@
|
||||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd"
|
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd"
|
||||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
|
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
|
||||||
<modelVersion>4.0.0</modelVersion>
|
<modelVersion>4.0.0</modelVersion>
|
||||||
<groupId>com.baeldung.spring-boot-crud</groupId>
|
|
||||||
<artifactId>spring-boot-crud</artifactId>
|
<artifactId>spring-boot-crud</artifactId>
|
||||||
<version>0.1.0</version>
|
|
||||||
<name>spring-boot-crud</name>
|
<name>spring-boot-crud</name>
|
||||||
|
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>org.springframework.boot</groupId>
|
<artifactId>parent-boot-2</artifactId>
|
||||||
<artifactId>spring-boot-starter-parent</artifactId>
|
<groupId>com.baeldung</groupId>
|
||||||
<version>2.0.6.RELEASE</version>
|
<version>0.0.1-SNAPSHOT</version>
|
||||||
|
<relativePath>../parent-boot-2</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
||||||
<dependencies>
|
<dependencies>
|
||||||
|
@ -41,11 +40,7 @@
|
||||||
<artifactId>h2</artifactId>
|
<artifactId>h2</artifactId>
|
||||||
<scope>runtime</scope>
|
<scope>runtime</scope>
|
||||||
</dependency>
|
</dependency>
|
||||||
</dependencies>
|
</dependencies>
|
||||||
<properties>
|
|
||||||
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
|
|
||||||
<java.version>1.8</java.version>
|
|
||||||
</properties>
|
|
||||||
|
|
||||||
<build>
|
<build>
|
||||||
<finalName>spring-boot-crud</finalName>
|
<finalName>spring-boot-crud</finalName>
|
||||||
|
@ -62,4 +57,10 @@
|
||||||
</plugin>
|
</plugin>
|
||||||
</plugins>
|
</plugins>
|
||||||
</build>
|
</build>
|
||||||
|
|
||||||
|
<properties>
|
||||||
|
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
|
||||||
|
<java.version>1.8</java.version>
|
||||||
|
</properties>
|
||||||
|
|
||||||
</project>
|
</project>
|
|
@ -6,10 +6,10 @@
|
||||||
<!-- this needs to use the boot parent directly in order to not inherit logback dependencies -->
|
<!-- this needs to use the boot parent directly in order to not inherit logback dependencies -->
|
||||||
|
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>org.springframework.boot</groupId>
|
<groupId>org.springframework.boot</groupId>
|
||||||
<artifactId>spring-boot-starter-parent</artifactId>
|
<artifactId>spring-boot-starter-parent</artifactId>
|
||||||
<version>2.1.1.RELEASE</version>
|
<version>2.1.1.RELEASE</version>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
||||||
<dependencies>
|
<dependencies>
|
||||||
<dependency>
|
<dependency>
|
||||||
|
|
|
@ -6,10 +6,10 @@
|
||||||
<!-- this needs to use the boot parent directly in order to not inherit logback dependencies -->
|
<!-- this needs to use the boot parent directly in order to not inherit logback dependencies -->
|
||||||
|
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>org.springframework.boot</groupId>
|
<groupId>org.springframework.boot</groupId>
|
||||||
<artifactId>spring-boot-starter-parent</artifactId>
|
<artifactId>spring-boot-starter-parent</artifactId>
|
||||||
<version>2.1.1.RELEASE</version>
|
<version>2.1.1.RELEASE</version>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
||||||
<dependencies>
|
<dependencies>
|
||||||
<dependency>
|
<dependency>
|
||||||
|
|
|
@ -8,13 +8,15 @@
|
||||||
<version>1.0.0</version>
|
<version>1.0.0</version>
|
||||||
|
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>org.springframework.boot</groupId>
|
<artifactId>parent-boot-2</artifactId>
|
||||||
<artifactId>spring-boot-starter-parent</artifactId>
|
<groupId>com.baeldung</groupId>
|
||||||
<version>2.1.2.RELEASE</version>
|
<version>0.0.1-SNAPSHOT</version>
|
||||||
|
<relativePath>../parent-boot-2</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
||||||
<properties>
|
<properties>
|
||||||
<java.version>1.8</java.version>
|
<java.version>1.8</java.version>
|
||||||
|
<spring-boot.version>2.1.2.RELEASE</spring-boot.version>
|
||||||
</properties>
|
</properties>
|
||||||
|
|
||||||
<dependencies>
|
<dependencies>
|
||||||
|
|
Loading…
Reference in New Issue