autoconfig-macros


License

License

Categories

Categories

Auto Application Layer Libs Code Generators config Configuration
GroupId

GroupId

io.methvin.play
ArtifactId

ArtifactId

autoconfig-macros_2.13
Last Version

Last Version

0.3.2
Release Date

Release Date

Type

Type

jar
Description

Description

autoconfig-macros
autoconfig-macros
Project URL

Project URL

https://github.com/gmethvin/play-autoconfig
Project Organization

Project Organization

Greg Methvin
Source Code Management

Source Code Management

https://github.com/gmethvin/play-autoconfig

Download autoconfig-macros_2.13

How to add to project

<!-- https://jarcasting.com/artifacts/io.methvin.play/autoconfig-macros_2.13/ -->
<dependency>
    <groupId>io.methvin.play</groupId>
    <artifactId>autoconfig-macros_2.13</artifactId>
    <version>0.3.2</version>
</dependency>
// https://jarcasting.com/artifacts/io.methvin.play/autoconfig-macros_2.13/
implementation 'io.methvin.play:autoconfig-macros_2.13:0.3.2'
// https://jarcasting.com/artifacts/io.methvin.play/autoconfig-macros_2.13/
implementation ("io.methvin.play:autoconfig-macros_2.13:0.3.2")
'io.methvin.play:autoconfig-macros_2.13:jar:0.3.2'
<dependency org="io.methvin.play" name="autoconfig-macros_2.13" rev="0.3.2">
  <artifact name="autoconfig-macros_2.13" type="jar" />
</dependency>
@Grapes(
@Grab(group='io.methvin.play', module='autoconfig-macros_2.13', version='0.3.2')
)
libraryDependencies += "io.methvin.play" % "autoconfig-macros_2.13" % "0.3.2"
[io.methvin.play/autoconfig-macros_2.13 "0.3.2"]

Dependencies

compile (3)

Group / Artifact Type Version
org.scala-lang : scala-library jar 2.13.0
com.typesafe.play : play_2.13 jar 2.7.3
org.scala-lang : scala-reflect jar 2.13.0

test (1)

Group / Artifact Type Version
org.scalatest : scalatest_2.13 jar 3.0.8

Project Modules

There are no modules declared in this project.

Play AutoConfig

Maven

AutoConfig is a utility for type-safe configuration in Play (2.6.0 and later). The library provides a convenient macro to generate ConfigLoader instances for arbitrary classes.

Usage

To add to your sbt build:

libraryDependencies += "io.methvin.play" %% "autoconfig-macros" % playAutoConfigVersion

Replace playAutoConfigVersion with the latest version: maven central version

Creating configuration classes

It's good practice to create type-safe classes representing your configuration instead of passing around raw Configuration or Config objects. This way your application components can simply access properties of your class rather than having to read and convert configuration individually. This also makes it easier to create instances for unit testing.

For example, suppose I want to read configuration for a hypothetical API I'm making calls to. Let's assume I need an API key, an API password, and I want to configure the request timeout. So I create a class like this:

import play.api._
import io.methvin.play.autoconfig._
import scala.concurrent.duration._

case class FooApiConfig(
  apiKey: String,
  apiPassword: String,
  requestTimeout: Duration
)
object FooApiConfig {
  implicit val loader: ConfigLoader[FooApiConfig] = AutoConfig.loader
  def fromConfiguration(conf: Configuration) = conf.get[FooApiConfig]("api.foo")
}

The fromConfiguration method shows how you'd use the ConfigLoader using the Configuration#get method. In this case it would read a configuration object that looks like this:

api.foo {
  apiKey = "abcdef"
  apiPassword = "secret"
  requestTimeout = 1 minute
}

The loader macro goes through each parameter of the default constructor and looks for a ConfigLoader in scope for that type, then uses the loader to load the key in the object of the same name. In this case Play already provides loaders for String and Duration. If you had a custom object type nested inside, you could generate a loader in the exact same way.

Custom naming

The macro also supports custom names, for example:

import play.api._
import io.methvin.play.autoconfig._
import scala.concurrent.duration._

case class FooApiConfig(
  @ConfigName("api-key") apiKey: String,
  @ConfigName("api-password") apiPassword: String,
  @ConfigName("request-timeout") requestTimeout: Duration
)
object FooApiConfig {
  implicit val loader: ConfigLoader[FooApiConfig] = AutoConfig.loader
  def fromConfiguration(conf: Configuration) = conf.get[FooApiConfig]("api.foo")
}

This will change the name of the properties used for each field:

api.foo {
  api-key = "abcdef"
  api-password = "secret"
  request-timeout = 1 minute
}

Using an alternate constructor

You can also use an alternate constructor:

import play.api._
import io.methvin.play.autoconfig._
import scala.concurrent.duration._

case class FooApiConfig(
  apiKey: String,
  apiPassword: String,
  requestTimeout: Duration
) {
  @ConfigConstructor def this(key: String, password: String, timeout: Int) = {
    this(key, password, duration.millis)
  } 
}

The field names will be taken from the argument names of the alternate constructor, or their associated @ConfigName annotation.

Binding configuration classes

Once you've written your config class, you'll need to make it available to the components that use it. If you're using compile-time DI, you can write something like this:

trait FooApiComponents {
  def configuration: Configuration
  lazy val fooApiConfig: FooApiConfig = configuration.get[FooApiConfig]("api.foo")
  lazy val fooApi: FooApi = new FooApi(fooApiConfig)
}

If using Guice you can add a @Provides method to your module with your config class:

class ConfigModule extends AbstractModule {
  def configure: Unit = { /* ... */ }

  @Provides def fooApiConfig(conf: Configuration): FooApiConfig =
    conf.get[FooApiConfig]("api.foo")
}

License

This library is licensed under the Apache License, version 2.0. See the LICENSE file for details.

Versions

Version
0.3.2
0.3.1
0.3.0