commit
8fe7a2c427
26
.github/workflows/ci.yaml
vendored
Normal file
26
.github/workflows/ci.yaml
vendored
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
name: CI
|
||||||
|
on: [ push ]
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
deploy:
|
||||||
|
env:
|
||||||
|
REPO: ${{ github.event.repository.name }}
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Check out repository code
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Setup JDK
|
||||||
|
uses: actions/setup-java@v4
|
||||||
|
with:
|
||||||
|
distribution: 'zulu'
|
||||||
|
java-version: '21'
|
||||||
|
|
||||||
|
- name: Build Runner
|
||||||
|
run: ./mvnw --no-transfer-progress clean verify -Pnative -Dquarkus.native.remote-container-build=true
|
||||||
|
|
||||||
|
- name: Build Container
|
||||||
|
run: docker build -f src/main/docker/Dockerfile.native-micro -t ${{ env.REPO }} .
|
||||||
|
|
||||||
|
- name: Deploy
|
||||||
|
run: "docker compose up -d"
|
9
.gitignore
vendored
Normal file
9
.gitignore
vendored
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
*~
|
||||||
|
|
||||||
|
searxng-docker.service
|
||||||
|
caddy
|
||||||
|
srv
|
||||||
|
searxng/uwsgi.ini
|
||||||
|
|
||||||
|
.idea
|
||||||
|
target
|
1
.mvn/wrapper/.gitignore
vendored
Normal file
1
.mvn/wrapper/.gitignore
vendored
Normal file
@ -0,0 +1 @@
|
|||||||
|
maven-wrapper.jar
|
93
.mvn/wrapper/MavenWrapperDownloader.java
vendored
Normal file
93
.mvn/wrapper/MavenWrapperDownloader.java
vendored
Normal file
@ -0,0 +1,93 @@
|
|||||||
|
/*
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one
|
||||||
|
* or more contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. The ASF 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.InputStream;
|
||||||
|
import java.net.Authenticator;
|
||||||
|
import java.net.PasswordAuthentication;
|
||||||
|
import java.net.URI;
|
||||||
|
import java.net.URL;
|
||||||
|
import java.nio.file.Files;
|
||||||
|
import java.nio.file.Path;
|
||||||
|
import java.nio.file.Paths;
|
||||||
|
import java.nio.file.StandardCopyOption;
|
||||||
|
import java.util.concurrent.ThreadLocalRandom;
|
||||||
|
|
||||||
|
public final class MavenWrapperDownloader {
|
||||||
|
private static final String WRAPPER_VERSION = "3.3.2";
|
||||||
|
|
||||||
|
private static final boolean VERBOSE = Boolean.parseBoolean(System.getenv("MVNW_VERBOSE"));
|
||||||
|
|
||||||
|
public static void main(String[] args) {
|
||||||
|
log("Apache Maven Wrapper Downloader " + WRAPPER_VERSION);
|
||||||
|
|
||||||
|
if (args.length != 2) {
|
||||||
|
System.err.println(" - ERROR wrapperUrl or wrapperJarPath parameter missing");
|
||||||
|
System.exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
log(" - Downloader started");
|
||||||
|
final URL wrapperUrl = URI.create(args[0]).toURL();
|
||||||
|
final String jarPath = args[1].replace("..", ""); // Sanitize path
|
||||||
|
final Path wrapperJarPath = Paths.get(jarPath).toAbsolutePath().normalize();
|
||||||
|
downloadFileFromURL(wrapperUrl, wrapperJarPath);
|
||||||
|
log("Done");
|
||||||
|
} catch (IOException e) {
|
||||||
|
System.err.println("- Error downloading: " + e.getMessage());
|
||||||
|
if (VERBOSE) {
|
||||||
|
e.printStackTrace();
|
||||||
|
}
|
||||||
|
System.exit(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void downloadFileFromURL(URL wrapperUrl, Path wrapperJarPath)
|
||||||
|
throws IOException {
|
||||||
|
log(" - Downloading to: " + wrapperJarPath);
|
||||||
|
if (System.getenv("MVNW_USERNAME") != null && System.getenv("MVNW_PASSWORD") != null) {
|
||||||
|
final String username = System.getenv("MVNW_USERNAME");
|
||||||
|
final char[] password = System.getenv("MVNW_PASSWORD").toCharArray();
|
||||||
|
Authenticator.setDefault(new Authenticator() {
|
||||||
|
@Override
|
||||||
|
protected PasswordAuthentication getPasswordAuthentication() {
|
||||||
|
return new PasswordAuthentication(username, password);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
Path temp = wrapperJarPath
|
||||||
|
.getParent()
|
||||||
|
.resolve(wrapperJarPath.getFileName() + "."
|
||||||
|
+ Long.toUnsignedString(ThreadLocalRandom.current().nextLong()) + ".tmp");
|
||||||
|
try (InputStream inStream = wrapperUrl.openStream()) {
|
||||||
|
Files.copy(inStream, temp, StandardCopyOption.REPLACE_EXISTING);
|
||||||
|
Files.move(temp, wrapperJarPath, StandardCopyOption.REPLACE_EXISTING);
|
||||||
|
} finally {
|
||||||
|
Files.deleteIfExists(temp);
|
||||||
|
}
|
||||||
|
log(" - Downloader complete");
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void log(String msg) {
|
||||||
|
if (VERBOSE) {
|
||||||
|
System.out.println(msg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
20
.mvn/wrapper/maven-wrapper.properties
vendored
Normal file
20
.mvn/wrapper/maven-wrapper.properties
vendored
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
# Licensed to the Apache Software Foundation (ASF) under one
|
||||||
|
# or more contributor license agreements. See the NOTICE file
|
||||||
|
# distributed with this work for additional information
|
||||||
|
# regarding copyright ownership. The ASF 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.
|
||||||
|
wrapperVersion=3.3.2
|
||||||
|
distributionType=source
|
||||||
|
distributionUrl=https://repo.maven.apache.org/maven2/org/apache/maven/apache-maven/3.9.8/apache-maven-3.9.8-bin.zip
|
||||||
|
wrapperUrl=https://repo.maven.apache.org/maven2/org/apache/maven/wrapper/maven-wrapper/3.3.2/maven-wrapper-3.3.2.jar
|
11
compose.yml
Normal file
11
compose.yml
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
services:
|
||||||
|
api:
|
||||||
|
container_name: ${REPO}
|
||||||
|
image: ${REPO}
|
||||||
|
restart: always
|
||||||
|
networks:
|
||||||
|
- reverse_proxy
|
||||||
|
|
||||||
|
networks:
|
||||||
|
reverse_proxy:
|
||||||
|
external: true
|
206
pom.xml
Normal file
206
pom.xml
Normal file
@ -0,0 +1,206 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||||
|
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
|
||||||
|
<modelVersion>4.0.0</modelVersion>
|
||||||
|
<groupId>de.arindy</groupId>
|
||||||
|
<artifactId>mythodea-api</artifactId>
|
||||||
|
<version>1.0.0-SNAPSHOT</version>
|
||||||
|
|
||||||
|
<properties>
|
||||||
|
<compiler-plugin.version>3.13.0</compiler-plugin.version>
|
||||||
|
<kotlin.version>2.0.21</kotlin.version>
|
||||||
|
<maven.compiler.release>21</maven.compiler.release>
|
||||||
|
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
|
||||||
|
<project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding>
|
||||||
|
<quarkus.platform.artifact-id>quarkus-bom</quarkus.platform.artifact-id>
|
||||||
|
<quarkus.platform.group-id>io.quarkus.platform</quarkus.platform.group-id>
|
||||||
|
<quarkus.platform.version>3.15.1</quarkus.platform.version>
|
||||||
|
<skipITs>true</skipITs>
|
||||||
|
<surefire-plugin.version>3.3.1</surefire-plugin.version>
|
||||||
|
</properties>
|
||||||
|
|
||||||
|
<dependencyManagement>
|
||||||
|
<dependencies>
|
||||||
|
<dependency>
|
||||||
|
<groupId>${quarkus.platform.group-id}</groupId>
|
||||||
|
<artifactId>${quarkus.platform.artifact-id}</artifactId>
|
||||||
|
<version>${quarkus.platform.version}</version>
|
||||||
|
<type>pom</type>
|
||||||
|
<scope>import</scope>
|
||||||
|
</dependency>
|
||||||
|
</dependencies>
|
||||||
|
</dependencyManagement>
|
||||||
|
|
||||||
|
<dependencies>
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.quarkus</groupId>
|
||||||
|
<artifactId>quarkus-rest-jackson</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.quarkus</groupId>
|
||||||
|
<artifactId>quarkus-kotlin</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.quarkus</groupId>
|
||||||
|
<artifactId>quarkus-arc</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.quarkus</groupId>
|
||||||
|
<artifactId>quarkus-rest</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.quarkus</groupId>
|
||||||
|
<artifactId>quarkus-junit5</artifactId>
|
||||||
|
<scope>test</scope>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.rest-assured</groupId>
|
||||||
|
<artifactId>rest-assured</artifactId>
|
||||||
|
<scope>test</scope>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.rest-assured</groupId>
|
||||||
|
<artifactId>kotlin-extensions</artifactId>
|
||||||
|
<scope>test</scope>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.jetbrains.kotlin</groupId>
|
||||||
|
<artifactId>kotlin-stdlib-jdk8</artifactId>
|
||||||
|
<version>${kotlin.version}</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.jetbrains.kotlin</groupId>
|
||||||
|
<artifactId>kotlin-test</artifactId>
|
||||||
|
<version>${kotlin.version}</version>
|
||||||
|
<scope>test</scope>
|
||||||
|
</dependency>
|
||||||
|
</dependencies>
|
||||||
|
|
||||||
|
<build>
|
||||||
|
<sourceDirectory>src/main/kotlin</sourceDirectory>
|
||||||
|
<testSourceDirectory>src/test/kotlin</testSourceDirectory>
|
||||||
|
<plugins>
|
||||||
|
<plugin>
|
||||||
|
<groupId>${quarkus.platform.group-id}</groupId>
|
||||||
|
<artifactId>quarkus-maven-plugin</artifactId>
|
||||||
|
<version>${quarkus.platform.version}</version>
|
||||||
|
<extensions>true</extensions>
|
||||||
|
<executions>
|
||||||
|
<execution>
|
||||||
|
<goals>
|
||||||
|
<goal>build</goal>
|
||||||
|
<goal>generate-code</goal>
|
||||||
|
<goal>generate-code-tests</goal>
|
||||||
|
<goal>native-image-agent</goal>
|
||||||
|
</goals>
|
||||||
|
</execution>
|
||||||
|
</executions>
|
||||||
|
</plugin>
|
||||||
|
<plugin>
|
||||||
|
<artifactId>maven-compiler-plugin</artifactId>
|
||||||
|
<version>${compiler-plugin.version}</version>
|
||||||
|
<configuration>
|
||||||
|
<parameters>true</parameters>
|
||||||
|
</configuration>
|
||||||
|
</plugin>
|
||||||
|
<plugin>
|
||||||
|
<artifactId>maven-surefire-plugin</artifactId>
|
||||||
|
<version>${surefire-plugin.version}</version>
|
||||||
|
<configuration>
|
||||||
|
<systemPropertyVariables>
|
||||||
|
<java.util.logging.manager>org.jboss.logmanager.LogManager</java.util.logging.manager>
|
||||||
|
<maven.home>${maven.home}</maven.home>
|
||||||
|
</systemPropertyVariables>
|
||||||
|
</configuration>
|
||||||
|
</plugin>
|
||||||
|
<plugin>
|
||||||
|
<artifactId>maven-failsafe-plugin</artifactId>
|
||||||
|
<version>${surefire-plugin.version}</version>
|
||||||
|
<executions>
|
||||||
|
<execution>
|
||||||
|
<goals>
|
||||||
|
<goal>integration-test</goal>
|
||||||
|
<goal>verify</goal>
|
||||||
|
</goals>
|
||||||
|
</execution>
|
||||||
|
</executions>
|
||||||
|
<configuration>
|
||||||
|
<systemPropertyVariables>
|
||||||
|
<native.image.path>${project.build.directory}/${project.build.finalName}-runner
|
||||||
|
</native.image.path>
|
||||||
|
<java.util.logging.manager>org.jboss.logmanager.LogManager</java.util.logging.manager>
|
||||||
|
<maven.home>${maven.home}</maven.home>
|
||||||
|
</systemPropertyVariables>
|
||||||
|
</configuration>
|
||||||
|
</plugin>
|
||||||
|
<plugin>
|
||||||
|
<groupId>org.jetbrains.kotlin</groupId>
|
||||||
|
<artifactId>kotlin-maven-plugin</artifactId>
|
||||||
|
<version>${kotlin.version}</version>
|
||||||
|
<executions>
|
||||||
|
<execution>
|
||||||
|
<id>compile</id>
|
||||||
|
<phase>compile</phase>
|
||||||
|
<goals>
|
||||||
|
<goal>compile</goal>
|
||||||
|
</goals>
|
||||||
|
<configuration>
|
||||||
|
<sourceDirs>
|
||||||
|
<source>src/main/kotlin</source>
|
||||||
|
<source>target/generated-sources/annotations</source>
|
||||||
|
</sourceDirs>
|
||||||
|
</configuration>
|
||||||
|
</execution>
|
||||||
|
<execution>
|
||||||
|
<id>test-compile</id>
|
||||||
|
<phase>test-compile</phase>
|
||||||
|
<goals>
|
||||||
|
<goal>test-compile</goal>
|
||||||
|
</goals>
|
||||||
|
<configuration>
|
||||||
|
<sourceDirs>
|
||||||
|
<source>src/test/kotlin</source>
|
||||||
|
<source>target/generated-test-sources/test-annotations</source>
|
||||||
|
</sourceDirs>
|
||||||
|
</configuration>
|
||||||
|
</execution>
|
||||||
|
</executions>
|
||||||
|
<dependencies>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.jetbrains.kotlin</groupId>
|
||||||
|
<artifactId>kotlin-maven-allopen</artifactId>
|
||||||
|
<version>${kotlin.version}</version>
|
||||||
|
</dependency>
|
||||||
|
</dependencies>
|
||||||
|
<configuration>
|
||||||
|
<javaParameters>true</javaParameters>
|
||||||
|
<jvmTarget>1.8</jvmTarget>
|
||||||
|
<compilerPlugins>
|
||||||
|
<plugin>all-open</plugin>
|
||||||
|
</compilerPlugins>
|
||||||
|
<pluginOptions>
|
||||||
|
<option>all-open:annotation=jakarta.ws.rs.Path</option>
|
||||||
|
<option>all-open:annotation=jakarta.enterprise.context.ApplicationScoped</option>
|
||||||
|
<option>all-open:annotation=jakarta.persistence.Entity</option>
|
||||||
|
<option>all-open:annotation=io.quarkus.test.junit.QuarkusTest</option>
|
||||||
|
</pluginOptions>
|
||||||
|
</configuration>
|
||||||
|
</plugin>
|
||||||
|
</plugins>
|
||||||
|
</build>
|
||||||
|
|
||||||
|
<profiles>
|
||||||
|
<profile>
|
||||||
|
<id>native</id>
|
||||||
|
<activation>
|
||||||
|
<property>
|
||||||
|
<name>native</name>
|
||||||
|
</property>
|
||||||
|
</activation>
|
||||||
|
<properties>
|
||||||
|
<skipITs>false</skipITs>
|
||||||
|
<quarkus.native.enabled>true</quarkus.native.enabled>
|
||||||
|
</properties>
|
||||||
|
</profile>
|
||||||
|
</profiles>
|
||||||
|
</project>
|
11
src/main/docker/Dockerfile.native
Normal file
11
src/main/docker/Dockerfile.native
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
FROM registry.access.redhat.com/ubi8/ubi-minimal:8.10
|
||||||
|
WORKDIR /work/
|
||||||
|
RUN chown 1001 /work \
|
||||||
|
&& chmod "g+rwX" /work \
|
||||||
|
&& chown 1001:root /work
|
||||||
|
COPY --chown=1001:root target/*-runner /work/application
|
||||||
|
|
||||||
|
EXPOSE 8080
|
||||||
|
USER 1001
|
||||||
|
|
||||||
|
ENTRYPOINT ["./application", "-Dquarkus.http.host=0.0.0.0"]
|
11
src/main/docker/Dockerfile.native-micro
Normal file
11
src/main/docker/Dockerfile.native-micro
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
FROM quay.io/quarkus/quarkus-micro-image:2.0
|
||||||
|
WORKDIR /work/
|
||||||
|
RUN chown 1001 /work \
|
||||||
|
&& chmod "g+rwX" /work \
|
||||||
|
&& chown 1001:root /work
|
||||||
|
COPY --chown=1001:root target/*-runner /work/application
|
||||||
|
|
||||||
|
EXPOSE 8080
|
||||||
|
USER 1001
|
||||||
|
|
||||||
|
ENTRYPOINT ["./application", "-Dquarkus.http.host=0.0.0.0"]
|
19
src/main/kotlin/de/arindy/mythodea/api/GreetingResource.kt
Normal file
19
src/main/kotlin/de/arindy/mythodea/api/GreetingResource.kt
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
package de.arindy.mythodea.api
|
||||||
|
|
||||||
|
import jakarta.inject.Inject
|
||||||
|
import jakarta.ws.rs.GET
|
||||||
|
import jakarta.ws.rs.Path
|
||||||
|
import jakarta.ws.rs.Produces
|
||||||
|
import jakarta.ws.rs.QueryParam
|
||||||
|
import jakarta.ws.rs.core.MediaType
|
||||||
|
import java.time.Instant
|
||||||
|
|
||||||
|
@Path("date/")
|
||||||
|
class GreetingResource @Inject constructor(var repository: Repository) {
|
||||||
|
|
||||||
|
@GET
|
||||||
|
@Path("from")
|
||||||
|
@Produces(MediaType.APPLICATION_JSON)
|
||||||
|
fun from(@QueryParam("epochMillis") epochMillis: Long): MitrasperanDate =
|
||||||
|
mitrasperan(if (epochMillis > 0) Instant.ofEpochMilli(epochMillis) else Instant.now())
|
||||||
|
}
|
248
src/main/kotlin/de/arindy/mythodea/api/MitrasperanCalendar.kt
Normal file
248
src/main/kotlin/de/arindy/mythodea/api/MitrasperanCalendar.kt
Normal file
@ -0,0 +1,248 @@
|
|||||||
|
package de.arindy.mythodea.api
|
||||||
|
|
||||||
|
import java.util.Calendar
|
||||||
|
import java.util.Calendar.DAY_OF_MONTH
|
||||||
|
import java.util.Calendar.HOUR_OF_DAY
|
||||||
|
import java.util.Calendar.MINUTE
|
||||||
|
import java.util.Calendar.MONTH
|
||||||
|
import java.util.Calendar.SECOND
|
||||||
|
import java.util.Calendar.YEAR
|
||||||
|
import java.util.TimeZone
|
||||||
|
|
||||||
|
class MitrasperanCalendar {
|
||||||
|
|
||||||
|
fun get(calendar: Calendar): MitrasperanDate {
|
||||||
|
calendar.set(HOUR_OF_DAY, 12)
|
||||||
|
calendar.set(MINUTE, 0)
|
||||||
|
calendar.set(SECOND, 0)
|
||||||
|
val moladStart = Calendar.getInstance(TimeZone.getTimeZone("Europe/Berlin"))
|
||||||
|
moladStart.set(2045, 10, 10, 0, 0, 0)
|
||||||
|
val moladEnd = Calendar.getInstance(TimeZone.getTimeZone("Europe/Berlin"))
|
||||||
|
moladEnd.set(2046, 10, 30, 0, 0, 0)
|
||||||
|
var startOfMolad = false
|
||||||
|
if (moladStart.get(YEAR) == calendar.get(YEAR) && moladStart.get(MONTH) == calendar.get(MONTH) && moladStart.get(
|
||||||
|
DAY_OF_MONTH
|
||||||
|
) == calendar.get(DAY_OF_MONTH)
|
||||||
|
) {
|
||||||
|
startOfMolad = true
|
||||||
|
}
|
||||||
|
if (calendar.after(moladStart) && calendar.before(moladEnd)) {
|
||||||
|
calendar.add(Calendar.DATE, -1)
|
||||||
|
}
|
||||||
|
val absDate = absoluteFromGregorianDate(calendar)
|
||||||
|
var y: Int
|
||||||
|
var m: Int
|
||||||
|
var day: Int
|
||||||
|
var temp: Int
|
||||||
|
|
||||||
|
/* Approximation */
|
||||||
|
val approx = (absDate + 1373429) / 366
|
||||||
|
|
||||||
|
/* Search forward from the approximation */
|
||||||
|
y = approx
|
||||||
|
while (true) {
|
||||||
|
temp = absoluteFromMitrasperanDate(MitrasperanDate(1, 7, y + 1))
|
||||||
|
if (absDate < temp) break
|
||||||
|
y++
|
||||||
|
}
|
||||||
|
val year = y
|
||||||
|
|
||||||
|
/* Starting month for search for month */
|
||||||
|
temp = absoluteFromMitrasperanDate(MitrasperanDate(1, 1, year))
|
||||||
|
val start = if (absDate < temp) 7
|
||||||
|
else 1
|
||||||
|
|
||||||
|
/* Search forward from either Tishri or Nisan */
|
||||||
|
m = start
|
||||||
|
while (true) {
|
||||||
|
temp = absoluteFromMitrasperanDate(MitrasperanDate(getLastDayOfMitrasperanMonth(m, year), m, year))
|
||||||
|
if (absDate <= temp) break
|
||||||
|
m++
|
||||||
|
}
|
||||||
|
val month = m
|
||||||
|
|
||||||
|
/* Calculate the day by subtraction */
|
||||||
|
temp = absoluteFromMitrasperanDate(MitrasperanDate(1, month, year))
|
||||||
|
day = absDate - temp + 1
|
||||||
|
|
||||||
|
if (startOfMolad) {
|
||||||
|
day++
|
||||||
|
}
|
||||||
|
|
||||||
|
return MitrasperanDate(day, if (month == 12 && !leapYear(year)) 13 else month, year(year), calendar.get(MONTH))
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun leapYear(year: Int): Boolean {
|
||||||
|
return ((7 * year) + 1) % 19 < 7
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun getLastDayOfGregorianMonth(month: Int, year: Int): Int {
|
||||||
|
return if ((month == 2) &&
|
||||||
|
((year % 4) == 0) &&
|
||||||
|
((year % 400) != 100) &&
|
||||||
|
((year % 400) != 200) &&
|
||||||
|
((year % 400) != 300)
|
||||||
|
) 29
|
||||||
|
else intArrayOf(31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31)[month - 1]
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun year(year: Int): Int {
|
||||||
|
return year - 5763
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun absoluteFromGregorianDate(date: Calendar): Int {
|
||||||
|
/* Days so far this month */
|
||||||
|
var value = date.get(DAY_OF_MONTH)
|
||||||
|
|
||||||
|
/* Days in prior months this year */
|
||||||
|
var m = 1
|
||||||
|
while (m < date.get(MONTH) + 1) {
|
||||||
|
value += getLastDayOfGregorianMonth(m, date.get(YEAR))
|
||||||
|
m++
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Days in prior years */
|
||||||
|
value += (365 * (date.get(YEAR) - 1))
|
||||||
|
|
||||||
|
/* Julian leap days in prior years ... */
|
||||||
|
value += ((date.get(YEAR) - 1) / 4)
|
||||||
|
|
||||||
|
/* ... minus prior century years ... */
|
||||||
|
value -= ((date.get(YEAR) - 1) / 100)
|
||||||
|
|
||||||
|
/* ... plus prior years divisible by 400 */
|
||||||
|
value += ((date.get(YEAR) - 1) / 400)
|
||||||
|
|
||||||
|
return (value)
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun absoluteFromMitrasperanDate(date: MitrasperanDate): Int {
|
||||||
|
var value: Int
|
||||||
|
var m: Int
|
||||||
|
|
||||||
|
/* Days so far this month */
|
||||||
|
value = date.day
|
||||||
|
var returnValue = value
|
||||||
|
|
||||||
|
/* If before Tishri */
|
||||||
|
if (date.month < 7) {
|
||||||
|
/* Then add days in prior months this year before and */
|
||||||
|
/* after Nisan. */
|
||||||
|
m = 7
|
||||||
|
while (m <= getLastMonthOfMitrasperanYear(date.year)) {
|
||||||
|
value = getLastDayOfMitrasperanMonth(m, date.year)
|
||||||
|
returnValue += value
|
||||||
|
m++
|
||||||
|
}
|
||||||
|
m = 1
|
||||||
|
while (m < date.month) {
|
||||||
|
value = getLastDayOfMitrasperanMonth(m, date.year)
|
||||||
|
returnValue += value
|
||||||
|
m++
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
m = 7
|
||||||
|
while (m < date.month) {
|
||||||
|
value = getLastDayOfMitrasperanMonth(m, date.year)
|
||||||
|
returnValue += value
|
||||||
|
m++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Days in prior years */
|
||||||
|
value = mitrasperanCalendarElapsedDays(date.year)
|
||||||
|
returnValue += value
|
||||||
|
|
||||||
|
/* Days elapsed before absolute date 1 */
|
||||||
|
value = 1373429
|
||||||
|
returnValue -= value
|
||||||
|
|
||||||
|
return (returnValue)
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun getLastMonthOfMitrasperanYear(year: Int): Int {
|
||||||
|
return if (leapYear(year)) 13
|
||||||
|
else 12
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun getLastDayOfMitrasperanMonth(month: Int, year: Int): Int {
|
||||||
|
if ((month == 2) ||
|
||||||
|
(month == 4) ||
|
||||||
|
(month == 6) ||
|
||||||
|
(month == 10) ||
|
||||||
|
(month == 13)
|
||||||
|
) return 29
|
||||||
|
if ((month == 12) && (!leapYear(year))) return 29
|
||||||
|
if ((month == 8) && (!longYear(year))) return 29
|
||||||
|
if ((month == 9) && (shortYear(year))) return 29
|
||||||
|
return 30
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun daysInMitrasperanYear(year: Int): Int {
|
||||||
|
return (mitrasperanCalendarElapsedDays(year + 1) -
|
||||||
|
mitrasperanCalendarElapsedDays(year))
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun longYear(year: Int): Boolean {
|
||||||
|
return (daysInMitrasperanYear(year) % 10) == 5
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun shortYear(year: Int): Boolean {
|
||||||
|
return (daysInMitrasperanYear(year) % 10) == 3
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun mitrasperanCalendarElapsedDays(year: Int): Int {
|
||||||
|
var monthsElapsed: Int
|
||||||
|
val hoursElapsed: Int
|
||||||
|
val parts: Int
|
||||||
|
var alternativeDay: Int
|
||||||
|
|
||||||
|
/* Months in complete cycles so far */
|
||||||
|
var value = 235 * ((year - 1) / 19)
|
||||||
|
monthsElapsed = value
|
||||||
|
|
||||||
|
/* Regular months in this cycle */
|
||||||
|
value = 12 * ((year - 1) % 19)
|
||||||
|
monthsElapsed += value
|
||||||
|
|
||||||
|
/* Leap months this cycle */
|
||||||
|
value = ((((year - 1) % 19) * 7) + 1) / 19
|
||||||
|
monthsElapsed += value
|
||||||
|
|
||||||
|
val partsElapsed = (((monthsElapsed % 1080) * 793) + 204)
|
||||||
|
hoursElapsed = (5 +
|
||||||
|
(monthsElapsed * 12) +
|
||||||
|
((monthsElapsed / 1080) * 793) +
|
||||||
|
(partsElapsed / 1080))
|
||||||
|
|
||||||
|
/* Conjunction day */
|
||||||
|
val day = 1 + (29 * monthsElapsed) + (hoursElapsed / 24)
|
||||||
|
|
||||||
|
/* Conjunction parts */
|
||||||
|
parts = ((hoursElapsed % 24) * 1080) +
|
||||||
|
(partsElapsed % 1080)
|
||||||
|
|
||||||
|
/* If new moon is at or after midday, */
|
||||||
|
alternativeDay = if ((parts >= 19440) || /* ...or is on a Tuesday... */
|
||||||
|
(((day % 7) == 2) && /* at 9 hours, 204 parts or later */
|
||||||
|
(parts >= 9924) && /* of a common year */
|
||||||
|
(!leapYear(year))) || /* ...or is on a Monday at... */
|
||||||
|
(((day % 7) == 1) && /* 15 hours, 589 parts or later... */
|
||||||
|
(parts >= 16789) && /* at the end of a leap year */
|
||||||
|
(leapYear(year - 1)))
|
||||||
|
) /* Then postpone Rosh HaShanah one day */
|
||||||
|
day + 1
|
||||||
|
else day
|
||||||
|
|
||||||
|
/* If Rosh HaShanah would occur on Sunday, Wednesday, */
|
||||||
|
/* or Friday */
|
||||||
|
if (((alternativeDay % 7) == 0) ||
|
||||||
|
((alternativeDay % 7) == 3) ||
|
||||||
|
((alternativeDay % 7) == 5)
|
||||||
|
) /* Then postpone it one (more) day and return */
|
||||||
|
alternativeDay++
|
||||||
|
|
||||||
|
return (alternativeDay)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
78
src/main/kotlin/de/arindy/mythodea/api/MitrasperanDate.kt
Normal file
78
src/main/kotlin/de/arindy/mythodea/api/MitrasperanDate.kt
Normal file
@ -0,0 +1,78 @@
|
|||||||
|
package de.arindy.mythodea.api
|
||||||
|
|
||||||
|
import io.quarkus.logging.Log
|
||||||
|
import java.time.Instant
|
||||||
|
import java.util.Calendar
|
||||||
|
import java.util.Date
|
||||||
|
import java.util.TimeZone
|
||||||
|
|
||||||
|
data class MitrasperanDate internal constructor(
|
||||||
|
val day: Int,
|
||||||
|
val month: Int,
|
||||||
|
val year: Int,
|
||||||
|
private val gregorianMonth: Int,
|
||||||
|
val yearNDE: Int = if (month <= 6 || gregorianMonth >= 7) year + 1 else year,
|
||||||
|
val yearNDK: Int = year - 19,
|
||||||
|
val dayString: String = DAYS[day - 1],
|
||||||
|
val monthString: String = MONTHS[month - 1]
|
||||||
|
) {
|
||||||
|
|
||||||
|
constructor(day: Int, month: Int, year: Int) : this(day, month, year, -1)
|
||||||
|
|
||||||
|
companion object {
|
||||||
|
private val MONTHS = arrayOf(
|
||||||
|
"Wandelmond",
|
||||||
|
"Launing",
|
||||||
|
"Blütenmond",
|
||||||
|
"Brachmond",
|
||||||
|
"Weidmond",
|
||||||
|
"Naiba",
|
||||||
|
"Holzmond",
|
||||||
|
"Goldmond",
|
||||||
|
"Scheiding",
|
||||||
|
"Blaumond",
|
||||||
|
"Gilbhart",
|
||||||
|
"Eismond",
|
||||||
|
"Fralt",
|
||||||
|
)
|
||||||
|
private val DAYS = arrayOf(
|
||||||
|
"1. Erztag",
|
||||||
|
"1. Fyrstag",
|
||||||
|
"1. Bindetag",
|
||||||
|
"1. Meerstag",
|
||||||
|
"1. Wintstag",
|
||||||
|
"1. Mahntag",
|
||||||
|
"2. Erztag",
|
||||||
|
"2. Fyrstag",
|
||||||
|
"2. Bindetag",
|
||||||
|
"2. Meerstag",
|
||||||
|
"2. Wintstag",
|
||||||
|
"2. Mahntag",
|
||||||
|
"Weiße Vornacht",
|
||||||
|
"Weiße Nacht",
|
||||||
|
"Weiße Nacht",
|
||||||
|
"3. Erztag",
|
||||||
|
"3. Fyrstag",
|
||||||
|
"3. Bindetag",
|
||||||
|
"3. Meerstag",
|
||||||
|
"3. Wintstag",
|
||||||
|
"3. Mahntag",
|
||||||
|
"4. Erztag",
|
||||||
|
"4. Fyrstag",
|
||||||
|
"4. Bindetag",
|
||||||
|
"4. Meerstag",
|
||||||
|
"4. Wintstag",
|
||||||
|
"4. Mahntag",
|
||||||
|
"Schwarze Nacht",
|
||||||
|
"Schwarze Nacht",
|
||||||
|
"Pechnacht",
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fun mitrasperan(date: Instant): MitrasperanDate {
|
||||||
|
val calendar = Calendar.getInstance(TimeZone.getTimeZone("Europe/Berlin"))
|
||||||
|
calendar.time = Date.from(date)
|
||||||
|
Log.info(calendar.time.toString())
|
||||||
|
return MitrasperanCalendar().get(calendar)
|
||||||
|
}
|
10
src/main/kotlin/de/arindy/mythodea/api/RepoImpl.kt
Normal file
10
src/main/kotlin/de/arindy/mythodea/api/RepoImpl.kt
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
package de.arindy.mythodea.api
|
||||||
|
|
||||||
|
import jakarta.enterprise.context.RequestScoped
|
||||||
|
|
||||||
|
@RequestScoped
|
||||||
|
class RepoImpl: Repository {
|
||||||
|
override fun name(): String {
|
||||||
|
return "Hello from Quarkus REST"
|
||||||
|
}
|
||||||
|
}
|
6
src/main/kotlin/de/arindy/mythodea/api/Repository.kt
Normal file
6
src/main/kotlin/de/arindy/mythodea/api/Repository.kt
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
package de.arindy.mythodea.api
|
||||||
|
|
||||||
|
interface Repository {
|
||||||
|
|
||||||
|
fun name(): String
|
||||||
|
}
|
1
src/main/resources/application.properties
Normal file
1
src/main/resources/application.properties
Normal file
@ -0,0 +1 @@
|
|||||||
|
quarkus.http.port=8080
|
8
src/test/kotlin/org/acme/GreetingResourceIT.kt
Normal file
8
src/test/kotlin/org/acme/GreetingResourceIT.kt
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
package de.arindy.mythodea.api
|
||||||
|
|
||||||
|
import io.quarkus.test.junit.QuarkusIntegrationTest
|
||||||
|
|
||||||
|
@QuarkusIntegrationTest
|
||||||
|
class GreetingResourceIT : GreetingResourceTest() {
|
||||||
|
|
||||||
|
}
|
9
src/test/kotlin/org/acme/GreetingResourceTest.kt
Normal file
9
src/test/kotlin/org/acme/GreetingResourceTest.kt
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
package de.arindy.mythodea.api
|
||||||
|
|
||||||
|
import io.quarkus.test.junit.QuarkusTest
|
||||||
|
|
||||||
|
@QuarkusTest
|
||||||
|
class GreetingResourceTest {
|
||||||
|
|
||||||
|
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user