Основной метод приложения Ktor не вызывается при развертывании в AppEngine

Проблема

main метод приложения Ktor не вызывается при развертывании в AppEngine. В основном методе приложений используется логика для извлечения содержимого из запроса API на основе таймера и сохранения этой информации в базе данных Firestore, которую использует клиент.

Эта логика в настоящее время работает при развертывании в Jar в AppEngine. Однако реализация этого с помощью Ktor сэкономит время на развертывание и поможет защитить серверную службу в будущем, если потребуются конечные точки.

Ожидал

Метод main приложения Ktor вызывается после развертывания приложения в AppEngine аналогично тому, как вызывается основной метод приложения при запуске в IntelliJ.

Действительный

Метод main вызывается только после вызова размещенного в приложении маршрута.

ie: https://[yourProjectName].appspot.com

Настраивать

Основной метод

import io.ktor.application.Application
fun Application.main() {
    // App logic here.
}

build.gradle

buildscript {
ext.kotlin_version = '1.3.10'
ext.ktor_version = '1.0.0'
ext.appengine_version = '1.9.60'
ext.appengine_plugin_version = '1.3.4'
ext.junitJupiterVersion  = '5.0.3'

repositories {
    jcenter()
    mavenCentral()
}
dependencies {
    classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlin_version"
    classpath "com.google.cloud.tools:appengine-gradle-plugin:$appengine_plugin_version"
    classpath 'org.junit.platform:junit-platform-gradle-plugin:1.0.3'
    }
}

plugins {
    id 'org.jetbrains.kotlin.jvm' version '1.2.51'
}

apply plugin: 'java'
apply plugin: 'kotlin'
apply plugin: 'war'
apply plugin: 'com.google.cloud.tools.appengine'

sourceSets {
    main.kotlin.srcDirs = [ 'src/main/kotlin' ]
}

sourceCompatibility = 1.8

repositories {
    jcenter()
    mavenCentral()
    maven { url "https://kotlin.bintray.com/ktor" }
}

dependencies {
    implementation "org.jetbrains.kotlin:kotlin-stdlib-jdk8:$kotlin_version"
    implementation "io.ktor:ktor-server-servlet:$ktor_version"
    implementation "io.ktor:ktor-html-builder:$ktor_version"
    providedCompile "com.google.appengine:appengine:$appengine_version"
    implementation 'com.squareup.retrofit2:retrofit:2.3.0'
    implementation 'com.squareup.retrofit2:converter-gson:2.3.0'
    implementation 'com.squareup.retrofit2:adapter-rxjava:2.3.0'
    implementation 'io.reactivex.rxjava2:rxjava:2.1.1'
    implementation 'com.google.firebase:firebase-admin:6.3.0'
    implementation 'com.google.apis:google-api-services-youtube:v3-rev204-1.23.0'
    testCompile group: 'junit', name: 'junit', version: '4.12'
    // JUnit Jupiter API and TestEngine implementation
    testCompile("org.junit.jupiter:junit-jupiter-api:${junitJupiterVersion}")
    testRuntime("org.junit.jupiter:junit-jupiter-engine:${junitJupiterVersion}")
    testCompile("org.assertj:assertj-core:3.10.0")
    // To avoid compiler warnings about @API annotations in JUnit code
    testCompileOnly('org.apiguardian:apiguardian-api:1.0.0')
}

kotlin.experimental.coroutines = 'enable'

compileKotlin {
    kotlinOptions.jvmTarget = "1.8"
}
compileTestKotlin {
    kotlinOptions.jvmTarget = "1.8"
}

task run(dependsOn: appengineRun)

appengine {
    deploy {
    version = 'media-staging-1201181257pm'
    }
}

src / main / resources / application.conf

ktor {
    application {
        modules = [ InitializationKt.main ]
    }
}

src / main / webapp / WEB-INF /

appengine-web.xml

<?xml version="1.0" encoding="utf-8"?>
<appengine-web-app xmlns="http://appengine.google.com/ns/1.0">
    <threadsafe>true</threadsafe>
    <runtime>java8</runtime>
</appengine-web-app>

web.xml

<?xml version="1.0" encoding="ISO-8859-1" ?>
<web-app xmlns="http://java.sun.com/xml/ns/javaee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd" version="3.0">
    <servlet>
        <display-name>KtorServlet</display-name>
        <servlet-name>KtorServlet</servlet-name>
    <servlet-class>io.ktor.server.servlet.ServletApplicationEngine</servlet-class>
        <!-- path to application.conf file, required -->
        <init-param>
            <param-name>io.ktor.config</param-name>
            <param-value>application.conf</param-value>
        </init-param>
    </servlet>
    <servlet-mapping>
        <servlet-name>KtorServlet</servlet-name>
        <url-pattern>/</url-pattern>
    </servlet-mapping>
</web-app>

person Adam Hurwitz    schedule 01.12.2018    source источник


Ответы (2)


Спасибо Габриэлю Мачадо на Kotlin Slack за ответ на этот вопрос.

Габриэль рекомендует использовать Задание Cron < / a> в отличие от задачи таймера, поскольку могут возникнуть проблемы с потоком таймера на основе тип масштабирования.

person Adam Hurwitz    schedule 04.12.2018
comment
Последующий вопрос: Как запускать задания Cron в Kotlin Ktor? - person Adam Hurwitz; 29.12.2019

GAE, вероятно, ожидает, что ваше приложение будет выглядеть как Java-приложение (например, в стиле Java "main", а не в стиле Kotlin "main"). Посмотрите здесь, чтобы узнать больше о том, как это сделать: Как запустить класс Kotlin из командной строки?

person kenyee    schedule 02.12.2018