-
-
Notifications
You must be signed in to change notification settings - Fork 14
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
Support async locking #17
Comments
Hi, thanks for opening an issue. Could you give a code example? I do not quite get the point. |
@al8n Sure! Because In order to avoid this, I would use the let f = Arc::new(async_fs::File::open("lockfile"));
let f2 = f.clone();
unblock(move || f2.lock_shared()).await?;
// Now `f` refers to a locked file handle. As you can see, The unfortunate thing about this is that it's necessary to wrap the file in |
Ah, I got it. Yes, I will work on this in the near future. |
Hi, sorry for the delay; after I review the code, you can use |
@al8n |
From my point of view, it is hard to make it a "real async task" for locking. We cannot avoid the system call. Even Actually, unlike network I/O, you can have great performance improvement using async stuff. For file I/O, there is almost no benefit to using async implementation. This is why databases like |
I'm not trying to make the case that file IO can be faster with But in any case, I've found a workaround to get |
Ah, you mean you need some methods like async fn lock_exclusive(&self) -> std::io::Result<()> {
...
} If so, I think I can add those async methods in a new trait |
@al8n Yea that would be great! |
|
The issue with fs4 that there doesn't seem to have any async methods at all. It pretends to have them, but in reality simply blocking runtime internally, which is misleading and confusing. Tokio does have async file system methods because it uses Unless |
I don't agree with the premise that async APIs are harmful, but I agree that the implementation is incorrect. It should be using a thread pool instead of just calling a blocking syscall directly in the |
Async APIs are harmful if they are not in fact async. If they were implemented correctly I'd have nothing against them, but as it is right now, it is deeply problematic. As explained above, you can't do that as a trait on |
To be honest, I am not using any async file implementation when handling file I/O, as I mentioned above, async files almost have no benefits.
It is almost impossible, those methods do not require ownership. I just thought users should be aware that file locking is not a real "async task", but indeed, as @nazar-pc said, some users may ignore the documentation and naively believe they are actually async. |
I think the intentions were good, don't get me wrong, but having an async function that simply does blocking internally helps no one, there is zero benefit to it because anyone can simply call blocking function directly and get literally the same result, but at least they'll know they are blocking the runtime and could use There are a few options beyond simply removing async methods (which I'd prefer to make library smaller and easier to maintain):
None of the alternative options look particularly appealing to me to be completely honest. |
This is what I've opted to do in my own use case. |
Removing those APIs seems the simplest way, and let users handle this stuff by themselves. |
@al8n I'm fine with removing them and punting on a proper async implementation. Sorry if this whole adventure was not ultimately worth your time. I was just hoping to make the crate's API slightly nicer and didn't anticipate this type of issue. |
Right now, there are blocking methods of
AsyncFileExt
likelock_shared
andlock_exclusive
that could be async for implementations likeasync_fs
ortokio
.I can't see a simple way to do this as a user without wrapping
async_fs::File
in yet anotherArc
to pass it intoblocking::unblock
.The text was updated successfully, but these errors were encountered: