ApiUtils.java 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559
  1. /*
  2. * Nextcloud Talk application
  3. *
  4. * @author Mario Danic
  5. * @author Marcel Hibbe
  6. * @author Tim Krüger
  7. * Copyright (C) 2021 Tim Krüger <t@timkrueger.me>
  8. * Copyright (C) 2021-2022 Marcel Hibbe <dev@mhibbe.de>
  9. * Copyright (C) 2017-2018 Mario Danic <mario@lovelyhq.com>
  10. *
  11. * This program is free software: you can redistribute it and/or modify
  12. * it under the terms of the GNU General Public License as published by
  13. * the Free Software Foundation, either version 3 of the License, or
  14. * at your option) any later version.
  15. *
  16. * This program is distributed in the hope that it will be useful,
  17. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  18. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  19. * GNU General Public License for more details.
  20. *
  21. * You should have received a copy of the GNU General Public License
  22. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  23. */
  24. package com.nextcloud.talk.utils;
  25. import android.net.Uri;
  26. import android.text.TextUtils;
  27. import android.util.Log;
  28. import com.nextcloud.talk.BuildConfig;
  29. import com.nextcloud.talk.R;
  30. import com.nextcloud.talk.application.NextcloudTalkApplication;
  31. import com.nextcloud.talk.data.user.model.User;
  32. import com.nextcloud.talk.models.RetrofitBucket;
  33. import com.nextcloud.talk.utils.database.user.CapabilitiesUtilNew;
  34. import java.nio.charset.StandardCharsets;
  35. import java.util.HashMap;
  36. import java.util.Map;
  37. import androidx.annotation.NonNull;
  38. import androidx.annotation.Nullable;
  39. import okhttp3.Credentials;
  40. public class ApiUtils {
  41. public static final int APIv1 = 1;
  42. public static final int APIv2 = 2;
  43. public static final int APIv3 = 3;
  44. public static final int APIv4 = 4;
  45. public static final int AVATAR_SIZE_BIG = 512;
  46. public static final int AVATAR_SIZE_SMALL = 64;
  47. private static final String TAG = "ApiUtils";
  48. private static final String ocsApiVersion = "/ocs/v2.php";
  49. private static final String spreedApiVersion = "/apps/spreed/api/v1";
  50. private static final String spreedApiBase = ocsApiVersion + "/apps/spreed/api/v";
  51. private static final String userAgent = "Mozilla/5.0 (Android) Nextcloud-Talk v";
  52. public static String getUserAgent() {
  53. return userAgent + BuildConfig.VERSION_NAME;
  54. }
  55. /**
  56. * @deprecated This is only supported on API v1-3, in API v4+ please use
  57. * {@link ApiUtils#getUrlForAttendees(int, String, String)} instead.
  58. */
  59. @Deprecated
  60. public static String getUrlForRemovingParticipantFromConversation(String baseUrl, String roomToken, boolean isGuest) {
  61. String url = getUrlForParticipants(APIv1, baseUrl, roomToken);
  62. if (isGuest) {
  63. url += "/guests";
  64. }
  65. return url;
  66. }
  67. public static RetrofitBucket getRetrofitBucketForContactsSearch(String baseUrl, @Nullable String searchQuery) {
  68. RetrofitBucket retrofitBucket = new RetrofitBucket();
  69. retrofitBucket.setUrl(baseUrl + ocsApiVersion + "/apps/files_sharing/api/v1/sharees");
  70. Map<String, String> queryMap = new HashMap<>();
  71. if (searchQuery == null) {
  72. searchQuery = "";
  73. }
  74. queryMap.put("format", "json");
  75. queryMap.put("search", searchQuery);
  76. queryMap.put("itemType", "call");
  77. retrofitBucket.setQueryMap(queryMap);
  78. return retrofitBucket;
  79. }
  80. public static String getUrlForFilePreviewWithRemotePath(String baseUrl, String remotePath, int px) {
  81. return baseUrl + "/index.php/core/preview.png?file="
  82. + Uri.encode(remotePath, "UTF-8")
  83. + "&x=" + px + "&y=" + px + "&a=1&mode=cover&forceIcon=1";
  84. }
  85. public static String getUrlForFilePreviewWithFileId(String baseUrl, String fileId, int px) {
  86. return baseUrl + "/index.php/core/preview?fileId="
  87. + fileId + "&x=" + px + "&y=" + px + "&a=1&mode=cover&forceIcon=1";
  88. }
  89. public static String getSharingUrl(String baseUrl) {
  90. return baseUrl + ocsApiVersion + "/apps/files_sharing/api/v1/shares";
  91. }
  92. public static RetrofitBucket getRetrofitBucketForContactsSearchFor14(String baseUrl, @Nullable String searchQuery) {
  93. RetrofitBucket retrofitBucket = getRetrofitBucketForContactsSearch(baseUrl, searchQuery);
  94. retrofitBucket.setUrl(baseUrl + ocsApiVersion + "/core/autocomplete/get");
  95. retrofitBucket.getQueryMap().put("itemId", "new");
  96. return retrofitBucket;
  97. }
  98. public static String getUrlForCapabilities(String baseUrl) {
  99. return baseUrl + ocsApiVersion + "/cloud/capabilities";
  100. }
  101. public static int getCallApiVersion(User capabilities, int[] versions) throws NoSupportedApiException {
  102. return getConversationApiVersion(capabilities, versions);
  103. }
  104. public static int getConversationApiVersion(User user, int[] versions) throws NoSupportedApiException {
  105. boolean hasApiV4 = false;
  106. for (int version : versions) {
  107. hasApiV4 |= version == APIv4;
  108. }
  109. if (!hasApiV4) {
  110. Exception e = new Exception("Api call did not try conversation-v4 api");
  111. Log.d(TAG, e.getMessage(), e);
  112. }
  113. for (int version : versions) {
  114. if (user.hasSpreedFeatureCapability("conversation-v" + version)) {
  115. return version;
  116. }
  117. // Fallback for old API versions
  118. if ((version == APIv1 || version == APIv2)) {
  119. if (user.hasSpreedFeatureCapability("conversation-v2")) {
  120. return version;
  121. }
  122. if (version == APIv1 &&
  123. user.hasSpreedFeatureCapability("mention-flag") &&
  124. !user.hasSpreedFeatureCapability("conversation-v4")) {
  125. return version;
  126. }
  127. }
  128. }
  129. throw new NoSupportedApiException();
  130. }
  131. public static int getSignalingApiVersion(User user, int[] versions) throws NoSupportedApiException {
  132. for (int version : versions) {
  133. if (CapabilitiesUtilNew.hasSpreedFeatureCapability(user, "signaling-v" + version)) {
  134. return version;
  135. }
  136. if (version == APIv2 &&
  137. CapabilitiesUtilNew.hasSpreedFeatureCapability(user, "sip-support") &&
  138. !CapabilitiesUtilNew.hasSpreedFeatureCapability(user, "signaling-v3")) {
  139. return version;
  140. }
  141. if (version == APIv1 &&
  142. !CapabilitiesUtilNew.hasSpreedFeatureCapability(user, "signaling-v3")) {
  143. // Has no capability, we just assume it is always there when there is no v3 or later
  144. return version;
  145. }
  146. }
  147. throw new NoSupportedApiException();
  148. }
  149. public static int getChatApiVersion(User user, int[] versions) throws NoSupportedApiException {
  150. for (int version : versions) {
  151. if (version == APIv1 && CapabilitiesUtilNew.hasSpreedFeatureCapability(user, "chat-v2")) {
  152. // Do not question that chat-v2 capability shows the availability of api/v1/ endpoint *see no evil*
  153. return version;
  154. }
  155. }
  156. throw new NoSupportedApiException();
  157. }
  158. protected static String getUrlForApi(int version, String baseUrl) {
  159. return baseUrl + spreedApiBase + version;
  160. }
  161. public static String getUrlForRooms(int version, String baseUrl) {
  162. return getUrlForApi(version, baseUrl) + "/room";
  163. }
  164. public static String getUrlForRoom(int version, String baseUrl, String token) {
  165. return getUrlForRooms(version, baseUrl) + "/" + token;
  166. }
  167. public static String getUrlForAttendees(int version, String baseUrl, String token) {
  168. return getUrlForRoom(version, baseUrl, token) + "/attendees";
  169. }
  170. public static String getUrlForParticipants(int version, String baseUrl, String token) {
  171. if (token == null || token.isEmpty()) {
  172. Log.e(TAG, "token was null or empty");
  173. }
  174. return getUrlForRoom(version, baseUrl, token) + "/participants";
  175. }
  176. public static String getUrlForParticipantsActive(int version, String baseUrl, String token) {
  177. return getUrlForParticipants(version, baseUrl, token) + "/active";
  178. }
  179. public static String getUrlForParticipantsSelf(int version, String baseUrl, String token) {
  180. return getUrlForParticipants(version, baseUrl, token) + "/self";
  181. }
  182. public static String getUrlForParticipantsResendInvitations(int version, String baseUrl, String token) {
  183. return getUrlForParticipants(version, baseUrl, token) + "/resend-invitations";
  184. }
  185. public static String getUrlForRoomFavorite(int version, String baseUrl, String token) {
  186. return getUrlForRoom(version, baseUrl, token) + "/favorite";
  187. }
  188. public static String getUrlForRoomModerators(int version, String baseUrl, String token) {
  189. return getUrlForRoom(version, baseUrl, token) + "/moderators";
  190. }
  191. public static String getUrlForRoomNotificationLevel(int version, String baseUrl, String token) {
  192. return getUrlForRoom(version, baseUrl, token) + "/notify";
  193. }
  194. public static String getUrlForRoomPublic(int version, String baseUrl, String token) {
  195. return getUrlForRoom(version, baseUrl, token) + "/public";
  196. }
  197. public static String getUrlForRoomPassword(int version, String baseUrl, String token) {
  198. return getUrlForRoom(version, baseUrl, token) + "/password";
  199. }
  200. public static String getUrlForRoomReadOnlyState(int version, String baseUrl, String token) {
  201. return getUrlForRoom(version, baseUrl, token) + "/read-only";
  202. }
  203. public static String getUrlForRoomWebinaryLobby(int version, String baseUrl, String token) {
  204. return getUrlForRoom(version, baseUrl, token) + "/webinar/lobby";
  205. }
  206. public static String getUrlForRoomNotificationCalls(int version, String baseUrl, String token) {
  207. return getUrlForRoom(version, baseUrl, token) + "/notify-calls";
  208. }
  209. public static String getUrlForCall(int version, String baseUrl, String token) {
  210. return getUrlForApi(version, baseUrl) + "/call/" + token;
  211. }
  212. public static String getUrlForChat(int version, String baseUrl, String token) {
  213. return getUrlForApi(version, baseUrl) + "/chat/" + token;
  214. }
  215. public static String getUrlForMentionSuggestions(int version, String baseUrl, String token) {
  216. return getUrlForChat(version, baseUrl, token) + "/mentions";
  217. }
  218. public static String getUrlForChatMessage(int version, String baseUrl, String token, String messageId) {
  219. return getUrlForChat(version, baseUrl, token) + "/" + messageId;
  220. }
  221. public static String getUrlForChatSharedItems(int version, String baseUrl, String token) {
  222. return getUrlForChat(version, baseUrl, token) + "/share";
  223. }
  224. public static String getUrlForChatSharedItemsOverview(int version, String baseUrl, String token) {
  225. return getUrlForChatSharedItems(version, baseUrl, token) + "/overview";
  226. }
  227. public static String getUrlForSignaling(int version, String baseUrl) {
  228. return getUrlForApi(version, baseUrl) + "/signaling";
  229. }
  230. public static String getUrlForSignalingBackend(int version, String baseUrl) {
  231. return getUrlForSignaling(version, baseUrl) + "/backend";
  232. }
  233. public static String getUrlForSignalingSettings(int version, String baseUrl) {
  234. return getUrlForSignaling(version, baseUrl) + "/settings";
  235. }
  236. public static String getUrlForSignaling(int version, String baseUrl, String token) {
  237. return getUrlForSignaling(version, baseUrl) + "/" + token;
  238. }
  239. public static String getUrlForOpenConversations(int version, String baseUrl) {
  240. return getUrlForApi(version, baseUrl) + "/listed-room";
  241. }
  242. public static RetrofitBucket getRetrofitBucketForCreateRoom(int version, String baseUrl, String roomType,
  243. @Nullable String source,
  244. @Nullable String invite,
  245. @Nullable String conversationName) {
  246. RetrofitBucket retrofitBucket = new RetrofitBucket();
  247. retrofitBucket.setUrl(getUrlForRooms(version, baseUrl));
  248. Map<String, String> queryMap = new HashMap<>();
  249. queryMap.put("roomType", roomType);
  250. if (invite != null) {
  251. queryMap.put("invite", invite);
  252. }
  253. if (source != null) {
  254. queryMap.put("source", source);
  255. }
  256. if (conversationName != null) {
  257. queryMap.put("roomName", conversationName);
  258. }
  259. retrofitBucket.setQueryMap(queryMap);
  260. return retrofitBucket;
  261. }
  262. public static RetrofitBucket getRetrofitBucketForAddParticipant(int version, String baseUrl, String token, String user) {
  263. RetrofitBucket retrofitBucket = new RetrofitBucket();
  264. retrofitBucket.setUrl(getUrlForParticipants(version, baseUrl, token));
  265. Map<String, String> queryMap = new HashMap<>();
  266. queryMap.put("newParticipant", user);
  267. retrofitBucket.setQueryMap(queryMap);
  268. return retrofitBucket;
  269. }
  270. public static RetrofitBucket getRetrofitBucketForAddParticipantWithSource(
  271. int version,
  272. String baseUrl,
  273. String token,
  274. String source,
  275. String id
  276. ) {
  277. RetrofitBucket retrofitBucket = getRetrofitBucketForAddParticipant(version, baseUrl, token, id);
  278. retrofitBucket.getQueryMap().put("source", source);
  279. return retrofitBucket;
  280. }
  281. public static String getUrlForUserProfile(String baseUrl) {
  282. return baseUrl + ocsApiVersion + "/cloud/user";
  283. }
  284. public static String getUrlForUserData(String baseUrl, String userId) {
  285. return baseUrl + ocsApiVersion + "/cloud/users/" + userId;
  286. }
  287. public static String getUrlForUserSettings(String baseUrl) {
  288. // FIXME Introduce API version
  289. return baseUrl + ocsApiVersion + spreedApiVersion + "/settings/user";
  290. }
  291. public static String getUrlPostfixForStatus() {
  292. return "/status.php";
  293. }
  294. public static String getUrlForAvatar(String baseUrl, String name, boolean requestBigSize) {
  295. int avatarSize = requestBigSize ? AVATAR_SIZE_BIG : AVATAR_SIZE_SMALL;
  296. return baseUrl + "/index.php/avatar/" + Uri.encode(name) + "/" + avatarSize;
  297. }
  298. public static String getUrlForGuestAvatar(String baseUrl, String name, boolean requestBigSize) {
  299. int avatarSize = requestBigSize ? AVATAR_SIZE_BIG : AVATAR_SIZE_SMALL;
  300. return baseUrl + "/index.php/avatar/guest/" + Uri.encode(name) + "/" + avatarSize;
  301. }
  302. public static String getUrlForConversationAvatar(int version, String baseUrl, String token) {
  303. return getUrlForRoom(version, baseUrl, token) + "/avatar";
  304. }
  305. public static String getUrlForConversationAvatarWithVersion(int version, String baseUrl, String token,
  306. boolean isDark,
  307. String avatarVersion) {
  308. String isDarkString = "";
  309. if (isDark) {
  310. isDarkString = "/dark";
  311. }
  312. String avatarVersionString = "";
  313. if (avatarVersion != null) {
  314. avatarVersionString = "?avatarVersion=" + avatarVersion;
  315. }
  316. return getUrlForRoom(version, baseUrl, token) + "/avatar" + isDarkString + avatarVersionString;
  317. }
  318. public static String getCredentials(String username, String token) {
  319. if (TextUtils.isEmpty(username) && TextUtils.isEmpty(token)) {
  320. return null;
  321. }
  322. return Credentials.basic(username, token, StandardCharsets.UTF_8);
  323. }
  324. public static String getUrlNextcloudPush(String baseUrl) {
  325. return baseUrl + ocsApiVersion + "/apps/notifications/api/v2/push";
  326. }
  327. public static String getUrlPushProxy() {
  328. return NextcloudTalkApplication.Companion.getSharedApplication().
  329. getApplicationContext().getResources().getString(R.string.nc_push_server_url) + "/devices";
  330. }
  331. // see https://github.com/nextcloud/notifications/blob/master/docs/ocs-endpoint-v2.md
  332. public static String getUrlForNcNotificationWithId(String baseUrl, String notificationId) {
  333. return baseUrl + ocsApiVersion + "/apps/notifications/api/v2/notifications/" + notificationId;
  334. }
  335. public static String getUrlForSearchByNumber(String baseUrl) {
  336. return baseUrl + ocsApiVersion + "/cloud/users/search/by-phone";
  337. }
  338. public static String getUrlForFileUpload(String baseUrl, String user, String remotePath) {
  339. return baseUrl + "/remote.php/dav/files/" + user + remotePath;
  340. }
  341. public static String getUrlForChunkedUpload(String baseUrl, String user) {
  342. return baseUrl + "/remote.php/dav/uploads/" + user;
  343. }
  344. public static String getUrlForFileDownload(String baseUrl, String user, String remotePath) {
  345. return baseUrl + "/remote.php/dav/files/" + user + "/" + remotePath;
  346. }
  347. public static String getUrlForTempAvatar(String baseUrl) {
  348. return baseUrl + ocsApiVersion + "/apps/spreed/temp-user-avatar";
  349. }
  350. public static String getUrlForUserFields(String baseUrl) {
  351. return baseUrl + ocsApiVersion + "/cloud/user/fields";
  352. }
  353. public static String getUrlToSendLocation(int version, String baseUrl, String roomToken) {
  354. return getUrlForChat(version, baseUrl, roomToken) + "/share";
  355. }
  356. public static String getUrlForHoverCard(String baseUrl, String userId) {
  357. return baseUrl + ocsApiVersion +
  358. "/hovercard/v1/" + userId;
  359. }
  360. public static String getUrlForChatReadMarker(int version, String baseUrl, String roomToken) {
  361. return getUrlForChat(version, baseUrl, roomToken) + "/read";
  362. }
  363. /*
  364. * OCS Status API
  365. */
  366. public static String getUrlForStatus(String baseUrl) {
  367. return baseUrl + ocsApiVersion + "/apps/user_status/api/v1/user_status";
  368. }
  369. public static String getUrlForSetStatusType(String baseUrl) {
  370. return getUrlForStatus(baseUrl) + "/status";
  371. }
  372. public static String getUrlForPredefinedStatuses(String baseUrl) {
  373. return baseUrl + ocsApiVersion + "/apps/user_status/api/v1/predefined_statuses";
  374. }
  375. public static String getUrlForStatusMessage(String baseUrl) {
  376. return getUrlForStatus(baseUrl) + "/message";
  377. }
  378. public static String getUrlForSetCustomStatus(String baseUrl) {
  379. return baseUrl + ocsApiVersion + "/apps/user_status/api/v1/user_status/message/custom";
  380. }
  381. public static String getUrlForSetPredefinedStatus(String baseUrl) {
  382. return baseUrl + ocsApiVersion + "/apps/user_status/api/v1/user_status/message/predefined";
  383. }
  384. public static String getUrlForUserStatuses(String baseUrl) {
  385. return baseUrl + ocsApiVersion + "/apps/user_status/api/v1/statuses";
  386. }
  387. public static String getUrlForMessageReaction(String baseUrl,
  388. String roomToken,
  389. String messageId) {
  390. return baseUrl + ocsApiVersion + spreedApiVersion + "/reaction/" + roomToken + "/" + messageId;
  391. }
  392. @NonNull
  393. public static String getUrlForUnifiedSearch(@NonNull String baseUrl, @NonNull String providerId) {
  394. return baseUrl + ocsApiVersion + "/search/providers/" + providerId + "/search";
  395. }
  396. public static String getUrlForPoll(String baseUrl,
  397. String roomToken,
  398. String pollId) {
  399. return getUrlForPoll(baseUrl, roomToken) + "/" + pollId;
  400. }
  401. public static String getUrlForPoll(String baseUrl,
  402. String roomToken) {
  403. return baseUrl + ocsApiVersion + spreedApiVersion + "/poll/" + roomToken;
  404. }
  405. public static String getUrlForMessageExpiration(int version, String baseUrl, String token) {
  406. return getUrlForRoom(version, baseUrl, token) + "/message-expiration";
  407. }
  408. public static String getUrlForOpenGraph(String baseUrl) {
  409. return baseUrl + ocsApiVersion + "/references/resolve";
  410. }
  411. public static String getUrlForRecording(int version, String baseUrl, String token) {
  412. return getUrlForApi(version, baseUrl) + "/recording/" + token;
  413. }
  414. public static String getUrlForRequestAssistance(int version, String baseUrl, String token) {
  415. return getUrlForApi(version, baseUrl) + "/breakout-rooms/" + token + "/request-assistance";
  416. }
  417. public static String getUrlForConversationDescription(int version, String baseUrl, String token) {
  418. return getUrlForRoom(version, baseUrl, token) + "/description";
  419. }
  420. public static String getUrlForTranslation(String baseUrl) {
  421. return baseUrl + ocsApiVersion + "/translation/translate";
  422. }
  423. public static String getUrlForLanguages(String baseUrl) {
  424. return baseUrl + ocsApiVersion + "/translation/languages";
  425. }
  426. public static String getUrlForReminder(User user, String roomToken, String messageId, int version) {
  427. String url = ApiUtils.getUrlForChatMessage(version, user.getBaseUrl(), roomToken, messageId);
  428. return url + "/reminder";
  429. }
  430. public static String getUrlForRecordingConsent(int version, String baseUrl, String token) {
  431. return getUrlForRoom(version, baseUrl, token) + "/recording-consent";
  432. }
  433. public static String getUrlForInvitation(String baseUrl) {
  434. return baseUrl + ocsApiVersion + spreedApiVersion + "/federation/invitation";
  435. }
  436. public static String getUrlForInvitationAccept(String baseUrl, int id) {
  437. return getUrlForInvitation(baseUrl) + "/" + id;
  438. }
  439. public static String getUrlForInvitationReject(String baseUrl, int id) {
  440. return getUrlForInvitation(baseUrl) + "/" + id;
  441. }
  442. }