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

The AbstractCloseable.doCloseImmediately() race condition #700

Closed
artembilan opened this issue Mar 17, 2025 · 3 comments · Fixed by #702
Closed

The AbstractCloseable.doCloseImmediately() race condition #700

artembilan opened this issue Mar 17, 2025 · 3 comments · Fixed by #702
Assignees
Milestone

Comments

@artembilan
Copy link

Version

2.15.0

Bug description

The code there is like this:

    protected void doCloseImmediately() {
        closeFuture.setClosed();
        state.set(State.Closed);
    }

So, we fulfill CloseFuture first, and then set the state to Closed.
This way Closeable.close(Closeable closeable) exits earlier than we mark that Closeable as Closed.

Actual behavior

In my case the unit test is failing sporadically for this:

session.close();
assertThat(clientSession.isClosed()).isTrue();

Expected behavior

The AbstractCloseable.doCloseImmediately() has those expressions swapped:

    protected void doCloseImmediately() {
        state.set(State.Closed);
        closeFuture.setClosed();
     }

Relevant log output

Other information

No response

artembilan added a commit to spring-projects/spring-integration that referenced this issue Mar 17, 2025
@tomaswolf
Copy link
Member

Not sure. Depends on when the session is considered "closed". The current code seems to assume the session is fully closed once all the listeners on the closeFuture have run, and apparently there may be asynchronous things going on. So maybe an even more correct way to set this state would be

closeFuture.addListener(f -> state.set(State.Closed));
closeFuture.setClosed();

However, ClientSessionImpl.waitFor(...) considers the session closed once closeFuture.isClosed(). So perhaps switching the order like you suggest would be a way. It's a risky change, though, since AbstractCloseable is used for different things, not just sessions.

@tomaswolf
Copy link
Member

BTW, and unrelated to this: do you have tests for shared SSH session usage? We had issue #527 a while back, and it looked to me as if a shared SSH session might get closed too early; c.f. #527 (comment) .

@artembilan
Copy link
Author

Well, the ClientSessionImpl leads to this:

    @Override
    public final boolean isClosed() {
        return state.get() == State.Closed;
    }

So, calling the ClientSessionImpl.close() might not report the proper state because it waits only for CloseFuture.

It is risky to keep it like that.
Similar race conditions may happen in other places.

I guess the complexity of the logic is because there is that third Graceful state.
Well, with that we got two new (Graceful, Immediate) instead of Opened, Closed before, but that's different story.

Thanks for pinging me about that shared session problem!
Will look into that today.

tomaswolf added a commit to tomaswolf/mina-sshd that referenced this issue Mar 18, 2025
Set the state to "closed" before fulfilling the closeFuture.
@tomaswolf tomaswolf self-assigned this Mar 18, 2025
@tomaswolf tomaswolf added this to the 2.16.0 milestone Mar 18, 2025
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging a pull request may close this issue.

2 participants