Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Updating multiple traces in live server is applying only to the first trace #51

Closed
zakhenry opened this issue Nov 20, 2020 · 12 comments
Closed
Assignees
Labels
bug Something isn't working
Milestone

Comments

@zakhenry
Copy link
Contributor

zakhenry commented Nov 20, 2020

Steps to reproduce:

Create multiple traces, and in dynamic server mode try to update the values of each trace. All values appear to be assigned to the first trace.

I've narrowed this down to Plot.collectUpdates

fun Plot.collectUpdates(plotId: String, scope: CoroutineScope, updateInterval: Long): Flow<Update> {
    return config.flowChanges(scope, updateInterval).flatMapMerge { change ->
        flow<Update> {
            change["layout"].node?.let { emit(Update.Layout(plotId, it)) }
            change.getIndexed("data").values.mapNotNull { it.node }.forEachIndexed { index, metaItem ->
                emit(Update.Trace(plotId, index, metaItem))
            }
        }
    }

The index param that is passed to Update.Trace is always 0. This appears to be because there needs to be another inner loop like metaItem.getIndexed to actually iterate through the traces.

The result of this is evident when inspecting the websocket messages, as I can see the json message field "trace" is always 0, despite having the data for each one of the traces that I am updating.

I must apologise, I'm very new to Kotlin so I'm not really sure how to raise a PR that I can test locally to verify my fix.

@altavir
Copy link
Member

altavir commented Nov 20, 2020

I can't reproduce the problem. I've added the second trace to dynamicServer example here and it seems to behave correctly. Could you send me the whole example you are trying to run?

@zakhenry
Copy link
Contributor Author

zakhenry commented Nov 20, 2020

Hi @altavir thanks for your speedy response, here is a repro of the issue I'm seeing

import hep.dataforge.meta.invoke
import kotlinx.coroutines.GlobalScope
import kotlinx.coroutines.delay
import kotlinx.coroutines.isActive
import kotlinx.coroutines.launch
import kotlinx.html.a
import kotlinx.html.h1
import kscience.plotly.Plotly
import kscience.plotly.models.Bar
import kscience.plotly.plot
import kscience.plotly.server.close
import kscience.plotly.server.pushUpdates
import kscience.plotly.server.serve
import kscience.plotly.server.show
import kotlin.random.Random


fun main() {


    val initialValue = listOf(1, 2, 3, 4, 5, 6, 7, 8, 9, 10)

    val tracesList = arrayOfNulls<Bar>(3).mapIndexed { i, bar ->
        Bar {
            x.strings = initialValue.map { "Column: ${it}" }
            y.numbers = initialValue
            name = "Series ${i}"
        }
    }.associateBy { it.name }

    val server = Plotly.serve(port = 3872) {

        //root level plots go to default page
        page { plotly ->
            h1 { +"This is the plot page" }
            a("/other") { +"The other page" }
            plot(renderer = plotly) {
                traces(tracesList.values)
                layout {
                    title = "Other dynamic plot"
                    xaxis.title = "x axis name"
                    yaxis.title = "y axis name"
                }
            }
        }


        pushUpdates(100)       // start sending updates via websocket to the front-end
    }

    server.show()

    //Start pushing updates
    GlobalScope.launch {

        delay(1000)

        while (isActive) {

            repeat(10) { columnIndex ->

                repeat(3) { seriesIndex ->

                    delay(300)

                    tracesList["Series ${seriesIndex}"]?.let {

                        println("Updating ${it.name}, Column ${columnIndex}")

                        it.y.numbers = it.y.numbers.mapIndexed { i, currentValue ->
                            if (columnIndex == i) {
                                Random.nextInt(0, 100)
                            } else {
                                currentValue
                            }
                        }

                    }

                }

            }

        }
    }

    println("Press Enter to close server")
    readLine()

    server.close()
}

In this example, 3 series are created each with 10 initial bars. In the collect we iterate through both the series and the bars one by one and update the value to a random value. I would expect all bars to eventually have a random value, but the bars that are not in series 0 never update from their initial value

Note that I have my traces (Bars in this case) in a Map<String, Bar> as in my actual application the order in which the results are collected is completely unpredictable, so I need to actually look up which series to update

@altavir
Copy link
Member

altavir commented Nov 20, 2020

I think I've reproduced the problem. I will try to understand what happens and how to fix it.
The optimized example is placed here: https://github.com/mipt-npm/plotly.kt/blob/dev/examples/src/main/kotlin/misc/dynamicBars.kt

@altavir
Copy link
Member

altavir commented Nov 20, 2020

OK, I've finally nailed it. It was a really silly mistake and that is why it took so much time to catch. The problem is that in the method you've shown I use forEachIndexed, which catches the number of the change in list of trace changes, not the trace index. 🤦

I've fixed it. And it will work correctly in the next release (should be in a week or so).
image

@altavir
Copy link
Member

altavir commented Nov 20, 2020

Thank you very much for the report.

@zakhenry
Copy link
Contributor Author

Hey that's awesome, thanks for the responsiveness

@zakhenry
Copy link
Contributor Author

@altavir is there any chance this fix could be applied to 0.2? It doesn't seem to need to be bundled with the 0.3 changes I believe?

@altavir
Copy link
Member

altavir commented Nov 24, 2020

Yes, it is pretty simple, but 0.3 is coming in a few days since 1.4.20 is out and I am pressed for time right now. So sadly I do not have time for a separate release activity.

@zakhenry
Copy link
Contributor Author

Fix verified against 0.3.0-dev-rc 😁

@altavir
Copy link
Member

altavir commented Nov 27, 2020

@zakhenry Wow, I just started to write to you.

@zakhenry
Copy link
Contributor Author

Haha, i was literally just rebasing my plotly branch and got the notification from bintray, great timing!

@altavir
Copy link
Member

altavir commented Nov 27, 2020

The release on bintray will be there in several days after I check that the base DataForge library behaves correctly on 1.4.20 in all other libraries.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
bug Something isn't working
Projects
None yet
Development

No branches or pull requests

2 participants