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

Mutable variables maintain state across calls #157

Open
isaacabraham opened this issue Jan 21, 2016 · 6 comments
Open

Mutable variables maintain state across calls #157

isaacabraham opened this issue Jan 21, 2016 · 6 comments

Comments

@isaacabraham
Copy link
Contributor

I had a brief chat about this with @eiriktsarpalis but it's worth discussing here in more detail. Look at the following code sample: -

let mutable x = 5
let workflow = cloud {
    let! worker = Cloud.CurrentWorker
    x <- x + 1
    return worker.Id, x }
for _ in 1 .. 10 do
    workflow |> cluster.Run |> printfn "%A"

Running the above code produces something like the following: -

("mbrace://work-laptop:36553", 6)
("mbrace://work-laptop:36553", 7)
("mbrace://work-laptop:36553", 8)
("mbrace://work-laptop:36554", 6)
("mbrace://work-laptop:36554", 7)
("mbrace://work-laptop:36554", 8)
("mbrace://work-laptop:36554", 9)
("mbrace://work-laptop:36554", 10)
("mbrace://work-laptop:36554", 11)
("mbrace://work-laptop:36554", 12)

I would have expected the values to all be 6, whereas clearly the lifetime of a captured mutable variable is longer living. How does this behave?

What's even stranger is that if I then do

x <- 10

and rerun the workflow another ten times, I get the following output: -

("mbrace://work-laptop:36554", 11)
("mbrace://work-laptop:36554", 12)
("mbrace://work-laptop:36554", 13)
("mbrace://work-laptop:36554", 14)
("mbrace://work-laptop:36554", 15)
("mbrace://work-laptop:36555", 11)
("mbrace://work-laptop:36552", 11)
("mbrace://work-laptop:36552", 12)
("mbrace://work-laptop:36552", 13)
("mbrace://work-laptop:36552", 14)

Note that I haven't regenerated the workflow - I simple ran the loop again. So a mutable variable has some state of its own on each node, unless it's modified locally, in which case it gets reset.

By the way I'm not suggesting that this is a good pattern to adopt :-) But I didn't expect this behaviour at all.

@palladin
Copy link
Member

I'm pretty sure that I can create the same local observable phenomenon with async and ThreadStatic :)

@eiriktsarpalis
Copy link
Member

The behaviour you're seeing is related to two parameters:

  1. The way in which fsi compiles its top-level value bindings.
  2. A feature implemented in Vagabond in response to 1.

Fsi encodes all of its top-level value bindings as static fields. As such, they introduce what I call "implicit data dependencies" to cloud workflows that reference them, since they are not captured by their object graphs. So Vagabond includes functionality that manually replicates such dependencies (and any client-side mutations) across the cluster. If this didn't get done, the initial observed value on the remote process would always be uninitialized, i.e. 0 or null.

So what you're seeing is not entirely unexpected if you rephrase it in the following terms:

open System
let workflow = cloud {
    let! worker = Cloud.CurrentWorker
    let cwd = Environment.CurrentDirectory
    Environment.CurrentDirectory <- @"C:\"
    return worker.Id, cwd }
for _ in 1 .. 10 do
    workflow |> cluster.Run |> printfn "%A"

It's essentially just modifying global state in the context of the specific worker. Like @palladin said, same as mutating ThreadStatic fields in multi-threaded applications.

@eiriktsarpalis
Copy link
Member

Btw, representing repl values using static fields is not a necessity. The Roslyn C# repl generates code which uses session objects, effectively eliminating this weirdness you're seeing. We should consider doing the same with fsi codegen.

@isaacabraham
Copy link
Contributor Author

Yeah. You can also "fix" this behaviour simply by putting it in a do block :-) It's probably worth documenting this somewhere about static fields in fsx + mutables.

@dsyme
Copy link
Contributor

dsyme commented Jan 21, 2016

@eiriktsarpalis Re Fsi codegen - right, but as I understood from our discussions that comes at the cost of not allowing class-based members to access defined values. So in C# REPL scripting the equivalent of

let a = 1
let b = rnd()
type X() = static member M() = a + b // not allowed, "a" and "b" are not in scope?

is not allowed? (since in C# the "let" become instance declarations in an implied class, and the instance would need to be implicitly passed to M(), which is not done) So can we really adjust F# codegen to avoid statics?

@eiriktsarpalis
Copy link
Member

@dsyme Correct, class definitions in C# interactive cannot contain references to repl values at all. In F# the issue could be somewhat addressed by recovering the session object from a static location in such cases only.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

No branches or pull requests

4 participants