-
Notifications
You must be signed in to change notification settings - Fork 125
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
app, cmd/gogio: add android foreground permission and service #67
base: main
Are you sure you want to change the base?
Conversation
I've tested this with my application; but feel that there could be better integration with the notification library gioui.org/x/notify, so that the foreground service can share the same notification channels, in particular so that the text of the background notification can be updated. I'm not sure that this change should depend on that package though, but I think that the Service class needs to be declared in the application manifest. Any help resolving/improving the above details is very much appreciated. |
Nice work, thank you.
Why share notifications with niotify? This change is for foreground work, niotify is for notifications. However, see my review comments for a suggestion to customize foreground notification content text. |
Oh, and remember to sign-off your change and squash commits. |
well, the foreground service requires a notification - the code here just sets a static string which isn't very appealing. Android applications using foreground services tend to have some kind of content displayed in the service notification and it would be useful to have a clean way to interact with the notification - possibly the StartForeground method should return or accept this as an argument - and this is where it starts to overlap with the functionality provided by niotify. |
edit: fixed |
596a0f2
to
87a7c2f
Compare
I've refactored this PR to address issues raised above and changed the API to include setting the notification ID, title, and text. |
87a7c2f
to
d295b22
Compare
c65a7be
to
917b1da
Compare
edit: stopped using startForegroundService because I saw some crash with resuming the application from background with same error as: transistorsoft/nativescript-background-geolocation-lt#121 (comment) . Instead startService is used, which fixes this crash. |
206d76d
to
37f30b7
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Leaving a few review comments for your follow-ups.
f0f3c2a
to
84b41ef
Compare
84b41ef
to
44a59e2
Compare
698540a
to
61f8a00
Compare
61f8a00
to
66d3f9e
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Looks pretty good! I have a handful of comments, but it seems we can soon merge this.
app/permission/foreground/main.go
Outdated
|
||
// Start is a no-op on Linux, Windows, macOS; Android will | ||
// display a notification during background work; iOS isn't supported. | ||
func Start(title, text string) (func(), error) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Let's just delete this function, leaving just app.Start
and this package only for the permission. Note in the documentation for app.Start
that this package must be imported for Start
to work. Also, make sure Start
returns an error if the user forgets the import.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ok, I moved code out of the app/permission path and added os_nandroid.go to add a no-op Start() method for platforms other than android so that calls to Start() and its returned stop method have no effect.
|
||
static Intent startForegroundService(Context ctx, String title, String text) throws ClassNotFoundException { | ||
Intent intent = new Intent(); | ||
intent.setClass(ctx, ctx.getClassLoader().loadClass("org/gioui/GioForegroundService")); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Why not just GioForegroundService.class
instead of loadClass...
?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think that I tried doing that but the class doesn't exist unless it was first loaded.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Please post the error message you get. It doesn't make sense to me that org.gioui.Gio
can be loaded but org.gioui.GioForegroundService
can't.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
ActivityManager
Unable to start service Intent
{ cmp=org.mixnetworks.katzen/
org.gioui.GioForegroundServie (has extras) } U=0:
not found
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There must be more to the error than what you pasted. A stack trace or similar that can tell us why the class cannot be loaded.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
do you know how to get android to produce a stack trace here?
app/os_android.go
Outdated
var foregroundService struct { | ||
intent C.jobject | ||
mu sync.Mutex | ||
count int |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is much better than before, but users can still call their stop functions more than once. I suggest a map[*int]bool
where each call to app.Start
registers a new(int)
token in the map. The stop function then returns eearly if the map no longer contains the token.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm afraid I don't see how users can call their stop functions more than once as the returned function checks count while ohlding the lock in order to prevent stopService from being called more than once.
There should be only one instance of the GioForegroundService, so multiple calls to Start() only indicate that some part of the application is doing something that needs to continue running, and when all of the associated stop func() have been called should the service be actually stopped.
I don't see why a map is useful here, where a counter suffices.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
A program may have two unrelated background jobs, say (a) uploading an image and (b) checking for new messages. If they run concurrently, the job count is incremented to 2. A programming mistake causes (a) to calls its stop function twice, decrementing the count to 0, even though (b) is still in progress. The resulting missing notification is very hard to notice.
A map allows us to detect and ignore multiple calls to any particular stop function. In other words, it is no longer a (subtle) mistake to call your stop function twice.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
ok, this makes sense to me. Thank you for the explanation.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
(changes made)
fece051
to
e2f1cfa
Compare
For some reason github wont let me comment on the change requests directly any more. I moved code out of gioui.org/app/permissions, but I was not able to find a solution to check that gioui.org/app/permissions/foreground was imported when built. For example, I thought that BuildInfo might have package-level resolution, but it does not:
|
e2f1cfa
to
c517124
Compare
I don't think you need to check for the import. It should be enough to detect the Java exception/error when trying to open a foreground notification. |
Good Idea. I tried to catch java.lang.SecurityException, but the exception is not raised to the caller of android.startForegroundService, as it is thrown in the method onStartCommand which is called by the android environment after the service is started. Do you have any ideas how to handle this? |
One idea is to make |
7fd3b23
to
079448b
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I somehow failed to send this review out after writing it. I apologize.
|
||
static Intent startForegroundService(Context ctx, String title, String text) throws ClassNotFoundException { | ||
Intent intent = new Intent(); | ||
intent.setClass(ctx, ctx.getClassLoader().loadClass("org/gioui/GioForegroundService")); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There must be more to the error than what you pasted. A stack trace or similar that can tell us why the class cannot be loaded.
foregroundService.intent = C.jni_NewGlobalRef(env, foregroundService.intent) | ||
|
||
} else { | ||
panic(err) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
You should propagate this error to the caller. In particular when the XXX issue above is resolved so that startForegroundService
can fail for configuration reasons.
app/os_android.go
Outdated
// Start starts the foreground service | ||
func Start(title, text string) (stop func(), err error) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The documentation needs expanding. To make the documentation visible for everyone, rename this function startForeground
and add a documented StartForeground
in package app
that calls it.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
done
app/os_nandroid.go
Outdated
package app | ||
|
||
// Start is a no-op on platforms other than android | ||
func Start(title, text string) (stop func(), err error) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
As described above, rename this function to startForeground
.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
done
079448b
to
1c97118
Compare
5a91e41
to
a1625b7
Compare
This adds the permission android.permission.FOREGROUND_SERVICE and adds GioForegroundService which creates the tray Notification necessary to implement the Foreground Service. This adds the method Start to package app, which on android, notifies the system that the program will perform background work and that it shouldn't be killed. The foreground service is stopped using the cancel function returned by Start(). If multiple calls to Start are made, the foreground service will not be stopped until the final cancel function has been called. See https://developer.android.com/guide/components/foreground-services and https://developer.android.com/training/notify-user/build-notification Signed-off-by: Masala <masala@riseup.net>
0d543a0
to
1686874
Compare
67c77c9
to
46cc311
Compare
This adds the permission android.permission.FOREGROUND_SERVICE and adds
GioForegroundService which creates the tray Notification necessary to
implement the Foreground Service. The package foreground includes the method
StartForeground, which on android, notifies the system that the program
will perform background work and that it shouldn't be killed. It returns
a channel that should be closed when the background work is complete.
See https://developer.android.com/guide/components/foreground-services
and https://developer.android.com/training/notify-user/build-notification