{"id":5267,"date":"2018-06-11T03:01:59","date_gmt":"2018-06-11T10:01:59","guid":{"rendered":"https:\/\/www.couchbase.com\/blog\/?p=5267"},"modified":"2023-07-20T14:48:34","modified_gmt":"2023-07-20T21:48:34","slug":"kotlin-spring-boot-spring-data","status":"publish","type":"post","link":"https:\/\/www.couchbase.com\/blog\/pt\/kotlin-spring-boot-spring-data\/","title":{"rendered":"Couchbase com Kotlin, Spring Boot e Spring Data"},"content":{"rendered":"<p>No ano passado, comecei a aprender Kotlin e fiquei surpreso com a facilidade de converter um aplicativo Java. O IntelliJ e alguns outros IDEs oferecem boas ferramentas para convers\u00e3o autom\u00e1tica e, com alguns ajustes, voc\u00ea pode obter um c\u00f3digo muito mais conciso e menos propenso a erros.<\/p>\n<p><img loading=\"lazy\" decoding=\"async\" class=\"aligncenter wp-image-5268\" src=\"https:\/\/www.couchbase.com\/blog\/wp-content\/uploads\/2018\/06\/Screen-Shot-2018-06-04-at-10.34.39-PM-1024x252.png\" alt=\"\" width=\"721\" height=\"177\" srcset=\"https:\/\/www.couchbase.com\/blog\/wp-content\/uploads\/sites\/1\/2018\/06\/Screen-Shot-2018-06-04-at-10.34.39-PM-1024x252.png 1024w, https:\/\/www.couchbase.com\/blog\/wp-content\/uploads\/sites\/1\/2018\/06\/Screen-Shot-2018-06-04-at-10.34.39-PM-300x74.png 300w, https:\/\/www.couchbase.com\/blog\/wp-content\/uploads\/sites\/1\/2018\/06\/Screen-Shot-2018-06-04-at-10.34.39-PM-768x189.png 768w, https:\/\/www.couchbase.com\/blog\/wp-content\/uploads\/sites\/1\/2018\/06\/Screen-Shot-2018-06-04-at-10.34.39-PM-20x5.png 20w, https:\/\/www.couchbase.com\/blog\/wp-content\/uploads\/sites\/1\/2018\/06\/Screen-Shot-2018-06-04-at-10.34.39-PM.png 1120w\" sizes=\"auto, (max-width: 721px) 100vw, 721px\" \/><\/p>\n<p>Por isso, decidi criar um aplicativo de amostra para mostrar minha nova combina\u00e7\u00e3o favorita: Kotlin, <a href=\"https:\/\/www.couchbase.com\/blog\/pt\/couchbase-spring-boot-spring-data\/\">Spring Boot, Spring Data e Couchbase<\/a>:<\/p>\n<h2><strong>Cria\u00e7\u00e3o de um servi\u00e7o de perfil de usu\u00e1rio <\/strong><\/h2>\n<p>Voc\u00ea pode clonar o projeto inteiro aqui:<\/p>\n<p><a href=\"https:\/\/github.com\/couchbaselabs\/try-cb-kotlin\">https:\/\/github.com\/couchbaselabs\/try-cb-kotlin<\/a><\/p>\n<p>Vamos come\u00e7ar criando nossa classe principal:<\/p>\n<pre class=\"lang:default decode:true\">@SpringBootApplication\r\nopen class KotlinDemoApplication\r\n\r\nfun main(args: Array&lt;String&gt;) {\r\n    SpringApplication.run(KotlinDemoApplication::class.java, *args)\r\n}<\/pre>\n<p><strong>\u00a0Observa\u00e7\u00e3o:<\/strong> Sua classe deve ser <strong>aberto<\/strong> Caso contr\u00e1rio, voc\u00ea receber\u00e1 o seguinte erro:<\/p>\n<pre class=\"lang:default decode:true\">org.springframework.beans.factory.parsing.BeanDefinitionParsingException: Configuration problem: @Configuration class 'KotlinDemoApplication' may not be final. Remove the final modifier to continue.\r\nOffending resource: com.couchbase.KotlinDemoApplication\r\n\tat org.springframework.beans.factory.parsing.FailFastProblemReporter.error(FailFastProblemReporter.java:70) ~[spring-beans-4.3.13.RELEASE.jar:4.3.13.RELEASE]\r\n\tat org.springframework.context.annotation.ConfigurationClass.validate(ConfigurationClass.java:214) ~[spring-context-4.3.13.RELEASE.jar:4.3.13.RELEASE]\r\n<\/pre>\n<p>Aqui est\u00e1 nossa entidade de usu\u00e1rio, que \u00e9 muito semelhante a <a href=\"https:\/\/github.com\/couchbaselabs\/kubernetes-starter-kit\/blob\/master\/src\/main\/java\/com\/cb\/springdata\/sample\/entities\/User.java\">o de Java<\/a>:<\/p>\n<pre class=\"lang:default decode:true\">@Document\r\nclass User(): BasicEntity() {\r\n\r\n    constructor(id: String,\r\n                name: String,\r\n                address: Address,\r\n                preferences: List&lt;Preference&gt;,\r\n                securityRoles: List&lt;String&gt;): this(){\r\n\r\n        this.id = id;\r\n        this.name = name;\r\n        this.address = address;\r\n        this.preferences = preferences;\r\n        this.securityRoles = securityRoles;\r\n    }\r\n\r\n    @Id\r\n    var id: String? = null\r\n\r\n    @NotNull\r\n    var name: String? = null\r\n\r\n    @Field\r\n    var address: Address? = null\r\n\r\n    @Field\r\n    var preferences: List&lt;Preference&gt; = emptyList()\r\n\r\n    @Field\r\n    var securityRoles: List&lt;String&gt; = emptyList()\r\n}<\/pre>\n<ul>\n<li><strong>@Document:\u00a0<\/strong>A anota\u00e7\u00e3o do Couchbase que define uma entidade, semelhante a\u00a0<strong><em>@Entidade<\/em><\/strong>\u00a0em JPA. O Couchbase adicionar\u00e1 automaticamente uma propriedade chamada\u00a0<strong><em>_classe<\/em><\/strong>\u00a0no documento para us\u00e1-lo como o tipo de documento.<\/li>\n<li><strong>@Id:\u00a0<\/strong>A chave do documento<\/li>\n<li><strong>@Campo:<\/strong>\u00a0As anota\u00e7\u00f5es do Couchbase, semelhantes \u00e0s do JPA\u00a0<strong><em>@Coluna<\/em><\/strong>. N\u00e3o \u00e9 obrigat\u00f3rio, mas recomendamos us\u00e1-lo.<\/li>\n<\/ul>\n<p>O mapeamento de atributos no Couchbase \u00e9 muito simples. Eles ser\u00e3o mapeados diretamente para a estrutura correspondente no JSON:<\/p>\n<ul>\n<li><strong>Propriedades simples:<\/strong> Mapeamento direto para JSON:<\/li>\n<\/ul>\n<pre class=\"lang:js decode:true\">{\r\n \"id\": \"user::1\",\r\n \"name\": \"Denis Rosa\"\r\n}<\/pre>\n<ul>\n<li><strong>Matrizes: <\/strong>Como \u00e9 de se esperar, matrizes como <strong>fun\u00e7\u00f5es de seguran\u00e7a<\/strong> ser\u00e3o convertidos em matrizes JSON:<strong>\u00a0<\/strong><\/li>\n<\/ul>\n<pre class=\"lang:default decode:true\">{\r\n \"securityRoles\": [\"admin\", \"user\"]\r\n}<\/pre>\n<p>&nbsp;<\/p>\n<ul>\n<li><strong>Entidades aninhadas:<\/strong> Voc\u00ea odeia mapear <strong>@ManyToOne<\/strong> relacionamentos? Eu tamb\u00e9m. Como estamos usando um banco de dados de documentos, n\u00e3o h\u00e1 mais necessidade de escrever esses relacionamentos; as entidades aninhadas tamb\u00e9m s\u00e3o traduzidas diretamente para JSON.<\/li>\n<\/ul>\n<pre class=\"lang:default decode:true\">{  \r\n   \"id\":\"user::1\",\r\n   \"name\":\"Denis Rosa\",\r\n   \"address\":{  \r\n      \"streetName\":\"A Street Somewhere\",\r\n      \"houseNumber\":\"42\",\r\n      \"postalCode\":\"81234\",\r\n      \"city\":\"Munich\",\r\n      \"country\":\"DE\"\r\n   },\r\n   \"preferences\":[  \r\n      {  \r\n         \"name\":\"lang\",\r\n         \"value\":\"EN\"\r\n      }\r\n   ],\r\n   \"securityRoles\":[  \r\n      \"admin\",\r\n      \"user\"\r\n   ]\r\n}<\/pre>\n<h2><strong>Reposit\u00f3rios<\/strong><\/h2>\n<p>Agora, vamos dar uma olhada em como ser\u00e1 o nosso reposit\u00f3rio:<\/p>\n<pre class=\"lang:default decode:true\">@N1qlPrimaryIndexed\r\n@ViewIndexed(designDoc = \"user\")\r\ninterface UserRepository : CouchbasePagingAndSortingRepository&lt;User, String&gt; {\r\n\r\n    fun findByName(name: String): List&lt;User&gt;\r\n\r\n    @Query(\"#{#n1ql.selectEntity} where #{#n1ql.filter} and ANY preference IN \" + \" preferences SATISFIES preference.name = $1 END\")\r\n    fun findUsersByPreferenceName(name: String): List&lt;User&gt;\r\n\r\n    @Query(\"#{#n1ql.selectEntity} where #{#n1ql.filter} and meta().id = $1 and ARRAY_CONTAINS(securityRoles, $2)\")\r\n    fun hasRole(userId: String, role: String): User\r\n}<\/pre>\n<p>&nbsp;<\/p>\n<ul>\n<li><strong>@N1qlPrimaryIndexed<\/strong>: Este\u00a0<a href=\"https:\/\/docs.spring.io\/spring-data\/couchbase\/docs\/current\/api\/index.html?org\/springframework\/data\/couchbase\/core\/query\/N1qlPrimaryIndexed.html\">anota\u00e7\u00e3o\u00a0<\/a>garante que o bucket associado ao reposit\u00f3rio atual ter\u00e1 um \u00edndice prim\u00e1rio N1QL<\/li>\n<li><strong>@ViewIndexed: \u00a0<\/strong>Isso\u00a0<a href=\"https:\/\/docs.spring.io\/spring-data\/couchbase\/docs\/current\/api\/index.html?org\/springframework\/data\/couchbase\/core\/query\/ViewIndexed.html\">anota\u00e7\u00e3o\u00a0<\/a>permite que voc\u00ea defina o nome do documento de design e o nome da visualiza\u00e7\u00e3o, bem como um mapa personalizado e uma fun\u00e7\u00e3o de redu\u00e7\u00e3o.<\/li>\n<\/ul>\n<p>Como voc\u00ea pode ver abaixo, \u00e9 poss\u00edvel aproveitar todos os\u00a0<a href=\"https:\/\/docs.spring.io\/spring-data\/couchbase\/docs\/current\/reference\/html\/#couchbase.repository.querying\">Palavras-chave do Spring Data<\/a>\u00a0para consultar o banco de dados, como\u00a0<strong><em>Encontrar<\/em><\/strong>,\u00a0<strong><em>Entre<\/em><\/strong>,\u00a0<strong><em>IsGreaterThan<\/em><\/strong>,\u00a0 <strong><em>Como<\/em><\/strong>, <strong><em>Existe<\/em><\/strong>etc.<\/p>\n<pre class=\"lang:default decode:true\">    fun findByName(name: String): List&lt;User&gt;<\/pre>\n<p>O reposit\u00f3rio est\u00e1 ampliando\u00a0<strong><em>CouchbasePagingAndSortingRepository<\/em><\/strong>que permite que voc\u00ea pagine suas consultas simplesmente adicionando um\u00a0<strong><em>Pagin\u00e1vel<\/em><\/strong>\u00a0param no final da defini\u00e7\u00e3o do m\u00e9todo. Se precisar escrever consultas mais avan\u00e7adas, voc\u00ea tamb\u00e9m pode usar o N1QL:<\/p>\n<pre class=\"lang:default decode:true\">    @Query(\"#{#n1ql.selectEntity} where #{#n1ql.filter} and ANY preference IN \" + \" preferences SATISFIES preference.name = $1 END\")\r\n    fun findUsersByPreferenceName(name: String): List&lt;User&gt;\r\n\r\n    @Query(\"#{#n1ql.selectEntity} where #{#n1ql.filter} and meta().id = $1 and ARRAY_CONTAINS(securityRoles, $2)\")\r\n    fun hasRole(userId: String, role: String): User<\/pre>\n<p>As consultas acima t\u00eam algumas sugest\u00f5es de sintaxe para torn\u00e1-las menores:<\/p>\n<ul>\n<li><strong>#(#n1ql.bucket):\u00a0<\/strong><span style=\"font-style: inherit\">O uso desta sintaxe evita a codifica\u00e7\u00e3o do nome do bucket em sua consulta<\/span><\/li>\n<li><strong>#{#n1ql.selectEntity}:\u00a0<\/strong>a\u00e7\u00facar-sintaxe para\u00a0<strong><em>SELECT * FROM #(#n1ql.bucket)<\/em><\/strong><strong>:<\/strong><\/li>\n<li><strong style=\"font-style: inherit\">#{#n1ql.filter}:\u00a0<\/strong><span style=\"font-style: inherit\">syntax-sugar para filtrar o documento por tipo, o que tecnicamente significa\u00a0<\/span><strong style=\"font-style: inherit\"><em>class = 'myPackage.MyClassName'<\/em><\/strong><em><span style=\"font-style: inherit\">\u00a0(<\/span><strong style=\"font-style: inherit\"><span style=\"font-style: inherit\">_classe<\/span><\/strong><\/em>\u00a0\u00e9 o atributo adicionado automaticamente ao documento para definir seu tipo quando voc\u00ea est\u00e1 trabalhando com o Couchbase no Spring Data )<\/li>\n<li><strong>#{#n1ql.fields}\u00a0<\/strong>ser\u00e1 substitu\u00eddo pela lista de campos (por exemplo, para uma cl\u00e1usula SELECT) necess\u00e1ria para reconstruir a entidade.<\/li>\n<li><strong>#{#n1ql.delete}\u00a0<\/strong><span style=\"font-style: inherit\">ser\u00e1 substitu\u00eddo pela instru\u00e7\u00e3o delete from.<\/span><\/li>\n<li><strong>#{#n1ql.returning}\u00a0<\/strong><span style=\"font-style: inherit\">ser\u00e1 substitu\u00eddo pela cl\u00e1usula de retorno necess\u00e1ria para a reconstru\u00e7\u00e3o da entidade.<\/span><\/li>\n<\/ul>\n<h2><strong>Servi\u00e7os<\/strong><\/h2>\n<p>Nosso servi\u00e7o basicamente encaminha solicita\u00e7\u00f5es para o nosso reposit\u00f3rio, mas se voc\u00ea precisar escrever consultas ad-hoc, este \u00e9 o lugar certo:<\/p>\n<pre class=\"lang:default decode:true\">@Service\r\nclass UserService {\r\n\r\n    @Autowired\r\n    lateinit var userRepository: UserRepository;\r\n\r\n    fun findByName(name: String): List&lt;User&gt; = userRepository.findByName(name)\r\n\r\n    fun findById(userId: String) = userRepository.findOne(userId)\r\n\r\n    fun save(@Valid user: User) = userRepository.save(user)\r\n\r\n    fun findUsersByPreferenceName(name: String): List&lt;User&gt; = userRepository.findUsersByPreferenceName(name)\r\n\r\n    fun hasRole(userId: String, role: String): Boolean {\r\n        return userRepository.hasRole(userId, role) != null\r\n    }\r\n\r\n    \/**\r\n     * Example of ad hoc queries\r\n     *\/\r\n    fun findUserByAddress(streetName: String = \"\", number: String = \"\", postalCode: String = \"\",\r\n                          city: String = \"\", country: String = \"\"): List&lt;User&gt; {\r\n\r\n        var query = \"SELECT meta(b).id as id, b.* FROM \" + getBucketName() + \" b WHERE  b._class = '\" + User::class.java.getName() + \"' \"\r\n\r\n        if (!streetName.isNullOrBlank()) query += \" and b.address.streetName = '$streetName' \"\r\n\r\n        if (!number.isNullOrBlank()) query += \" and b.address.houseNumber = '$number' \"\r\n\r\n        if (!postalCode.isNullOrBlank()) query += \" and b.address.postalCode = '$postalCode' \"\r\n\r\n        if (!city.isNullOrBlank()) query += \" and b.address.city = '$city' \"\r\n\r\n        if (!country.isNullOrBlank()) query += \" and b.address.country = '$country' \"\r\n\r\n        val params = N1qlParams.build().consistency(ScanConsistency.REQUEST_PLUS).adhoc(true)\r\n        val paramQuery = N1qlQuery.parameterized(query, JsonObject.create(), params)\r\n        return userRepository.getCouchbaseOperations().findByN1QLProjection(paramQuery, User::class.java)\r\n    }\r\n\r\n    fun getBucketName() = userRepository.getCouchbaseOperations().getCouchbaseBucket().bucketManager().info().name()\r\n}<\/pre>\n<h2><strong>Controladores<\/strong><\/h2>\n<p>Por fim, vamos adicionar tamb\u00e9m um controlador para testar nossos servi\u00e7os via rest:<\/p>\n<pre class=\"lang:default decode:true\">@RestController\r\n@RequestMapping(\"\/api\/user\")\r\nclass UserController {\r\n\r\n    @Autowired\r\n    lateinit var userService: UserService\r\n\r\n    @GetMapping(value = \"\/{id}\")\r\n    fun findById(@PathParam(\"id\") id: String) = userService.findById(id)\r\n\r\n\r\n    @GetMapping(value = \"\/preference\")\r\n    fun findPreference(@RequestParam(\"name\") name: String): List&lt;User&gt; {\r\n        return userService.findUsersByPreferenceName(name)\r\n    }\r\n\r\n    @GetMapping(value = \"\/find\")\r\n    fun findUserByName(@RequestParam(\"name\") name: String): List&lt;User&gt; {\r\n        return userService.findByName(name)\r\n    }\r\n\r\n    @PostMapping(value = \"\/save\")\r\n    fun findUserByName(@RequestBody user: User) = userService.save(user)\r\n\r\n    @GetMapping(value = \"\/findByAddress\")\r\n    fun findByAddress(@RequestParam(\"streetName\", defaultValue = \"\") streetName: String,\r\n                      @RequestParam(\"number\", defaultValue = \"\") number: String,\r\n                      @RequestParam(\"postalCode\", defaultValue = \"\") postalCode: String,\r\n                      @RequestParam(\"city\", defaultValue = \"\") city: String,\r\n                      @RequestParam(\"country\", defaultValue = \"\") country: String): List&lt;User&gt; {\r\n        return userService.findUserByAddress(streetName, number, postalCode, city, country);\r\n    }\r\n\r\n}<\/pre>\n<h2><strong>Escrevendo testes de integra\u00e7\u00e3o com Kotlin<\/strong><\/h2>\n<p>Para executar os testes de integra\u00e7\u00e3o corretamente, n\u00e3o se esque\u00e7a de configurar as credenciais de seu banco de dados na se\u00e7\u00e3o <strong>application.properties<\/strong> file:<\/p>\n<pre class=\"lang:default decode:true\">spring.couchbase.bootstrap-hosts=localhost\r\nspring.couchbase.bucket.name=test\r\nspring.couchbase.bucket.password=somePassword\r\nspring.data.couchbase.auto-index=true<\/pre>\n<p>Aqui, voc\u00ea pode ver a apar\u00eancia de nossos testes:<\/p>\n<pre class=\"lang:default decode:true\">@Test\r\n    fun testComposedAddress() {\r\n        val address1 = Address(\"street1\", \"1\", \"0000\", \"santo andre\", \"br\")\r\n        val address2 = Address(\"street1\", \"2\", \"0000\", \"santo andre\", \"br\")\r\n        val address3 = Address(\"street2\", \"12\", \"1111\", \"munich\", \"de\")\r\n\r\n        userService.save(User(USER_1, \"user1\", address1, emptyList(), emptyList()))\r\n        userService.save(User(\"user::2\", \"user2\", address2, emptyList(), emptyList()))\r\n        userService.save(User(\"user::3\", \"user3\", address3, emptyList(), emptyList()))\r\n\r\n        var users = userService.findUserByAddress(streetName = \"street1\")\r\n        assertThat(users, hasSize&lt;Any&gt;(2))\r\n\r\n        users = userService.findUserByAddress(streetName = \"street1\", number=  \"1\")\r\n        assertThat(users, hasSize&lt;Any&gt;(1))\r\n\r\n        users = userService.findUserByAddress(country = \"de\")\r\n        assertThat(users, hasSize&lt;Any&gt;(1))\r\n    }<\/pre>\n<h3><strong>Depend\u00eancias de Kotlin e Maven<\/strong><\/h3>\n<p>O Kotlin est\u00e1 evoluindo rapidamente, portanto, lembre-se de usar as vers\u00f5es mais recentes de cada depend\u00eancia:<\/p>\n<pre class=\"lang:xhtml decode:true\">    &lt;dependencies&gt;\r\n        &lt;dependency&gt;\r\n            &lt;groupId&gt;org.jetbrains.kotlin&lt;\/groupId&gt;\r\n            &lt;artifactId&gt;kotlin-stdlib-jdk8&lt;\/artifactId&gt;\r\n            &lt;version&gt;1.2.41&lt;\/version&gt;\r\n        &lt;\/dependency&gt;\r\n        &lt;dependency&gt;\r\n            &lt;groupId&gt;org.jetbrains.kotlin&lt;\/groupId&gt;\r\n            &lt;artifactId&gt;kotlin-reflect&lt;\/artifactId&gt;\r\n            &lt;version&gt;1.2.41&lt;\/version&gt;\r\n        &lt;\/dependency&gt;\r\n        &lt;dependency&gt;\r\n            &lt;groupId&gt;com.fasterxml.jackson.module&lt;\/groupId&gt;\r\n            &lt;artifactId&gt;jackson-module-kotlin&lt;\/artifactId&gt;\r\n            &lt;version&gt;2.9.5&lt;\/version&gt;\r\n        &lt;\/dependency&gt;\r\n        &lt;dependency&gt;\r\n            &lt;groupId&gt;org.springframework.boot&lt;\/groupId&gt;\r\n            &lt;artifactId&gt;spring-boot-starter-data-couchbase&lt;\/artifactId&gt;\r\n        &lt;\/dependency&gt;\r\n        &lt;dependency&gt;\r\n            &lt;groupId&gt;org.springframework.boot&lt;\/groupId&gt;\r\n            &lt;artifactId&gt;spring-boot-starter-data-rest&lt;\/artifactId&gt;\r\n        &lt;\/dependency&gt;\r\n        &lt;dependency&gt;\r\n            &lt;groupId&gt;org.springframework.boot&lt;\/groupId&gt;\r\n            &lt;artifactId&gt;spring-boot-starter-test&lt;\/artifactId&gt;\r\n            &lt;scope&gt;test&lt;\/scope&gt;\r\n        &lt;\/dependency&gt;\r\n        &lt;dependency&gt;\r\n            &lt;groupId&gt;org.hamcrest&lt;\/groupId&gt;\r\n            &lt;artifactId&gt;hamcrest-library&lt;\/artifactId&gt;\r\n            &lt;version&gt;1.3&lt;\/version&gt;\r\n        &lt;\/dependency&gt;\r\n        &lt;dependency&gt;\r\n            &lt;groupId&gt;org.springframework.data&lt;\/groupId&gt;\r\n            &lt;artifactId&gt;spring-data-couchbase&lt;\/artifactId&gt;\r\n        &lt;\/dependency&gt;\r\n        &lt;dependency&gt;\r\n            &lt;groupId&gt;org.jetbrains.kotlin&lt;\/groupId&gt;\r\n            &lt;artifactId&gt;kotlin-stdlib-jdk8&lt;\/artifactId&gt;\r\n            &lt;version&gt;${kotlin.version}&lt;\/version&gt;\r\n        &lt;\/dependency&gt;\r\n        &lt;dependency&gt;\r\n            &lt;groupId&gt;org.jetbrains.kotlin&lt;\/groupId&gt;\r\n            &lt;artifactId&gt;kotlin-test&lt;\/artifactId&gt;\r\n            &lt;version&gt;${kotlin.version}&lt;\/version&gt;\r\n            &lt;scope&gt;test&lt;\/scope&gt;\r\n        &lt;\/dependency&gt;\r\n        &lt;dependency&gt;\r\n            &lt;groupId&gt;org.jetbrains.kotlin&lt;\/groupId&gt;\r\n            &lt;artifactId&gt;kotlin-maven-allopen&lt;\/artifactId&gt;\r\n            &lt;version&gt;1.2.41&lt;\/version&gt;\r\n        &lt;\/dependency&gt;\r\n    &lt;\/dependencies&gt;<\/pre>\n<p>Voc\u00ea pode visualizar todo o <strong>pom.xml<\/strong> <a href=\"https:\/\/github.com\/couchbaselabs\/try-cb-kotlin\/blob\/master\/pom.xml\">aqui<\/a>.<\/p>","protected":false},"excerpt":{"rendered":"<p>Last year I started learning Kotlin and I was surprised at how easy it was to convert a Java application. IntelliJ and a few other IDEs offer nice tools for automatic conversion, and with a few adjustments you can end [&hellip;]<\/p>","protected":false},"author":8754,"featured_media":13873,"comment_status":"open","ping_status":"open","sticky":false,"template":"","format":"standard","meta":{"_acf_changed":false,"inline_featured_image":false,"footnotes":""},"categories":[1814,1815],"tags":[],"ppma_author":[9059],"class_list":["post-5267","post","type-post","status-publish","format-standard","has-post-thumbnail","hentry","category-application-design","category-best-practices-and-tutorials"],"acf":[],"yoast_head":"<!-- This site is optimized with the Yoast SEO Premium plugin v25.7.1 (Yoast SEO v25.7) - https:\/\/yoast.com\/wordpress\/plugins\/seo\/ -->\n<title>Couchbase with Kotlin, Spring Boot and Spring Data<\/title>\n<meta name=\"description\" content=\"See how easy it is to convert a Java application. Check out how to create a sample application using: Kotlin, Spring Boot, Spring Data, and Couchbase.\" \/>\n<meta name=\"robots\" content=\"index, follow, max-snippet:-1, max-image-preview:large, max-video-preview:-1\" \/>\n<link rel=\"canonical\" href=\"https:\/\/www.couchbase.com\/blog\/pt\/kotlin-spring-boot-spring-data\/\" \/>\n<meta property=\"og:locale\" content=\"pt_BR\" \/>\n<meta property=\"og:type\" content=\"article\" \/>\n<meta property=\"og:title\" content=\"Couchbase with Kotlin, Spring Boot and Spring Data\" \/>\n<meta property=\"og:description\" content=\"See how easy it is to convert a Java application. Check out how to create a sample application using: Kotlin, Spring Boot, Spring Data, and Couchbase.\" \/>\n<meta property=\"og:url\" content=\"https:\/\/www.couchbase.com\/blog\/pt\/kotlin-spring-boot-spring-data\/\" \/>\n<meta property=\"og:site_name\" content=\"The Couchbase Blog\" \/>\n<meta property=\"article:published_time\" content=\"2018-06-11T10:01:59+00:00\" \/>\n<meta property=\"article:modified_time\" content=\"2023-07-20T21:48:34+00:00\" \/>\n<meta property=\"og:image\" content=\"https:\/\/www.couchbase.com\/blog\/wp-content\/uploads\/2018\/06\/Screen-Shot-2018-06-04-at-10.34.39-PM-1024x252.png\" \/>\n<meta name=\"author\" content=\"Denis Rosa, Developer Advocate, Couchbase\" \/>\n<meta name=\"twitter:card\" content=\"summary_large_image\" \/>\n<meta name=\"twitter:creator\" content=\"@deniswsrosa\" \/>\n<meta name=\"twitter:label1\" content=\"Written by\" \/>\n\t<meta name=\"twitter:data1\" content=\"Denis Rosa, Developer Advocate, Couchbase\" \/>\n\t<meta name=\"twitter:label2\" content=\"Est. reading time\" \/>\n\t<meta name=\"twitter:data2\" content=\"3 minutos\" \/>\n<script type=\"application\/ld+json\" class=\"yoast-schema-graph\">{\"@context\":\"https:\/\/schema.org\",\"@graph\":[{\"@type\":\"Article\",\"@id\":\"https:\/\/www.couchbase.com\/blog\/kotlin-spring-boot-spring-data\/#article\",\"isPartOf\":{\"@id\":\"https:\/\/www.couchbase.com\/blog\/kotlin-spring-boot-spring-data\/\"},\"author\":{\"name\":\"Denis Rosa, Developer Advocate, Couchbase\",\"@id\":\"https:\/\/www.couchbase.com\/blog\/#\/schema\/person\/fe3c5273e805e72a5294611a48f62257\"},\"headline\":\"Couchbase with Kotlin, Spring Boot and Spring Data\",\"datePublished\":\"2018-06-11T10:01:59+00:00\",\"dateModified\":\"2023-07-20T21:48:34+00:00\",\"mainEntityOfPage\":{\"@id\":\"https:\/\/www.couchbase.com\/blog\/kotlin-spring-boot-spring-data\/\"},\"wordCount\":602,\"commentCount\":0,\"publisher\":{\"@id\":\"https:\/\/www.couchbase.com\/blog\/#organization\"},\"image\":{\"@id\":\"https:\/\/www.couchbase.com\/blog\/kotlin-spring-boot-spring-data\/#primaryimage\"},\"thumbnailUrl\":\"https:\/\/www.couchbase.com\/blog\/wp-content\/uploads\/sites\/1\/2022\/11\/couchbase-nosql-dbaas.png\",\"articleSection\":[\"Application Design\",\"Best Practices and Tutorials\"],\"inLanguage\":\"pt-BR\",\"potentialAction\":[{\"@type\":\"CommentAction\",\"name\":\"Comment\",\"target\":[\"https:\/\/www.couchbase.com\/blog\/kotlin-spring-boot-spring-data\/#respond\"]}]},{\"@type\":\"WebPage\",\"@id\":\"https:\/\/www.couchbase.com\/blog\/kotlin-spring-boot-spring-data\/\",\"url\":\"https:\/\/www.couchbase.com\/blog\/kotlin-spring-boot-spring-data\/\",\"name\":\"Couchbase with Kotlin, Spring Boot and Spring Data\",\"isPartOf\":{\"@id\":\"https:\/\/www.couchbase.com\/blog\/#website\"},\"primaryImageOfPage\":{\"@id\":\"https:\/\/www.couchbase.com\/blog\/kotlin-spring-boot-spring-data\/#primaryimage\"},\"image\":{\"@id\":\"https:\/\/www.couchbase.com\/blog\/kotlin-spring-boot-spring-data\/#primaryimage\"},\"thumbnailUrl\":\"https:\/\/www.couchbase.com\/blog\/wp-content\/uploads\/sites\/1\/2022\/11\/couchbase-nosql-dbaas.png\",\"datePublished\":\"2018-06-11T10:01:59+00:00\",\"dateModified\":\"2023-07-20T21:48:34+00:00\",\"description\":\"See how easy it is to convert a Java application. Check out how to create a sample application using: Kotlin, Spring Boot, Spring Data, and Couchbase.\",\"breadcrumb\":{\"@id\":\"https:\/\/www.couchbase.com\/blog\/kotlin-spring-boot-spring-data\/#breadcrumb\"},\"inLanguage\":\"pt-BR\",\"potentialAction\":[{\"@type\":\"ReadAction\",\"target\":[\"https:\/\/www.couchbase.com\/blog\/kotlin-spring-boot-spring-data\/\"]}]},{\"@type\":\"ImageObject\",\"inLanguage\":\"pt-BR\",\"@id\":\"https:\/\/www.couchbase.com\/blog\/kotlin-spring-boot-spring-data\/#primaryimage\",\"url\":\"https:\/\/www.couchbase.com\/blog\/wp-content\/uploads\/sites\/1\/2022\/11\/couchbase-nosql-dbaas.png\",\"contentUrl\":\"https:\/\/www.couchbase.com\/blog\/wp-content\/uploads\/sites\/1\/2022\/11\/couchbase-nosql-dbaas.png\",\"width\":1800,\"height\":630},{\"@type\":\"BreadcrumbList\",\"@id\":\"https:\/\/www.couchbase.com\/blog\/kotlin-spring-boot-spring-data\/#breadcrumb\",\"itemListElement\":[{\"@type\":\"ListItem\",\"position\":1,\"name\":\"Home\",\"item\":\"https:\/\/www.couchbase.com\/blog\/\"},{\"@type\":\"ListItem\",\"position\":2,\"name\":\"Couchbase with Kotlin, Spring Boot and Spring Data\"}]},{\"@type\":\"WebSite\",\"@id\":\"https:\/\/www.couchbase.com\/blog\/#website\",\"url\":\"https:\/\/www.couchbase.com\/blog\/\",\"name\":\"The Couchbase Blog\",\"description\":\"Couchbase, the NoSQL Database\",\"publisher\":{\"@id\":\"https:\/\/www.couchbase.com\/blog\/#organization\"},\"potentialAction\":[{\"@type\":\"SearchAction\",\"target\":{\"@type\":\"EntryPoint\",\"urlTemplate\":\"https:\/\/www.couchbase.com\/blog\/?s={search_term_string}\"},\"query-input\":{\"@type\":\"PropertyValueSpecification\",\"valueRequired\":true,\"valueName\":\"search_term_string\"}}],\"inLanguage\":\"pt-BR\"},{\"@type\":\"Organization\",\"@id\":\"https:\/\/www.couchbase.com\/blog\/#organization\",\"name\":\"The Couchbase Blog\",\"url\":\"https:\/\/www.couchbase.com\/blog\/\",\"logo\":{\"@type\":\"ImageObject\",\"inLanguage\":\"pt-BR\",\"@id\":\"https:\/\/www.couchbase.com\/blog\/#\/schema\/logo\/image\/\",\"url\":\"https:\/\/www.couchbase.com\/blog\/wp-content\/uploads\/2023\/04\/admin-logo.png\",\"contentUrl\":\"https:\/\/www.couchbase.com\/blog\/wp-content\/uploads\/2023\/04\/admin-logo.png\",\"width\":218,\"height\":34,\"caption\":\"The Couchbase Blog\"},\"image\":{\"@id\":\"https:\/\/www.couchbase.com\/blog\/#\/schema\/logo\/image\/\"}},{\"@type\":\"Person\",\"@id\":\"https:\/\/www.couchbase.com\/blog\/#\/schema\/person\/fe3c5273e805e72a5294611a48f62257\",\"name\":\"Denis Rosa, Developer Advocate, Couchbase\",\"image\":{\"@type\":\"ImageObject\",\"inLanguage\":\"pt-BR\",\"@id\":\"https:\/\/www.couchbase.com\/blog\/#\/schema\/person\/image\/be0716f6199cfb09417c92cf7a8fa8d6\",\"url\":\"https:\/\/secure.gravatar.com\/avatar\/f8d1f5c13115122cab89d0f229b904480bfe20d3dfbb093fe9734cda5235d419?s=96&d=mm&r=g\",\"contentUrl\":\"https:\/\/secure.gravatar.com\/avatar\/f8d1f5c13115122cab89d0f229b904480bfe20d3dfbb093fe9734cda5235d419?s=96&d=mm&r=g\",\"caption\":\"Denis Rosa, Developer Advocate, Couchbase\"},\"description\":\"Denis Rosa is a Developer Advocate for Couchbase and lives in Munich - Germany. He has a solid experience as a software engineer and speaks fluently Java, Python, Scala and Javascript. Denis likes to write about search, Big Data, AI, Microservices and everything else that would help developers to make a beautiful, faster, stable and scalable app.\",\"sameAs\":[\"https:\/\/x.com\/deniswsrosa\"],\"url\":\"https:\/\/www.couchbase.com\/blog\/pt\/author\/denis-rosa\/\"}]}<\/script>\n<!-- \/ Yoast SEO Premium plugin. -->","yoast_head_json":{"title":"Couchbase com Kotlin, Spring Boot e Spring Data","description":"See how easy it is to convert a Java application. Check out how to create a sample application using: Kotlin, Spring Boot, Spring Data, and Couchbase.","robots":{"index":"index","follow":"follow","max-snippet":"max-snippet:-1","max-image-preview":"max-image-preview:large","max-video-preview":"max-video-preview:-1"},"canonical":"https:\/\/www.couchbase.com\/blog\/pt\/kotlin-spring-boot-spring-data\/","og_locale":"pt_BR","og_type":"article","og_title":"Couchbase with Kotlin, Spring Boot and Spring Data","og_description":"See how easy it is to convert a Java application. Check out how to create a sample application using: Kotlin, Spring Boot, Spring Data, and Couchbase.","og_url":"https:\/\/www.couchbase.com\/blog\/pt\/kotlin-spring-boot-spring-data\/","og_site_name":"The Couchbase Blog","article_published_time":"2018-06-11T10:01:59+00:00","article_modified_time":"2023-07-20T21:48:34+00:00","og_image":[{"url":"https:\/\/www.couchbase.com\/blog\/wp-content\/uploads\/2018\/06\/Screen-Shot-2018-06-04-at-10.34.39-PM-1024x252.png","type":"","width":"","height":""}],"author":"Denis Rosa, Developer Advocate, Couchbase","twitter_card":"summary_large_image","twitter_creator":"@deniswsrosa","twitter_misc":{"Written by":"Denis Rosa, Developer Advocate, Couchbase","Est. reading time":"3 minutos"},"schema":{"@context":"https:\/\/schema.org","@graph":[{"@type":"Article","@id":"https:\/\/www.couchbase.com\/blog\/kotlin-spring-boot-spring-data\/#article","isPartOf":{"@id":"https:\/\/www.couchbase.com\/blog\/kotlin-spring-boot-spring-data\/"},"author":{"name":"Denis Rosa, Developer Advocate, Couchbase","@id":"https:\/\/www.couchbase.com\/blog\/#\/schema\/person\/fe3c5273e805e72a5294611a48f62257"},"headline":"Couchbase with Kotlin, Spring Boot and Spring Data","datePublished":"2018-06-11T10:01:59+00:00","dateModified":"2023-07-20T21:48:34+00:00","mainEntityOfPage":{"@id":"https:\/\/www.couchbase.com\/blog\/kotlin-spring-boot-spring-data\/"},"wordCount":602,"commentCount":0,"publisher":{"@id":"https:\/\/www.couchbase.com\/blog\/#organization"},"image":{"@id":"https:\/\/www.couchbase.com\/blog\/kotlin-spring-boot-spring-data\/#primaryimage"},"thumbnailUrl":"https:\/\/www.couchbase.com\/blog\/wp-content\/uploads\/sites\/1\/2022\/11\/couchbase-nosql-dbaas.png","articleSection":["Application Design","Best Practices and Tutorials"],"inLanguage":"pt-BR","potentialAction":[{"@type":"CommentAction","name":"Comment","target":["https:\/\/www.couchbase.com\/blog\/kotlin-spring-boot-spring-data\/#respond"]}]},{"@type":"WebPage","@id":"https:\/\/www.couchbase.com\/blog\/kotlin-spring-boot-spring-data\/","url":"https:\/\/www.couchbase.com\/blog\/kotlin-spring-boot-spring-data\/","name":"Couchbase com Kotlin, Spring Boot e Spring Data","isPartOf":{"@id":"https:\/\/www.couchbase.com\/blog\/#website"},"primaryImageOfPage":{"@id":"https:\/\/www.couchbase.com\/blog\/kotlin-spring-boot-spring-data\/#primaryimage"},"image":{"@id":"https:\/\/www.couchbase.com\/blog\/kotlin-spring-boot-spring-data\/#primaryimage"},"thumbnailUrl":"https:\/\/www.couchbase.com\/blog\/wp-content\/uploads\/sites\/1\/2022\/11\/couchbase-nosql-dbaas.png","datePublished":"2018-06-11T10:01:59+00:00","dateModified":"2023-07-20T21:48:34+00:00","description":"See how easy it is to convert a Java application. Check out how to create a sample application using: Kotlin, Spring Boot, Spring Data, and Couchbase.","breadcrumb":{"@id":"https:\/\/www.couchbase.com\/blog\/kotlin-spring-boot-spring-data\/#breadcrumb"},"inLanguage":"pt-BR","potentialAction":[{"@type":"ReadAction","target":["https:\/\/www.couchbase.com\/blog\/kotlin-spring-boot-spring-data\/"]}]},{"@type":"ImageObject","inLanguage":"pt-BR","@id":"https:\/\/www.couchbase.com\/blog\/kotlin-spring-boot-spring-data\/#primaryimage","url":"https:\/\/www.couchbase.com\/blog\/wp-content\/uploads\/sites\/1\/2022\/11\/couchbase-nosql-dbaas.png","contentUrl":"https:\/\/www.couchbase.com\/blog\/wp-content\/uploads\/sites\/1\/2022\/11\/couchbase-nosql-dbaas.png","width":1800,"height":630},{"@type":"BreadcrumbList","@id":"https:\/\/www.couchbase.com\/blog\/kotlin-spring-boot-spring-data\/#breadcrumb","itemListElement":[{"@type":"ListItem","position":1,"name":"Home","item":"https:\/\/www.couchbase.com\/blog\/"},{"@type":"ListItem","position":2,"name":"Couchbase with Kotlin, Spring Boot and Spring Data"}]},{"@type":"WebSite","@id":"https:\/\/www.couchbase.com\/blog\/#website","url":"https:\/\/www.couchbase.com\/blog\/","name":"Blog do Couchbase","description":"Couchbase, o banco de dados NoSQL","publisher":{"@id":"https:\/\/www.couchbase.com\/blog\/#organization"},"potentialAction":[{"@type":"SearchAction","target":{"@type":"EntryPoint","urlTemplate":"https:\/\/www.couchbase.com\/blog\/?s={search_term_string}"},"query-input":{"@type":"PropertyValueSpecification","valueRequired":true,"valueName":"search_term_string"}}],"inLanguage":"pt-BR"},{"@type":"Organization","@id":"https:\/\/www.couchbase.com\/blog\/#organization","name":"Blog do Couchbase","url":"https:\/\/www.couchbase.com\/blog\/","logo":{"@type":"ImageObject","inLanguage":"pt-BR","@id":"https:\/\/www.couchbase.com\/blog\/#\/schema\/logo\/image\/","url":"https:\/\/www.couchbase.com\/blog\/wp-content\/uploads\/2023\/04\/admin-logo.png","contentUrl":"https:\/\/www.couchbase.com\/blog\/wp-content\/uploads\/2023\/04\/admin-logo.png","width":218,"height":34,"caption":"The Couchbase Blog"},"image":{"@id":"https:\/\/www.couchbase.com\/blog\/#\/schema\/logo\/image\/"}},{"@type":"Person","@id":"https:\/\/www.couchbase.com\/blog\/#\/schema\/person\/fe3c5273e805e72a5294611a48f62257","name":"Denis Rosa, defensor dos desenvolvedores, Couchbase","image":{"@type":"ImageObject","inLanguage":"pt-BR","@id":"https:\/\/www.couchbase.com\/blog\/#\/schema\/person\/image\/be0716f6199cfb09417c92cf7a8fa8d6","url":"https:\/\/secure.gravatar.com\/avatar\/f8d1f5c13115122cab89d0f229b904480bfe20d3dfbb093fe9734cda5235d419?s=96&d=mm&r=g","contentUrl":"https:\/\/secure.gravatar.com\/avatar\/f8d1f5c13115122cab89d0f229b904480bfe20d3dfbb093fe9734cda5235d419?s=96&d=mm&r=g","caption":"Denis Rosa, Developer Advocate, Couchbase"},"description":"Denis Rosa \u00e9 um Developer Advocate do Couchbase e mora em Munique, na Alemanha. Ele tem uma s\u00f3lida experi\u00eancia como engenheiro de software e fala fluentemente Java, Python, Scala e Javascript. Denis gosta de escrever sobre pesquisa, Big Data, IA, microsservi\u00e7os e tudo o mais que possa ajudar os desenvolvedores a criar um aplicativo bonito, mais r\u00e1pido, est\u00e1vel e escal\u00e1vel.","sameAs":["https:\/\/x.com\/deniswsrosa"],"url":"https:\/\/www.couchbase.com\/blog\/pt\/author\/denis-rosa\/"}]}},"authors":[{"term_id":9059,"user_id":8754,"is_guest":0,"slug":"denis-rosa","display_name":"Denis Rosa, Developer Advocate, Couchbase","avatar_url":"https:\/\/secure.gravatar.com\/avatar\/f8d1f5c13115122cab89d0f229b904480bfe20d3dfbb093fe9734cda5235d419?s=96&d=mm&r=g","first_name":"Denis","last_name":"Rosa, Developer Advocate, Couchbase","user_url":"","author_category":"","description":"Denis Rosa \u00e9 um Developer Advocate do Couchbase e mora em Munique, na Alemanha. Ele tem uma s\u00f3lida experi\u00eancia como engenheiro de software e fala fluentemente Java, Python, Scala e Javascript. Denis gosta de escrever sobre pesquisa, Big Data, IA, microsservi\u00e7os e tudo o mais que possa ajudar os desenvolvedores a criar um aplicativo bonito, mais r\u00e1pido, est\u00e1vel e escal\u00e1vel."}],"_links":{"self":[{"href":"https:\/\/www.couchbase.com\/blog\/pt\/wp-json\/wp\/v2\/posts\/5267","targetHints":{"allow":["GET"]}}],"collection":[{"href":"https:\/\/www.couchbase.com\/blog\/pt\/wp-json\/wp\/v2\/posts"}],"about":[{"href":"https:\/\/www.couchbase.com\/blog\/pt\/wp-json\/wp\/v2\/types\/post"}],"author":[{"embeddable":true,"href":"https:\/\/www.couchbase.com\/blog\/pt\/wp-json\/wp\/v2\/users\/8754"}],"replies":[{"embeddable":true,"href":"https:\/\/www.couchbase.com\/blog\/pt\/wp-json\/wp\/v2\/comments?post=5267"}],"version-history":[{"count":0,"href":"https:\/\/www.couchbase.com\/blog\/pt\/wp-json\/wp\/v2\/posts\/5267\/revisions"}],"wp:featuredmedia":[{"embeddable":true,"href":"https:\/\/www.couchbase.com\/blog\/pt\/wp-json\/wp\/v2\/media\/13873"}],"wp:attachment":[{"href":"https:\/\/www.couchbase.com\/blog\/pt\/wp-json\/wp\/v2\/media?parent=5267"}],"wp:term":[{"taxonomy":"category","embeddable":true,"href":"https:\/\/www.couchbase.com\/blog\/pt\/wp-json\/wp\/v2\/categories?post=5267"},{"taxonomy":"post_tag","embeddable":true,"href":"https:\/\/www.couchbase.com\/blog\/pt\/wp-json\/wp\/v2\/tags?post=5267"},{"taxonomy":"author","embeddable":true,"href":"https:\/\/www.couchbase.com\/blog\/pt\/wp-json\/wp\/v2\/ppma_author?post=5267"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}