
com.github.mauricio.async.db.postgresql.column.PostgreSQLIntervalEncoderDecoder.scala Maven / Gradle / Ivy
/*
* Copyright 2013 Maurício Linhares
* Copyright 2013 Dylan Simon
*
* Maurício Linhares licenses this file to you 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.
*/
package com.github.mauricio.async.db.postgresql.column
import com.github.mauricio.async.db.column.ColumnEncoderDecoder
import com.github.mauricio.async.db.exceptions.DateEncoderNotAvailableException
import com.github.mauricio.async.db.util.Log
import org.joda.time.{Period, ReadablePeriod, ReadableDuration}
import org.joda.time.format.{ISOPeriodFormat, PeriodFormatterBuilder}
object PostgreSQLIntervalEncoderDecoder extends ColumnEncoderDecoder {
private val log = Log.getByName(this.getClass.getName)
/* Postgres accepts all ISO8601 formats. */
private val formatter = ISOPeriodFormat.standard
override def encode(value: Any): String = {
value match {
case t: ReadablePeriod => formatter.print(t)
case t: ReadableDuration => t.toString // defaults to ISO8601
case _ => throw new DateEncoderNotAvailableException(value)
}
}
/* these should only be used for parsing: */
private def postgresYMDBuilder(builder: PeriodFormatterBuilder) =
builder.appendYears
.appendSuffix(" year", " years")
.appendSeparator(" ")
.appendMonths
.appendSuffix(" mon", " mons")
.appendSeparator(" ")
.appendDays
.appendSuffix(" day", " days")
.appendSeparator(" ")
private val postgres_verboseParser =
postgresYMDBuilder(
new PeriodFormatterBuilder().appendLiteral("@ ")
).appendHours
.appendSuffix(" hour", " hours")
.appendSeparator(" ")
.appendMinutes
.appendSuffix(" min", " mins")
.appendSeparator(" ")
.appendSecondsWithOptionalMillis
.appendSuffix(" sec", " secs")
.toFormatter
private def postgresHMSBuilder(builder: PeriodFormatterBuilder) =
builder
// .printZeroAlways // really all-or-nothing
.rejectSignedValues(true) // XXX: sign should apply to all
.appendHours
.appendSuffix(":")
.appendMinutes
.appendSuffix(":")
.appendSecondsWithOptionalMillis
private val hmsParser =
postgresHMSBuilder(new PeriodFormatterBuilder()).toFormatter
private val postgresParser =
postgresHMSBuilder(
postgresYMDBuilder(new PeriodFormatterBuilder())
).toFormatter
/* These sql_standard parsers don't handle negative signs correctly. */
private def sqlDTBuilder(builder: PeriodFormatterBuilder) =
postgresHMSBuilder(builder.appendDays.appendSeparator(" "))
private val sqlDTParser =
sqlDTBuilder(new PeriodFormatterBuilder()).toFormatter
private val sqlParser =
sqlDTBuilder(
new PeriodFormatterBuilder().printZeroAlways
.rejectSignedValues(true) // XXX: sign should apply to both
.appendYears
.appendSeparator("-")
.appendMonths
.rejectSignedValues(false)
.printZeroNever
.appendSeparator(" ")
).toFormatter
/* This supports all positive intervals, and intervalstyle of postgres_verbose, and iso_8601 perfectly.
* If intervalstyle is set to postgres or sql_standard, some negative intervals may be rejected.
*/
def decode(value: String): Period = {
if (value.isEmpty)
/* huh? */
Period.ZERO
else {
val format =
(
if (value(0).equals('P'))
/* iso_8601 */
formatter
else if (value.startsWith("@ "))
postgres_verboseParser
else {
/* try to guess based on what comes after the first number */
val i = value.indexWhere(
!_.isDigit,
if ("-+".contains(value(0))) 1 else 0
)
if (i < 0 || ":.".contains(value(i)))
/* simple HMS (to support group negation) */
hmsParser
else if (value(i).equals('-'))
/* sql_standard: Y-M */
sqlParser
else if (
value(i).equals(' ') && i + 1 < value.length && value(
i + 1
).isDigit
)
/* sql_standard: D H:M:S */
sqlDTParser
else
postgresParser
}
)
if ((format eq hmsParser) && value(0).equals('-'))
format.parsePeriod(value.substring(1)).negated
else if (value.endsWith(" ago"))
/* only really applies to postgres_verbose, but shouldn't hurt */
format.parsePeriod(value.stripSuffix(" ago")).negated
else
format.parsePeriod(value)
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy