Repositories
Entity repositories provide a high-level abstraction for managing entities in the database. They offer methods for creating, reading, updating, and deleting entities, as well as querying and filtering based on specific criteria.
Getting a Repository
- Kotlin
- Java
Storm provides two ways to obtain a repository. The generic entity() method returns a built-in repository with standard CRUD operations. For custom query methods, define your own interface extending EntityRepository and retrieve it with repository() (covered below in Custom Repositories).
val orm = ORMTemplate.of(dataSource)
// Generic entity repository
val userRepository = orm.entity(User::class)
// Or using extension function
val userRepository = orm.entity<User>()
The Java API follows the same pattern as Kotlin. The generic entity() method provides standard CRUD operations; custom interfaces use repository().
var orm = ORMTemplate.of(dataSource);
// Generic entity repository
EntityRepository<User, Integer> userRepository = orm.entity(User.class);
Basic CRUD Operations
- Kotlin
- Java
All CRUD operations use the entity's primary key (marked with @PK) for identity. Insert returns the entity with any database-generated fields populated (such as auto-increment IDs). Update and delete match by primary key. Query methods accept metamodel-based filter expressions that compile to parameterized WHERE clauses.
// Create
val user = orm insert User(
email = "alice@example.com",
name = "Alice",
birthDate = LocalDate.of(1990, 5, 15)
)
// Read
val found: User? = orm.entity<User>().findById(user.id)
val alice: User? = orm.find(User_.name eq "Alice")
val all: List<User> = orm.findAll(User_.city eq city)
// Update
orm update user.copy(name = "Alice Johnson")
// Delete
orm delete user
// Delete by condition
orm.delete<User>(User_.city eq city)
// Delete all
orm.deleteAll<User>()
// Delete all (builder approach, requires unsafe() to confirm intent)
orm.entity(User::class).delete().unsafe().executeUpdate()
Java CRUD operations use the fluent builder pattern. Since Java records are immutable, updates require constructing a new record instance with the changed field values.
// Insert
User user = userRepository.insertAndFetch(new User(
null, "alice@example.com", "Alice", LocalDate.of(1990, 5, 15), city
));
// Read
Optional<User> found = userRepository.select()
.where(User_.id, EQUALS, user.id())
.getOptionalResult();
List<User> all = userRepository.select()
.where(User_.city, EQUALS, city)
.getResultList();
// Update
userRepository.update(new User(
user.id(), "alice@example.com", "Alice Johnson", user.birthDate(), user.city()
));
// Delete
userRepository.delete(user);
// Delete all
userRepository.deleteAll();
// Delete all (builder approach, requires unsafe() to confirm intent)
userRepository.delete().unsafe().executeUpdate();
Storm rejects DELETE and UPDATE queries that have no WHERE clause, throwing a PersistenceException. This prevents accidental bulk deletions, which is especially important because QueryBuilder is immutable and a lost where() return value would silently drop the filter. Call unsafe() to opt out of this check when you intentionally want to affect all rows. The deleteAll() convenience method calls unsafe() internally.
Storm uses dirty checking to determine which columns to include in the UPDATE statement. See Dirty Checking for configuration details.
Streaming
- Kotlin
- Java
For result sets that may be large, streaming avoids loading all rows into memory at once. Kotlin's Flow provides automatic resource management through structured concurrency: the underlying database cursor and connection are released when the flow completes or is cancelled, without requiring explicit cleanup.
val users: Flow<User> = userRepository.selectAll()
val count = users.count()
// Collect to list
val userList: List<User> = users.toList()
Java streams over database results hold open a database cursor and connection. You must close the stream explicitly, either with try-with-resources or by calling close(). Failing to close the stream leaks database connections.
try (Stream<User> users = userRepository.selectAll()) {
List<Integer> userIds = users.map(User::id).toList();
}
Unique Key Lookups
When a field is annotated with @UK, the metamodel generates a Metamodel.Key instance that enables type-safe single-result lookups:
- Kotlin
- Java
val user: User? = userRepository.findBy(User_.email, "alice@example.com")
val user: User = userRepository.getBy(User_.email, "alice@example.com") // throws if not found
Optional<User> user = userRepository.findBy(User_.email, "alice@example.com");
User user = userRepository.getBy(User_.email, "alice@example.com"); // throws if not found
Since @PK implies @UK, primary key fields also work with findBy and getBy.
Entities loaded within a transaction are cached. See Entity Cache for details.
Offset-Based Pagination
Storm provides built-in Page and Pageable types for offset-based pagination. These eliminate the need to write manual LIMIT/OFFSET queries or define your own page wrapper. The repository handles the count query and result slicing automatically. For query-builder-level pagination (manual offset/limit, Page with query builder), see Pagination and Scrolling: Pagination.
Page and Pageable
A Pageable describes a pagination request: which page to fetch, how many results per page, and an optional sort order. A Page holds the results along with metadata such as the total number of matching results, the total number of pages, and navigation helpers.
Page field / method | Description |
|---|---|
content | The list of results for this page |
totalCount | Total number of matching rows across all pages |
pageNumber() | Zero-based index of the current page |
pageSize() | Maximum number of elements per page |
totalPages() | Total number of pages |
hasNext() | Whether a next page exists |
hasPrevious() | Whether a previous page exists |
nextPageable() | Returns a Pageable for the next page (preserves sort orders) |
previousPageable() | Returns a Pageable for the previous page (preserves sort orders) |
Create a Pageable using one of the factory methods:
Pageable.ofSize(pageSize)creates a request for the first page (page 0) with the given size.Pageable.of(pageNumber, pageSize)creates a request for a specific page.- Chain
.sortBy(field)or.sortByDescending(field)to add sort orders.
Basic Usage
The simplest way to paginate is to call page(pageNumber, pageSize) on a repository. For more control over sorting, construct a Pageable and pass it to page(pageable).
- Kotlin
- Java
// First page of 20 users
val page1: Page<User> = userRepository.page(0, 20)
// Using Pageable with sort order
val pageable = Pageable.ofSize(20).sortBy(User_.name)
val page: Page<User> = userRepository.page(pageable)
// Navigate to next page
if (page.hasNext()) {
val nextPage = userRepository.page(page.nextPageable())
}
// First page of 20 users
Page<User> page1 = userRepository.page(0, 20);
// Using Pageable with sort order
Pageable pageable = Pageable.ofSize(20).sortBy(User_.name);
Page<User> page = userRepository.page(pageable);
// Navigate to next page
if (page.hasNext()) {
Page<User> nextPage = userRepository.page(page.nextPageable());
}
Ref Variants
Use pageRef to load only primary keys instead of full entities, returning a Page<Ref<E>>. This is useful when you need identifiers for a subsequent batch operation without the overhead of fetching full entity data.
- Kotlin
- Java
val refPage: Page<Ref<User>> = userRepository.pageRef(0, 20)
Page<Ref<User>> refPage = userRepository.pageRef(0, 20);
Scrolling
Repositories provide convenience methods for scrolling through result sets, where a unique column value (typically the primary key) acts as a cursor. This approach avoids the performance issues of OFFSET on large tables, because the database can seek directly to the cursor position using an index rather than scanning and discarding skipped rows.
The key parameter must be a Metamodel.Key, which is generated for fields annotated with @UK or @PK. See Metamodel for details.
The scroll method accepts a Scrollable<E> that captures the cursor state (key, page size, direction, and cursor values) and returns a Window<E> containing the page content, informational hasNext/hasPrevious flags, and Scrollable<E> navigation tokens for fetching the adjacent window. Navigation tokens (nextScrollable(), previousScrollable()) are always present when the window has content; they are only null when the window is empty. The hasNext and hasPrevious flags indicate whether more results existed at query time, but they do not gate access to the navigation tokens. Since new data may appear after the query, the developer decides whether to follow a cursor.
Create a Scrollable using the factory methods, then use the navigation tokens on the returned Window to move forward or backward:
- Kotlin
- Java
// First page of 20 users ordered by ID
val window: Window<User> = userRepository.scroll(Scrollable.of(User_.id, 20))
// Next page (nextScrollable() is non-null whenever the window has content)
val next: Window<User> = userRepository.scroll(window.nextScrollable())
// Previous page
val previous: Window<User> = userRepository.scroll(window.previousScrollable())
// Optionally check hasNext/hasPrevious to decide whether to follow the cursor.
// These flags reflect a snapshot at query time; new data may appear afterward.
if (window.hasNext()) {
// more results existed when the query ran
}
To scroll through a filtered subset, use the query builder with scroll as a terminal operation. The filter and cursor conditions are combined with AND.
val activeWindow = userRepository.select()
.where(User_.active, EQUALS, true)
.scroll(Scrollable.of(User_.id, 20))
val nextActive = userRepository.select()
.where(User_.active, EQUALS, true)
.scroll(activeWindow.nextScrollable())
For backward scrolling (starting from the end of the result set), use .backward():
val lastWindow: Window<User> = userRepository.scroll(Scrollable.of(User_.id, 20).backward())
The scroll methods handle ordering internally and reject explicit orderBy() calls. Backward scrolling returns results in descending key order; reverse the list if you need ascending order for display. See Pagination and Scrolling: Scrolling for full details on ordering constraints.
The same scrolling methods described in the Kotlin section are available on Java repositories. The scroll method accepts a Scrollable<E> and returns a Window<E> containing the page content(), informational hasNext()/hasPrevious() flags, and Scrollable<E> navigation tokens that are always present when the window has content.
// First page of 20 users ordered by ID
Window<User> window = userRepository.scroll(Scrollable.of(User_.id, 20));
// Next page (nextScrollable() is non-null whenever the window has content)
Window<User> next = userRepository.scroll(window.nextScrollable());
// Previous page
Window<User> previous = userRepository.scroll(window.previousScrollable());
// Optionally check hasNext/hasPrevious to decide whether to follow the cursor.
// These flags reflect a snapshot at query time; new data may appear afterward.
if (window.hasNext()) {
// more results existed when the query ran
}
For filtered results, use the query builder and call scroll as a terminal operation. The filter and cursor conditions are combined with AND.
Window<User> activeWindow = userRepository.select()
.where(User_.active, EQUALS, true)
.scroll(Scrollable.of(User_.id, 20));
For backward scrolling (starting from the end of the result set), use .backward():
Window<User> lastWindow = userRepository.scroll(Scrollable.of(User_.id, 20).backward());
As with Kotlin, the scroll methods handle ordering internally and reject explicit orderBy() calls. Backward scrolling returns results in descending key order. See Pagination and Scrolling: Scrolling for full details.
Scrolling with Sort
When you need to sort by a non-unique column (for example, a date or status), use the Scrollable.of overload that accepts a separate sort column. This accepts a key column (typically the primary key) as a unique tiebreaker, and a sort column for the primary sort order, to guarantee deterministic paging even when sort values repeat.
- Kotlin
- Java
// First page sorted by creation date, with ID as tiebreaker
val window: Window<Post> = postRepository.scroll(Scrollable.of(Post_.id, Post_.createdAt, 20))
// Next page
val next: Window<Post> = postRepository.scroll(window.nextScrollable())
// With filter (use query builder)
val activeWindow = postRepository.select()
.where(Post_.active, EQUALS, true)
.scroll(Scrollable.of(Post_.id, Post_.createdAt, 20))
// First page sorted by creation date, with ID as tiebreaker
Window<Post> window = postRepository.scroll(Scrollable.of(Post_.id, Post_.createdAt, 20));
// Next page
Window<Post> next = postRepository.scroll(window.nextScrollable());
The Window carries navigation tokens (nextScrollable(), previousScrollable()) that encode the cursor values internally, so the client does not need to extract cursor values manually. These tokens are always non-null when the window contains content. For REST APIs, nextCursor() and previousCursor() provide a convenient serialized form: nextCursor() returns null when hasNext is false, and previousCursor() returns null when hasPrevious is false.
For queries that need joins, projections, or more complex filtering, use the query builder and call scroll as a terminal operation. See Pagination and Scrolling: Scrolling for full details on how scrolling composes with WHERE and ORDER BY clauses, including indexing recommendations.
Pagination vs. Scrolling
Storm supports two strategies for traversing large result sets. The table below summarizes the trade-offs to help you choose.
| Factor | Pagination (page) | Scrolling (scroll) |
|---|---|---|
| Request type | Pageable | Scrollable<T> |
| Result type | Page | Window |
| Navigation | page number | cursor |
| Count query | yes | no |
| Random access | yes | no |
| Performance at page 1 | Good | Good |
| Performance at page 1,000 | Degrades (database must skip rows) | Consistent (index seek) |
| Handles concurrent inserts | Rows may shift between pages | Stable cursor |
| Navigate forward | page.nextPageable() | window.nextScrollable() |
| Navigate backward | page.previousPageable() | window.previousScrollable() |
Use pagination when you need random page access or a total count (for example, displaying "Page 3 of 12" in a UI). Use scrolling when you need consistent performance over deep result sets or when the data changes frequently between requests.
Refs
Refs are lightweight identifiers that carry only the record type and primary key. Selecting refs instead of full entities reduces memory usage and network bandwidth when you only need IDs for subsequent operations, such as batch lookups or filtering. See Refs for a detailed discussion.
- Kotlin
- Java
// Select refs (lightweight identifiers)
val refs: Flow<Ref<User>> = userRepository.selectAllRef()
// Select by refs
val users: Flow<User> = userRepository.selectByRef(refs)
Ref operations in Java return Stream objects that must be closed. Refs carry only the primary key and record type, making them suitable for batch operations where loading full records would be wasteful.
// Select refs (lightweight identifiers)
try (Stream<Ref<User>> refs = userRepository.selectAllRef()) {
// Process refs
}
// Select by refs
List<Ref<User>> refList = ...;
try (Stream<User> users = userRepository.selectByRef(refList.stream())) {
// Process users
}
Custom Repositories
- Kotlin
- Java
Custom repositories let you encapsulate domain-specific queries behind a typed interface. Define an interface that extends EntityRepository, add methods with default implementations that use the inherited query API, and retrieve it from orm.repository(). This keeps query logic in a single place and makes it testable through interface substitution.
The advantage over using the generic entity() repository is that custom methods express domain intent (e.g., findByEmail) rather than exposing raw query construction to callers.
interface UserRepository : EntityRepository<User, Int> {
// Custom query method
fun findByEmail(email: String): User? =
find(User_.email eq email)
// Custom query with multiple conditions
fun findByNameInCity(name: String, city: City): List<User> =
findAll((User_.city eq city) and (User_.name eq name))
}
Get the repository:
val userRepository: UserRepository = orm.repository<UserRepository>()
Java custom repositories follow the same pattern as Kotlin, using default methods to provide implementations. The fluent builder API chains where, and, and or calls to construct type-safe filter expressions.
interface UserRepository extends EntityRepository<User, Integer> {
// Custom query method
default Optional<User> findByEmail(String email) {
return select()
.where(User_.email, EQUALS, email)
.getOptionalResult();
}
// Custom query with multiple conditions
default List<User> findByNameInCity(String name, City city) {
return select()
.where(it -> it.where(User_.city, EQUALS, city)
.and(it.where(User_.name, EQUALS, name)))
.getResultList();
}
}
Get the repository:
UserRepository userRepository = orm.repository(UserRepository.class);
Repository with Spring
- Kotlin
- Java
Repositories can be injected using Spring's dependency injection:
@Service
class UserService(
private val userRepository: UserRepository
) {
fun findUser(email: String): User? =
userRepository.findByEmail(email)
}
Repositories can be injected using Spring's dependency injection:
@Service
public class UserService {
private final UserRepository userRepository;
public UserService(UserRepository userRepository) {
this.userRepository = userRepository;
}
public Optional<User> findUser(String email) {
return userRepository.findByEmail(email);
}
}
Spring Configuration
Storm repositories are plain interfaces, so Spring cannot discover them through component scanning. The RepositoryBeanFactoryPostProcessor bridges this gap by scanning specified packages for interfaces that extend EntityRepository or ProjectionRepository and registering proxy implementations as Spring beans. Once registered, you can inject repositories through standard constructor injection. See Spring Integration for full configuration details.
- Kotlin
- Java
@Configuration
class AcmeRepositoryBeanFactoryPostProcessor : RepositoryBeanFactoryPostProcessor() {
override val repositoryBasePackages: Array<String>
get() = arrayOf("com.acme.repository")
}
@Configuration
public class AcmeRepositoryBeanFactoryPostProcessor extends RepositoryBeanFactoryPostProcessor {
@Override
public String[] getRepositoryBasePackages() {
return new String[] { "com.acme.repository" };
}
}
Tips
- Use custom repositories. Encapsulate domain-specific queries in repository interfaces.
- Close streams. Always close
Streamresults to release database resources. - Prefer Kotlin Flow. Kotlin's Flow automatically handles resource cleanup.
- Use Spring injection. Let Spring manage repository lifecycle for cleaner code.