Correct meta item index representation in JSON
This commit is contained in:
parent
87326d05c7
commit
faeb737672
build.gradle.kts
dataforge-io/src/commonMain/kotlin/hep/dataforge/io
dataforge-meta/src
commonMain/kotlin/hep/dataforge
commonTest/kotlin/hep/dataforge/meta
dataforge-workspace/src/commonMain/kotlin/hep/dataforge/workspace
@ -7,7 +7,7 @@ plugins {
|
|||||||
id("org.jetbrains.dokka") version "0.10.1"
|
id("org.jetbrains.dokka") version "0.10.1"
|
||||||
}
|
}
|
||||||
|
|
||||||
val dataforgeVersion by extra("0.1.8-dev-5")
|
val dataforgeVersion by extra("0.1.8-dev-6")
|
||||||
|
|
||||||
val bintrayRepo by extra("dataforge")
|
val bintrayRepo by extra("dataforge")
|
||||||
val githubProject by extra("dataforge-core")
|
val githubProject by extra("dataforge-core")
|
||||||
|
@ -16,7 +16,6 @@ import kotlin.reflect.KClass
|
|||||||
@ExperimentalUnsignedTypes
|
@ExperimentalUnsignedTypes
|
||||||
data class PartialEnvelope(val meta: Meta, val dataOffset: UInt, val dataSize: ULong?)
|
data class PartialEnvelope(val meta: Meta, val dataOffset: UInt, val dataSize: ULong?)
|
||||||
|
|
||||||
|
|
||||||
interface EnvelopeFormat : IOFormat<Envelope> {
|
interface EnvelopeFormat : IOFormat<Envelope> {
|
||||||
val defaultMetaFormat: MetaFormatFactory get() = JsonMetaFormat
|
val defaultMetaFormat: MetaFormatFactory get() = JsonMetaFormat
|
||||||
|
|
||||||
@ -33,6 +32,10 @@ interface EnvelopeFormat : IOFormat<Envelope> {
|
|||||||
override fun Output.writeObject(obj: Envelope): Unit = writeEnvelope(obj)
|
override fun Output.writeObject(obj: Envelope): Unit = writeEnvelope(obj)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fun EnvelopeFormat.readPartial(input: Input) = input.readPartial()
|
||||||
|
|
||||||
|
fun EnvelopeFormat.read(input: Input) = input.readObject()
|
||||||
|
|
||||||
@Type(ENVELOPE_FORMAT_TYPE)
|
@Type(ENVELOPE_FORMAT_TYPE)
|
||||||
interface EnvelopeFormatFactory : IOFormatFactory<Envelope>, EnvelopeFormat {
|
interface EnvelopeFormatFactory : IOFormatFactory<Envelope>, EnvelopeFormat {
|
||||||
override val name: Name get() = "envelope".asName()
|
override val name: Name get() = "envelope".asName()
|
||||||
|
@ -166,6 +166,8 @@ class TaggedEnvelopeFormat(
|
|||||||
|
|
||||||
override fun Input.readObject(): Envelope =
|
override fun Input.readObject(): Envelope =
|
||||||
default.run { readObject() }
|
default.run { readObject() }
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
@ -6,7 +6,7 @@ import hep.dataforge.meta.descriptors.ItemDescriptor
|
|||||||
import hep.dataforge.meta.descriptors.NodeDescriptor
|
import hep.dataforge.meta.descriptors.NodeDescriptor
|
||||||
import hep.dataforge.meta.descriptors.ValueDescriptor
|
import hep.dataforge.meta.descriptors.ValueDescriptor
|
||||||
import hep.dataforge.names.NameToken
|
import hep.dataforge.names.NameToken
|
||||||
import hep.dataforge.names.toName
|
import hep.dataforge.names.withIndex
|
||||||
import hep.dataforge.values.*
|
import hep.dataforge.values.*
|
||||||
import kotlinx.serialization.json.*
|
import kotlinx.serialization.json.*
|
||||||
|
|
||||||
@ -35,29 +35,29 @@ private fun String.toJsonKey(descriptor: ItemDescriptor?) = descriptor?.attribut
|
|||||||
/**
|
/**
|
||||||
* Convert given [Meta] to [JsonObject]. Primitives and nodes are copied as is, same name siblings are treated as json arrays
|
* Convert given [Meta] to [JsonObject]. Primitives and nodes are copied as is, same name siblings are treated as json arrays
|
||||||
*/
|
*/
|
||||||
fun Meta.toJson(descriptor: NodeDescriptor? = null, index: String? = null): JsonObject {
|
private fun Meta.toJsonWithIndex(descriptor: NodeDescriptor?, indexValue: String?): JsonObject {
|
||||||
|
|
||||||
val elementMap = HashMap<String, JsonElement>()
|
val elementMap = HashMap<String, JsonElement>()
|
||||||
|
|
||||||
fun MetaItem<*>.toJsonElement(itemDescriptor: ItemDescriptor?, index: String? = null): JsonElement = when (this) {
|
fun MetaItem<*>.toJsonElement(itemDescriptor: ItemDescriptor?, index: String?): JsonElement = when (this) {
|
||||||
is MetaItem.ValueItem -> {
|
is MetaItem.ValueItem -> {
|
||||||
value.toJson(itemDescriptor as? ValueDescriptor)
|
value.toJson(itemDescriptor as? ValueDescriptor)
|
||||||
}
|
}
|
||||||
is MetaItem.NodeItem -> {
|
is MetaItem.NodeItem -> {
|
||||||
node.toJson(itemDescriptor as? NodeDescriptor, index)
|
node.toJsonWithIndex(itemDescriptor as? NodeDescriptor, index)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fun addElement(key: String) {
|
fun addElement(key: String) {
|
||||||
val itemDescriptor = descriptor?.items?.get(key)
|
val itemDescriptor = descriptor?.items?.get(key)
|
||||||
val jsonKey = key.toJsonKey(itemDescriptor)
|
val jsonKey = key.toJsonKey(itemDescriptor)
|
||||||
val items = getIndexed(key)
|
val items: Map<String, MetaItem<*>> = getIndexed(key)
|
||||||
when (items.size) {
|
when (items.size) {
|
||||||
0 -> {
|
0 -> {
|
||||||
//do nothing
|
//do nothing
|
||||||
}
|
}
|
||||||
1 -> {
|
1 -> {
|
||||||
elementMap[jsonKey] = items.values.first().toJsonElement(itemDescriptor)
|
elementMap[jsonKey] = items.values.first().toJsonElement(itemDescriptor, null)
|
||||||
}
|
}
|
||||||
else -> {
|
else -> {
|
||||||
val array = jsonArray {
|
val array = jsonArray {
|
||||||
@ -73,38 +73,29 @@ fun Meta.toJson(descriptor: NodeDescriptor? = null, index: String? = null): Json
|
|||||||
((descriptor?.items?.keys ?: emptySet()) + items.keys.map { it.body }).forEach(::addElement)
|
((descriptor?.items?.keys ?: emptySet()) + items.keys.map { it.body }).forEach(::addElement)
|
||||||
|
|
||||||
|
|
||||||
if (index != null) {
|
if (indexValue != null) {
|
||||||
elementMap["@index"] = JsonPrimitive(index)
|
val indexKey = descriptor?.indexKey ?: NodeDescriptor.DEFAULT_INDEX_KEY
|
||||||
|
elementMap[indexKey] = JsonPrimitive(indexValue)
|
||||||
}
|
}
|
||||||
|
|
||||||
return JsonObject(elementMap)
|
return JsonObject(elementMap)
|
||||||
|
|
||||||
// // use descriptor keys in the order they are declared
|
|
||||||
// val keys = (descriptor?.items?.keys ?: emptySet()) + this.items.keys.map { it.body }
|
|
||||||
//
|
|
||||||
// //TODO search for same name siblings and arrange them into arrays
|
|
||||||
// val map = this.items.entries.associate { (name, item) ->
|
|
||||||
// val itemDescriptor = descriptor?.items?.get(name.body)
|
|
||||||
// val key = name.toJsonKey(itemDescriptor)
|
|
||||||
// val value = when (item) {
|
|
||||||
// is MetaItem.ValueItem -> {
|
|
||||||
// item.value.toJson(itemDescriptor as? ValueDescriptor)
|
|
||||||
// }
|
|
||||||
// is MetaItem.NodeItem -> {
|
|
||||||
// item.node.toJson(itemDescriptor as? NodeDescriptor)
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
// key to value
|
|
||||||
// }
|
|
||||||
// return JsonObject(map)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fun JsonObject.toMeta(descriptor: NodeDescriptor? = null): Meta = JsonMeta(this, descriptor)
|
fun Meta.toJson(descriptor: NodeDescriptor? = null): JsonObject = toJsonWithIndex(descriptor, null)
|
||||||
|
|
||||||
|
fun JsonObject.toMeta(descriptor: NodeDescriptor? = null): JsonMeta = JsonMeta(this, descriptor)
|
||||||
|
|
||||||
fun JsonPrimitive.toValue(descriptor: ValueDescriptor?): Value {
|
fun JsonPrimitive.toValue(descriptor: ValueDescriptor?): Value {
|
||||||
return when (this) {
|
return when (this) {
|
||||||
JsonNull -> Null
|
JsonNull -> Null
|
||||||
else -> this.content.parseValue() // Optimize number and boolean parsing
|
is JsonLiteral -> {
|
||||||
|
when (body) {
|
||||||
|
true -> True
|
||||||
|
false -> False
|
||||||
|
is Number -> NumberValue(body as Number)
|
||||||
|
else -> StringValue(content)
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -131,9 +122,8 @@ fun JsonElement.toMetaItem(descriptor: ItemDescriptor? = null): MetaItem<JsonMet
|
|||||||
}
|
}
|
||||||
MetaItem.ValueItem(value)
|
MetaItem.ValueItem(value)
|
||||||
} else {
|
} else {
|
||||||
json {
|
//We can't return multiple items therefore we create top level node
|
||||||
"@value" to this@toMetaItem
|
json { "@json" to this@toMetaItem }.toMetaItem(descriptor)
|
||||||
}.toMetaItem(descriptor)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -143,44 +133,44 @@ fun JsonElement.toMetaItem(descriptor: ItemDescriptor? = null): MetaItem<JsonMet
|
|||||||
*/
|
*/
|
||||||
class JsonMeta(val json: JsonObject, val descriptor: NodeDescriptor? = null) : MetaBase() {
|
class JsonMeta(val json: JsonObject, val descriptor: NodeDescriptor? = null) : MetaBase() {
|
||||||
|
|
||||||
@Suppress("UNCHECKED_CAST")
|
private fun buildItems(): Map<NameToken, MetaItem<JsonMeta>> {
|
||||||
private operator fun MutableMap<String, MetaItem<JsonMeta>>.set(key: String, value: JsonElement): Unit {
|
val map = LinkedHashMap<NameToken, MetaItem<JsonMeta>>()
|
||||||
val itemDescriptor = descriptor?.items?.get(key)
|
|
||||||
when (value) {
|
json.forEach { (jsonKey, value) ->
|
||||||
is JsonPrimitive -> {
|
val key = NameToken(jsonKey)
|
||||||
this[key] =
|
val itemDescriptor = descriptor?.items?.get(jsonKey)
|
||||||
MetaItem.ValueItem(value.toValue(itemDescriptor as? ValueDescriptor)) as MetaItem<JsonMeta>
|
when (value) {
|
||||||
}
|
is JsonPrimitive -> {
|
||||||
is JsonObject -> {
|
map[key] = MetaItem.ValueItem(value.toValue(itemDescriptor as? ValueDescriptor))
|
||||||
this[key] = MetaItem.NodeItem(
|
}
|
||||||
JsonMeta(
|
is JsonObject -> {
|
||||||
value,
|
map[key] = MetaItem.NodeItem(
|
||||||
itemDescriptor as? NodeDescriptor
|
JsonMeta(
|
||||||
)
|
value,
|
||||||
)
|
itemDescriptor as? NodeDescriptor
|
||||||
}
|
|
||||||
is JsonArray -> {
|
|
||||||
when {
|
|
||||||
value.all { it is JsonPrimitive } -> {
|
|
||||||
val listValue = ListValue(
|
|
||||||
value.map {
|
|
||||||
//We already checked that all values are primitives
|
|
||||||
(it as JsonPrimitive).toValue(itemDescriptor as? ValueDescriptor)
|
|
||||||
}
|
|
||||||
)
|
)
|
||||||
this[key] = MetaItem.ValueItem(listValue) as MetaItem<JsonMeta>
|
)
|
||||||
}
|
}
|
||||||
else -> value.forEachIndexed { index, jsonElement ->
|
is JsonArray -> if (value.all { it is JsonPrimitive }) {
|
||||||
this["$key[$index]"] = jsonElement.toMetaItem(itemDescriptor)
|
val listValue = ListValue(
|
||||||
}
|
value.map {
|
||||||
|
//We already checked that all values are primitives
|
||||||
|
(it as JsonPrimitive).toValue(itemDescriptor as? ValueDescriptor)
|
||||||
|
}
|
||||||
|
)
|
||||||
|
map[key] = MetaItem.ValueItem(listValue)
|
||||||
|
} else value.forEachIndexed { index, jsonElement ->
|
||||||
|
val indexKey = (itemDescriptor as? NodeDescriptor)?.indexKey ?: NodeDescriptor.DEFAULT_INDEX_KEY
|
||||||
|
val indexValue: String = (jsonElement as? JsonObject)
|
||||||
|
?.get(indexKey)?.contentOrNull ?: index.toString() //In case index is non-string, the backward transformation will be broken.
|
||||||
|
|
||||||
|
val token = key.withIndex(indexValue)
|
||||||
|
map[token] = jsonElement.toMetaItem(itemDescriptor)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
return map
|
||||||
}
|
}
|
||||||
|
|
||||||
override val items: Map<NameToken, MetaItem<JsonMeta>> by lazy {
|
override val items: Map<NameToken, MetaItem<JsonMeta>> by lazy(::buildItems)
|
||||||
val map = LinkedHashMap<String, MetaItem<JsonMeta>>()
|
|
||||||
json.forEach { (key, value) -> map[key] = value }
|
|
||||||
map.mapKeys { it.key.toName().first()!! }
|
|
||||||
}
|
|
||||||
}
|
}
|
@ -135,12 +135,12 @@ fun <M : MutableMeta<M>> M.update(meta: Meta) {
|
|||||||
fun MutableMeta<*>.setIndexedItems(
|
fun MutableMeta<*>.setIndexedItems(
|
||||||
name: Name,
|
name: Name,
|
||||||
items: Iterable<MetaItem<*>>,
|
items: Iterable<MetaItem<*>>,
|
||||||
indexFactory: MetaItem<*>.(index: Int) -> String = { it.toString() }
|
indexFactory: (MetaItem<*>, index: Int) -> String = {_, index-> index.toString() }
|
||||||
) {
|
) {
|
||||||
val tokens = name.tokens.toMutableList()
|
val tokens = name.tokens.toMutableList()
|
||||||
val last = tokens.last()
|
val last = tokens.last()
|
||||||
items.forEachIndexed { index, meta ->
|
items.forEachIndexed { index, meta ->
|
||||||
val indexedToken = NameToken(last.body, last.index + meta.indexFactory(index))
|
val indexedToken = NameToken(last.body, last.index + indexFactory(meta, index))
|
||||||
tokens[tokens.lastIndex] = indexedToken
|
tokens[tokens.lastIndex] = indexedToken
|
||||||
set(Name(tokens), meta)
|
set(Name(tokens), meta)
|
||||||
}
|
}
|
||||||
@ -149,9 +149,9 @@ fun MutableMeta<*>.setIndexedItems(
|
|||||||
fun MutableMeta<*>.setIndexed(
|
fun MutableMeta<*>.setIndexed(
|
||||||
name: Name,
|
name: Name,
|
||||||
metas: Iterable<Meta>,
|
metas: Iterable<Meta>,
|
||||||
indexFactory: MetaItem<*>.(index: Int) -> String = { it.toString() }
|
indexFactory: (Meta, index: Int) -> String = { _, index-> index.toString() }
|
||||||
) {
|
) {
|
||||||
setIndexedItems(name, metas.map { MetaItem.NodeItem(it) }, indexFactory)
|
setIndexedItems(name, metas.map { MetaItem.NodeItem(it) }){item, index ->indexFactory(item.node!!, index)}
|
||||||
}
|
}
|
||||||
|
|
||||||
operator fun MutableMeta<*>.set(name: Name, metas: Iterable<Meta>): Unit = setIndexed(name, metas)
|
operator fun MutableMeta<*>.set(name: Name, metas: Iterable<Meta>): Unit = setIndexed(name, metas)
|
||||||
|
@ -50,9 +50,6 @@ fun <R> ReadWriteProperty<Any?, MetaItem<*>?>.transform(reader: (MetaItem<*>?) -
|
|||||||
fun <R> ReadWriteProperty<Any?, Value?>.transform(reader: (Value?) -> R): ReadWriteDelegateWrapper<Value?, R> =
|
fun <R> ReadWriteProperty<Any?, Value?>.transform(reader: (Value?) -> R): ReadWriteDelegateWrapper<Value?, R> =
|
||||||
map(reader = reader, writer = { Value.of(it) })
|
map(reader = reader, writer = { Value.of(it) })
|
||||||
|
|
||||||
/**
|
|
||||||
* A delegate that throws
|
|
||||||
*/
|
|
||||||
fun <R : Any> ReadWriteProperty<Any?, R?>.notNull(default: () -> R): ReadWriteProperty<Any?, R> {
|
fun <R : Any> ReadWriteProperty<Any?, R?>.notNull(default: () -> R): ReadWriteProperty<Any?, R> {
|
||||||
return ReadWriteDelegateWrapper(this,
|
return ReadWriteDelegateWrapper(this,
|
||||||
reader = { it ?: default() },
|
reader = { it ?: default() },
|
||||||
@ -60,7 +57,6 @@ fun <R : Any> ReadWriteProperty<Any?, R?>.notNull(default: () -> R): ReadWritePr
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
fun <M : MutableMeta<M>> M.item(default: Any? = null, key: Name? = null): MutableMetaDelegate<M> =
|
fun <M : MutableMeta<M>> M.item(default: Any? = null, key: Name? = null): MutableMetaDelegate<M> =
|
||||||
MutableMetaDelegate(this, key, default?.let { MetaItem.of(it) })
|
MutableMetaDelegate(this, key, default?.let { MetaItem.of(it) })
|
||||||
|
|
||||||
|
@ -95,6 +95,11 @@ class NodeDescriptor(config: Config = Config()) : ItemDescriptor(config) {
|
|||||||
*/
|
*/
|
||||||
var default by config.node()
|
var default by config.node()
|
||||||
|
|
||||||
|
/**
|
||||||
|
* An index field by which this node is identified in case of same name siblings construct
|
||||||
|
*/
|
||||||
|
var indexKey by config.string(DEFAULT_INDEX_KEY)
|
||||||
|
|
||||||
val items: Map<String, ItemDescriptor>
|
val items: Map<String, ItemDescriptor>
|
||||||
get() = config.getIndexed(ITEM_KEY).mapValues { (_, item) ->
|
get() = config.getIndexed(ITEM_KEY).mapValues { (_, item) ->
|
||||||
val node = item.node ?: error("Node descriptor must be a node")
|
val node = item.node ?: error("Node descriptor must be a node")
|
||||||
@ -182,6 +187,8 @@ class NodeDescriptor(config: Config = Config()) : ItemDescriptor(config) {
|
|||||||
val ITEM_KEY = "item".asName()
|
val ITEM_KEY = "item".asName()
|
||||||
val IS_NODE_KEY = "@isNode".asName()
|
val IS_NODE_KEY = "@isNode".asName()
|
||||||
|
|
||||||
|
const val DEFAULT_INDEX_KEY = "@index"
|
||||||
|
|
||||||
inline operator fun invoke(block: NodeDescriptor.() -> Unit) = NodeDescriptor().apply(block)
|
inline operator fun invoke(block: NodeDescriptor.() -> Unit) = NodeDescriptor().apply(block)
|
||||||
|
|
||||||
//TODO infer descriptor from spec
|
//TODO infer descriptor from spec
|
||||||
|
@ -113,6 +113,8 @@ data class NameToken(val body: String, val index: String = "") {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fun NameToken.withIndex(newIndex: String) = NameToken(body, newIndex)
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Convert a [String] to name parsing it and extracting name tokens and index syntax.
|
* Convert a [String] to name parsing it and extracting name tokens and index syntax.
|
||||||
* This operation is rather heavy so it should be used with care in high performance code.
|
* This operation is rather heavy so it should be used with care in high performance code.
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
package hep.dataforge.meta
|
package hep.dataforge.meta
|
||||||
|
|
||||||
|
import hep.dataforge.meta.descriptors.NodeDescriptor
|
||||||
import kotlinx.serialization.json.int
|
import kotlinx.serialization.json.int
|
||||||
import kotlinx.serialization.json.json
|
import kotlinx.serialization.json.json
|
||||||
import kotlinx.serialization.json.jsonArray
|
import kotlinx.serialization.json.jsonArray
|
||||||
@ -17,23 +18,34 @@ class JsonMetaTest {
|
|||||||
}
|
}
|
||||||
"nodeArray" to jsonArray {
|
"nodeArray" to jsonArray {
|
||||||
+json {
|
+json {
|
||||||
"index" to 1
|
"index" to "1"
|
||||||
|
"value" to 2
|
||||||
}
|
}
|
||||||
+json {
|
+json {
|
||||||
"index" to 2
|
"index" to "2"
|
||||||
|
"value" to 3
|
||||||
}
|
}
|
||||||
+json {
|
+json {
|
||||||
"index" to 3
|
"index" to "3"
|
||||||
|
"value" to 4
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
val descriptor = NodeDescriptor{
|
||||||
|
node("nodeArray"){
|
||||||
|
indexKey = "index"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
fun jsonMetaConversion() {
|
fun jsonMetaConversion() {
|
||||||
val meta = json.toMeta()
|
|
||||||
val reconstructed = meta.toJson()
|
|
||||||
println(json)
|
println(json)
|
||||||
|
val meta = json.toMeta(descriptor)
|
||||||
|
//println(meta)
|
||||||
|
val reconstructed = meta.toJson(descriptor)
|
||||||
println(reconstructed)
|
println(reconstructed)
|
||||||
assertEquals(2, reconstructed["nodeArray"]?.jsonArray?.get(1)?.jsonObject?.get("index")?.int)
|
assertEquals(2, reconstructed["nodeArray"]?.jsonArray?.get(1)?.jsonObject?.get("index")?.int)
|
||||||
|
assertEquals(json,reconstructed)
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -36,7 +36,11 @@ data class TaskModel(
|
|||||||
val dataDependencies = dependencies.filterIsInstance<DataDependency>()
|
val dataDependencies = dependencies.filterIsInstance<DataDependency>()
|
||||||
val taskDependencies = dependencies.filterIsInstance<TaskDependency<*>>()
|
val taskDependencies = dependencies.filterIsInstance<TaskDependency<*>>()
|
||||||
setIndexed("data".toName(), dataDependencies.map { it.toMeta() })
|
setIndexed("data".toName(), dataDependencies.map { it.toMeta() })
|
||||||
setIndexed("task".toName(), taskDependencies.map { it.toMeta() }) { taskDependencies[it].name.toString() }
|
setIndexed(
|
||||||
|
"task".toName(),
|
||||||
|
taskDependencies.map { it.toMeta() }) { _, index ->
|
||||||
|
taskDependencies[index].name.toString()
|
||||||
|
}
|
||||||
//TODO ensure all dependencies are listed
|
//TODO ensure all dependencies are listed
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user