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

Advertising BLE service data from macOS #91

Open
avaidyam opened this issue Sep 26, 2023 · 10 comments
Open

Advertising BLE service data from macOS #91

avaidyam opened this issue Sep 26, 2023 · 10 comments

Comments

@avaidyam
Copy link

I'm fairly sure you should be able to do this on macOS (not iOS perhaps?). Here's an example. Specifically, CBAdvertisementDataServiceDataKey should be what you need.

@grishka
Copy link
Owner

grishka commented Sep 26, 2023

I modified my test app to pass a Data for that key (the docs say that its value is a [CBUUID:Data] dictionary so that's what I was trying), it's still complaining that it's not an allowed key.
Снимок экрана 2023-09-26 в 23 09 24

The service as seen by Android doesn't have any service data either:
Screenshot_20230926-230948

@avaidyam
Copy link
Author

avaidyam commented Sep 26, 2023

Ah, I see. It looks like the nearby library actually recognizes this limitation too with a workaround:

/**
 * Starts advertising service data in a way that is supported by CoreBluetooth.
 *
 * Since CoreBluetooth doesn't support setting the @c CBAdvertisementDataServiceDataKey key, the
 * service list is advertised using @c CBAdvertisementDataServiceUUIDsKey and the associated data is
 * advertised using @c CBAdvertisementDataLocalNameKey. Since @c CBAdvertisementDataLocalNameKey
 * does not support binary data, the value is base64 encoded and truncated if the resulting value is
 * longer than 22 bytes. This also means we can only support advertising a single service.
 *
 * @param serviceData A dictionary that contains service-specific advertisement data.
 * @param completionHandler Called on a private queue with @c nil if successfully started
 *                          advertising or an error if one has occured.
 */
- (void)startAdvertisingData:(NSDictionary<CBUUID *, NSData *> *)serviceData
           completionHandler:(nullable GNCStartAdvertisingCompletionHandler)completionHandler;

Another relevant snippet:

  // Apple devices don't support advertising service data, so Apple devices advertise a base64
  // encoded local name, while other devices advertise service data. Here we attempt to reconstruct
  // service data by decoding the local name. If successful, this is possibly a Nearby advertisement
  // on an Apple device.
  NSString *localName = advertisementData[CBAdvertisementDataLocalNameKey];

  // ...  

  // A Nearby Apple advertisement should only have a single service, so simply grab the first one if
  // it exists.
  NSArray<CBUUID *> *serviceUUIDs = advertisementData[CBAdvertisementDataServiceUUIDsKey];

Yet another relevant snippet:

  // Maps service UUIDs to their service data.
  //
  // For each platform should follow to set the service UUID(key) and service
  // data(value):
  //
  // iOS    : 16 bit service UUID (type=0x03) + LocalName data (type=0x08)
  // Windows: Service data (type=0x16)
  // Android: 16 bit service UUID (type=0x03) + Service data (type=0x16)
  absl::flat_hash_map<Uuid, nearby::ByteArray> service_data;

Can you see if this method works instead?

@grishka
Copy link
Owner

grishka commented Sep 26, 2023

I tried this:
Screenshot_20230927-004716
Didn't trigger neither the notification nor the MDNS service.

@avaidyam
Copy link
Author

avaidyam commented Sep 26, 2023

It looks like iOS/macOS BLE advertisement was only added to the google/nearby project last month by @bourdakos1 actually, so perhaps the Android/Play services system library isn't ready for it yet. Of note, it looks like migration of the Nearby Share code to the public project only started 2 weeks ago as well.

@grishka
Copy link
Owner

grishka commented Sep 26, 2023

Okay then. Maybe they're preparing to ship their (inevitably worse lol) Nearby Share app for macOS.

Another thing to keep in mind is that google/nearby is their universal cross-platform implementation in C++ that is maybe used by projects like Chromium, but Play Services uses another, closed-source (as far as I can tell) pure-Java implementation.

@grishka
Copy link
Owner

grishka commented Mar 20, 2024

I decompiled the latest (as of now) version of Play Services and found this:
Снимок экрана 2024-03-21 в 01 29 58
So this fallback is implemented, but it's behind what looks like a server flag, that defaults to false:
Снимок экрана 2024-03-21 в 01 28 41
(I renamed things here and there to make the code easier to understand)

@grishka
Copy link
Owner

grishka commented Mar 20, 2024

In case someone wants to look at it themselves, the bulk of code dealing with the reception of these BLE advertisements is in com.google.android.gms.nearby.sharing.discovery.FastInitiation

@grishka
Copy link
Owner

grishka commented Aug 26, 2024

I'm totally fine with using private APIs, I'll test this one, although I don't have my hopes up

@grishka
Copy link
Owner

grishka commented Aug 26, 2024

Oh these APIs are the ones I've already tested and they didn't work on modern macOS versions
#4 (comment)

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

No branches or pull requests

3 participants