This is my template for a spring boot project that uses spring data and Couchbase. It includes integration and unit tests.
- Controller: is the presentation layer where the end points are located
- Service: is the service layer where the business logic resides
- Repository: is the persistence layer where the CRUD repository is located
- Spring Boot (spring-boot-starter-web, spring-boot-starter-tomcat, spring-boot-starter-test, spring-boot-starter-data-couchbase)
- Java 8
- Tomcat 8.5.x
- Couchbase 5.x
- Maven
- Integration Test (for the Controller): it uses the Spring Boot Test framework with mockMvc and hamcrest matchers
- Unit Test (for the Service): it uses the Mockito framework with hamcrest matchers, mock and injectMocks annotations
- User -> key: external Couchbase atomic counter (with manual prefix) | bucket: default | repo: crud repo
- Car -> key: doc attributes (with automatic prefix and suffix) | bucket: default | repo: template
- Product -> key: auto generated Couchbase unique number | bucket: default | repo: template
- Place -> key: external Couchbase atomic counter with doc attributes (with manual prefix) | bucket: non-default | repo: crud repo (for multiple buckets)
- Phone -> key: external Java UUID key with doc attributes (with automatic prefix) | bucket: non-default | repo: crud repo (for multiple buckets)
1. Get user by id. HTTP Method: GET
http://localhost:8080/springdatacouchbase/users/1
2. Create a user. HTTP Method: POST
http://localhost:8080/springdatacouchbase/users
{
"name": "Carlos",
"nicknames": ["charz"],
"age": 25,
"email": "carlos1@yopmail.com"
}
3. Update a user. HTTP Method: PUT
http://localhost:8080/springdatacouchbase/users/1
{
"name": "Carlos2",
"age": 26
}
4. Delete a user. HTTP Method: DELETE
http://localhost:8080/springdatacouchbase/users/1
5. Find user by email. HTTP Method: GET
http://localhost:8080/springdatacouchbase/users/find/email?email=carlos1@yopmail.com
6. Find user by nickname. HTTP Method: GET
http://localhost:8080/springdatacouchbase/users/find/nickname?nickname=charz
7. Find user by name. HTTP Method: GET
http://localhost:8080/springdatacouchbase/users/find/name?name=carlos
8. User exists? HTTP Method: GET
http://localhost:8080/springdatacouchbase/users/1/exists
9. Find all users. HTTP Method: GET
http://localhost:8080/springdatacouchbase/users/find/all
10. Count all users. HTTP Method: GET
http://localhost:8080/springdatacouchbase/users/count/all
11. Delete users by age (to exemplify DELETE with N1QL in the CRUD repo). HTTP Method: POST
http://localhost:8080/springdatacouchbase/users/delete/age?age=50
12. Get users by name using template N1ql projection with cover index. HTTP Method: GET
http://localhost:8080/springdatacouchbase/users/find/name/coverindex?name=carlos
13. Get users ids using template N1ql. HTTP Method: GET
http://localhost:8080/springdatacouchbase/users/find/ids
14. Get users using USE KEYS. HTTP Method: GET
http://localhost:8080/springdatacouchbase/users/find/usekeys?ids=1,2,3
- To create a primary index on the bucket.
CREATE PRIMARY INDEX `users_primary_index` ON `users` USING GSI;
- To delete the primary index of the bucket.
DROP INDEX `users`.`users_primary_index` USING GSI;
- To create a secondary index for the query 'findUsersWithName'
CREATE INDEX `idx_user_find_by_name` ON `users` (name) WHERE type = 'com.wedevol.springdatacouchbase.core.dao.doc.UserDoc' USING GSI;
- To delete the secondary index: idx_user_find_by_name.
DROP INDEX `users`.`idx_user_find_by_name` USING GSI;
- To create a secondary index for the query 'findUsersWithNickname'
CREATE INDEX `idx_user_find_by_nickname` ON `users` (DISTINCT ARRAY x FOR x IN nicknames END) WHERE type = 'com.wedevol.springdatacouchbase.core.dao.doc.UserDoc' USING GSI;
- To delete the secondary index: idx_user_find_by_nickname.
DROP INDEX `users`.`idx_user_find_by_nickname` USING GSI;
- In the UserDoc.java (@Document) we can annotate the key (@id) to be part of the json as well using @Field.
- In the UserRepository, the CrudRepository provides sophisticated CRUD functionality for the entity class that is being managed.
- Above Couchbase 5 the bucket name and password is the username and password located on the Security tab on the web console.
- For doc expiration use for example: @Document(expiry = 10) or @Document(expiryExpression = "${valid.document.expiry}").
- Key generation: UserDoc -> uses a Couchbase atomic counter. CarDoc -> uses doc attributes. ProductDoc -> uses random uuid.
- Couchbase CRUD Repository documentation: There you will find core concepts.
- Spring Data and Couchbase: There you will find more considerations when working with spring data and Couchbase.
- Introduction to spring data and Couchbase: There you will find an introduction example.
- Couchbase CRUD Example: There you will find a CRUD example.
- Couchbase 5: The new Couchbase 5 is role based and there are some changes that need to be considered.
- Spring Data Couchbase Properties: Spring Boot common application properties.
- Spring Data Couchbase Documentation: Spring Data Couchbase Documentation.
- Spring Boot 2 - Release Notes: Spring Boot 2.0 Release Notes.
- Couchbase Spring Data - More customization: Couchbase Spring Data - More Customization (in Spanish)
- Couchbase Spring Data - Official Git: Couchbase Spring Data Official Git
- Couchbase Spring Data - Pitfalls: Pitfalls to avoid
- Couchbase Spring Data - N1ql Examples: More N1ql examples using bucket and template
- Couchbase SDK - N1ql Examples: More Couchbase SDK N1ql Examples
I am Carlos Becerra - MSc. Softwware & Systems. But to tell you the truth, I'd prefer to be a passionate developer. You can contact me via:
Any improvement or comment about the project is always welcome! As well as others shared their code publicly I want to share mine! Thanks!
Copyright 2020 Carlos Becerra
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.