Using Custom SQL Queries
Service Builder creates finder methods that retrieve entities by their attributes (their column values). When you add a column as a parameter for the finder in your service.xml
file and run Service Builder, it generates the finder method in your persistence layer and adds methods to your service layer that invoke the finder. If your queries are simple enough, you can use Dynamic Query to access Liferay’s database. If you want to do something more complicated (like JOINs), you can write your own custom SQL queries.
Custom SQL is recommended for custom services. For querying out-of-the-box Liferay entities, use DynamicQuery or DSLQuery.
Deploy an Example
Start a new Liferay instance by running
docker run -it -m 8g -p 8080:8080 liferay/portal:7.4.3.120-ga120
Sign in to Liferay at http://localhost:8080. Use the email address test@liferay.com and the password test. When prompted, change the password to learn.
Then, follow these steps to deploy the example:
-
Download and unzip the
liferay-p9z0.zip
example project.curl https://resources.learn.liferay.com/dxp/latest/en/liferay-development/building-applications/data-frameworks/service-builder/advanced-queries/liferay-p9z0.zip -O
unzip liferay-p9z0.zip
-
Build and deploy the project module.
cd liferay-p9z0
./gradlew deploy -Ddeploy.docker.container.id=$(docker ps -lq)
NoteThis command is the same as copying the deployed jars to
/opt/liferay/osgi/modules
on the Docker container. -
Confirm the deployment in the Liferay Docker container console.
STARTED com.liferay.p9z0.web_1.0.0 [1386] STARTED com.liferay.p9z0.api_1.0.0 [1384] STARTED com.liferay.p9z0.service_1.0.0 [1385]
-
To verify the example module is working, open your browser to
https://localhost:8080
. -
Add the P9Z0 Portlet to a page. You can find the example portlet under Sample in Widgets.
-
Add an entry by entering a name and a description. Leave Hidden unchecked. Click Add, and the new entry appears under P9Z0 Entries.
-
Add another entry with a different name and description. This time, check Hidden. The new entry doesn’t appear under P9Z0 Entries.
This example uses custom SQL to retrieve only entries with a specified value in the database (hidden_ = false
).
Adding Custom SQL to the Code
- Open
P9Z0EntryLocalServiceImpl.java
. The code for the custom SQL request is defined in thegetEntries
method.
@Override
public List<P9Z0Entry> getP9Z0Entries(boolean hidden) {
Session session = null;
try {
session = p9z0EntryPersistence.openSession();
String sql = "select * from P9Z0_P9Z0Entry where hidden_ = ?";
SQLQuery sqlQuery = session.createSynchronizedSQLQuery(sql);
sqlQuery.addEntity("P9Z0_P9Z0Entry", P9Z0EntryImpl.class);
QueryPos queryPos = QueryPos.getInstance(sqlQuery);
queryPos.add(hidden);
return (List<P9Z0Entry>)QueryUtil.list(
sqlQuery, p9z0EntryPersistence.getDialect(), QueryUtil.ALL_POS,
QueryUtil.ALL_POS);
}
catch (Exception exception) {
throw new SystemException(exception);
}
finally {
p9z0EntryPersistence.closeSession(session);
}
}
- Store your SQL query in a string, replacing any variable values with a
?
.
String sql = "select * from P9Z0_P9Z0Entry where hidden_ = ?";
- Create an
SQLQuery
object. Liferay uses this object to handle SQL queries. The session creates anSQLQuery
object to avoid issues in a clustered environment. Use theaddEntity
method to define the class of the object your query returns.
SQLQuery sqlQuery = session.createSynchronizedSQLQuery(sql);
sqlQuery.addEntity("P9Z0_P9Z0Entry", P9Z0EntryImpl.class);
- If you used the
?
placeholder in the string, create aQueryPos
object from theSQLQuery
. Add the values to be replaced in the order they appear on the string. This example only uses one.
QueryPos queryPos = QueryPos.getInstance(sqlQuery);
queryPos.add(hidden);
QueryUtil
executes the query in the database. Thelist
method is used to makeGET
calls and returns a list of the values in the response.
return (List<P9Z0Entry>)QueryUtil.list(
sqlQuery, p9z0EntryPersistence.getDialect(), QueryUtil.ALL_POS,
QueryUtil.ALL_POS);