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

Initial port for wasmJs #170

Open
wants to merge 3 commits into
base: main
Choose a base branch
from
Open

Initial port for wasmJs #170

wants to merge 3 commits into from

Conversation

eygraber
Copy link

@eygraber eygraber commented Aug 30, 2024

I'd like to supply a common interface, but at first glance that looks like it might be a lot more work. I started work on commonizing the interface. Tests are currently failing; I am working on that now. Commonizing is done and tests are passing. This solution required having expect/actuals for primitives (so that js and wasmJs have a shared JsAny; see discussion here and elsewhere).

I'm trying to rework the API a bit so that none of that has to be exposed. Not sure if it is possible, but I'll give it a try.

It should be done now. There's still weirdness to support common code between js and wasmJs, but hopefully the API surface is a little better now, e.g. a lot of the type checking happen when you create a key instead of validating a key after it was created. There are still footguns available to the user, but they would be there anyways (e.g. unsafe casting).

It seems like there is partial support for null keys in js, but the spec seems to suggest that it shouldn't be allowed. I mirrored the null key support in wasmJs for now.

Fixes #157

  - I put this separate because it's a lot to have in the file
  - This would be a good candidate for context parameters when that is available
@cedrickcooke
Copy link
Collaborator

This is awesome, thank you! I'll start looking at it ASAP

@@ -1,17 +1,17 @@
package com.juul.indexeddb
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It looks like quite a few files are missing their package directive now. Just gonna leave this one comment to cover all of them.

Comment on lines +191 to +194
* This API is delicate. If you're passing in Kotlin objects directly, you're probably doing it wrong.
*
* Generally, you'll want to create an explicit `external interface` and pass that in, to guarantee that Kotlin
* doesn't mangle, prefix, or otherwise mess with your field names.
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Now that we have JsAny, I don't think we need the lines about delicate API and external interface. The same applies to all the other overloads of this function, and similar functions like put.

Comment on lines +18 to +19
public expect interface JsAny
public expect interface UnsafeJsAny : JsAny
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What's the difference between JsAny and UnsafeJsAny?

@PublishedApi
internal expect fun Array<IDBKey>.unsafeCast(): IDBKey

public expect interface JsAny
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Would love to see a doc comment on this explaining it. From my understanding, JsAny is meant as a marker of DB-safe types. Ideally something that includes theexternal interface MyRowType : JsAny use case.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

JsAny is really just there so that there can be common code for Js and Wasm. Wasm requires externals to extend JsAny, so that had to be represented in the common code.

I believe Kotlin is working on improving the interop story for Js and Wasm, so this can likely get simplified in the future.

Comment on lines +33 to +34
database.close()
deleteDatabase("auto-increment-keys")
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not opposed to losing the test wrapper if there was a real reason it had to go (I'm guessing it didn't play nice with wasm?), but if it's gone there needs to be a try ... finally for these cleanup lines. Otherwise a failing test leaves some garbage behind that can make future test runs (which should be successful) fail. Same goes for the other tests

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

There was an issue with Wasm, although I have a few ideas about how to make it work.

But worst case scenario I will add the try...finally

@cedrickcooke
Copy link
Collaborator

Absolutely heroic effort! A few comments, but nothing big/architectural, just some small details.

There's still weirdness to support common code between js and wasmJs, but hopefully the API surface is a little better now, e.g. a lot of the type checking happen when you create a key instead of validating a key after it was created. There are still footguns available to the user, but they would be there anyways (e.g. unsafe casting).

Love the improvements on this front. The footguns are now, at least, more explicit -- it looks a lot harder to accidentally pass something as JsAny than it was as dynamic.

@cedrickcooke cedrickcooke added the major Changes that should bump the MAJOR version number label Sep 4, 2024
@cedrickcooke cedrickcooke mentioned this pull request Sep 5, 2024
@mipastgt
Copy link

Any progress on this front?

@eygraber
Copy link
Author

I've been distracted, but I hope to get back to this in the coming weeks.

Copy link
Member

@twyatt twyatt left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is a bit outside my wheelhouse, but I'll try to review the code for readability/understandability more so than correctness of interacting with IndexedDB.
Just a couple comments so far. Overall, as @cedrickcooke mentioned, what an amazing effort on this! Thanks!

[![Slack](https://img.shields.io/badge/Slack-%23juul--libraries-ECB22E.svg?logo=&labelColor=611f69)](https://kotlinlang.slack.com/messages/juul-libraries/)

# Kotlin IndexedDB

A wrapper around [IndexedDB] which allows for access from Kotlin/JS code using `suspend` blocks and linear, non-callback based control flow.
A wrapper around [IndexedDB] which allows for access from Kotlin/JS or Kotlin/WasmJs code using `suspend` blocks and linear, non-callback based control flow.
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

nitpick: typically "Kotlin/Wasm" is used (without the "Js" attached).

Suggested change
A wrapper around [IndexedDB] which allows for access from Kotlin/JS or Kotlin/WasmJs code using `suspend` blocks and linear, non-callback based control flow.
A wrapper around [IndexedDB] which allows for access from Kotlin/JS or Kotlin/Wasm code using `suspend` blocks and linear, non-callback based control flow.

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I prefer the original wording because it is more precise. The use of IndexedDB is tied to the browser and thus to the wasmJS target and not WASM in general.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I did that to differentiate it from wasmWasi, although I guess the fact that it's a browser API should be enough to differentiate implicitly.

Comment on lines +5 to +6
import kotlin.js.unsafeCast
import kotlin.js.unsafeCast as jsUnsafeCast
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Curious the reason for the import aliasing?
Also, why both unsafeCast import and the alias for it too?

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Could be a mistake, I'll double check.

@apolostudio
Copy link

pinging this!

@eygraber
Copy link
Author

Sorry, haven't had much free time to get back to this. Hopefully will be able to soon.

@eygraber
Copy link
Author

Update: real life continues to get in the way of addressing comments 😞

I haven't forgotten about it, and will come back to it as soon as I'm able, but if anyone wants to pick up where I left off, feel free 😄

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
major Changes that should bump the MAJOR version number
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Provide Kotlin/Wasm support
5 participants