ApiUtils.java 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505
  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.util.HashMap;
  35. import java.util.Map;
  36. import androidx.annotation.NonNull;
  37. import androidx.annotation.Nullable;
  38. import okhttp3.Credentials;
  39. public class ApiUtils {
  40. public static final int APIv1 = 1;
  41. public static final int APIv2 = 2;
  42. public static final int APIv3 = 3;
  43. public static final int APIv4 = 4;
  44. public static final int AVATAR_SIZE_BIG = 512;
  45. public static final int AVATAR_SIZE_SMALL = 64;
  46. private static final String TAG = "ApiUtils";
  47. private static final String ocsApiVersion = "/ocs/v2.php";
  48. private static final String spreedApiVersion = "/apps/spreed/api/v1";
  49. private static final String spreedApiBase = ocsApiVersion + "/apps/spreed/api/v";
  50. private static final String userAgent = "Mozilla/5.0 (Android) Nextcloud-Talk v";
  51. public static String getUserAgent() {
  52. return userAgent + BuildConfig.VERSION_NAME;
  53. }
  54. /**
  55. * @deprecated This is only supported on API v1-3, in API v4+ please use {@link ApiUtils#getUrlForAttendees(int,
  56. * String, String)} instead.
  57. */
  58. @Deprecated
  59. public static String getUrlForRemovingParticipantFromConversation(String baseUrl, String roomToken, boolean isGuest) {
  60. String url = getUrlForParticipants(APIv1, baseUrl, roomToken);
  61. if (isGuest) {
  62. url += "/guests";
  63. }
  64. return url;
  65. }
  66. public static RetrofitBucket getRetrofitBucketForContactsSearch(String baseUrl, @Nullable String searchQuery) {
  67. RetrofitBucket retrofitBucket = new RetrofitBucket();
  68. retrofitBucket.setUrl(baseUrl + ocsApiVersion + "/apps/files_sharing/api/v1/sharees");
  69. Map<String, String> queryMap = new HashMap<>();
  70. if (searchQuery == null) {
  71. searchQuery = "";
  72. }
  73. queryMap.put("format", "json");
  74. queryMap.put("search", searchQuery);
  75. queryMap.put("itemType", "call");
  76. retrofitBucket.setQueryMap(queryMap);
  77. return retrofitBucket;
  78. }
  79. public static String getUrlForFilePreviewWithRemotePath(String baseUrl, String remotePath, int px) {
  80. return baseUrl + "/index.php/core/preview.png?file="
  81. + Uri.encode(remotePath, "UTF-8")
  82. + "&x=" + px + "&y=" + px + "&a=1&mode=cover&forceIcon=1";
  83. }
  84. public static String getUrlForFilePreviewWithFileId(String baseUrl, String fileId, int px) {
  85. return baseUrl + "/index.php/core/preview?fileId="
  86. + fileId + "&x=" + px + "&y=" + px + "&a=1&mode=cover&forceIcon=1";
  87. }
  88. public static String getSharingUrl(String baseUrl) {
  89. return baseUrl + ocsApiVersion + "/apps/files_sharing/api/v1/shares";
  90. }
  91. public static RetrofitBucket getRetrofitBucketForContactsSearchFor14(String baseUrl, @Nullable String searchQuery) {
  92. RetrofitBucket retrofitBucket = getRetrofitBucketForContactsSearch(baseUrl, searchQuery);
  93. retrofitBucket.setUrl(baseUrl + ocsApiVersion + "/core/autocomplete/get");
  94. retrofitBucket.getQueryMap().put("itemId", "new");
  95. return retrofitBucket;
  96. }
  97. public static String getUrlForCapabilities(String baseUrl) {
  98. return baseUrl + ocsApiVersion + "/cloud/capabilities";
  99. }
  100. public static int getCallApiVersion(User capabilities, int[] versions) throws NoSupportedApiException {
  101. return getConversationApiVersion(capabilities, versions);
  102. }
  103. public static int getConversationApiVersion(User user, int[] versions) throws NoSupportedApiException {
  104. boolean hasApiV4 = false;
  105. for (int version : versions) {
  106. hasApiV4 |= version == APIv4;
  107. }
  108. if (!hasApiV4) {
  109. Exception e = new Exception("Api call did not try conversation-v4 api");
  110. Log.d(TAG, e.getMessage(), e);
  111. }
  112. for (int version : versions) {
  113. if (user.hasSpreedFeatureCapability("conversation-v" + version)) {
  114. return version;
  115. }
  116. // Fallback for old API versions
  117. if ((version == APIv1 || version == APIv2)) {
  118. if (user.hasSpreedFeatureCapability("conversation-v2")) {
  119. return version;
  120. }
  121. if (version == APIv1 &&
  122. user.hasSpreedFeatureCapability("mention-flag") &&
  123. !user.hasSpreedFeatureCapability("conversation-v4")) {
  124. return version;
  125. }
  126. }
  127. }
  128. throw new NoSupportedApiException();
  129. }
  130. public static int getSignalingApiVersion(User user, int[] versions) throws NoSupportedApiException {
  131. for (int version : versions) {
  132. if (CapabilitiesUtilNew.hasSpreedFeatureCapability(user, "signaling-v" + version)) {
  133. return version;
  134. }
  135. if (version == APIv2 &&
  136. CapabilitiesUtilNew.hasSpreedFeatureCapability(user, "sip-support") &&
  137. !CapabilitiesUtilNew.hasSpreedFeatureCapability(user, "signaling-v3")) {
  138. return version;
  139. }
  140. if (version == APIv1 &&
  141. !CapabilitiesUtilNew.hasSpreedFeatureCapability(user, "signaling-v3")) {
  142. // Has no capability, we just assume it is always there when there is no v3 or later
  143. return version;
  144. }
  145. }
  146. throw new NoSupportedApiException();
  147. }
  148. public static int getChatApiVersion(User user, int[] versions) throws NoSupportedApiException {
  149. for (int version : versions) {
  150. if (version == APIv1 && CapabilitiesUtilNew.hasSpreedFeatureCapability(user, "chat-v2")) {
  151. // Do not question that chat-v2 capability shows the availability of api/v1/ endpoint *see no evil*
  152. return version;
  153. }
  154. }
  155. throw new NoSupportedApiException();
  156. }
  157. protected static String getUrlForApi(int version, String baseUrl) {
  158. return baseUrl + spreedApiBase + version;
  159. }
  160. public static String getUrlForRooms(int version, String baseUrl) {
  161. return getUrlForApi(version, baseUrl) + "/room";
  162. }
  163. public static String getUrlForRoom(int version, String baseUrl, String token) {
  164. return getUrlForRooms(version, baseUrl) + "/" + token;
  165. }
  166. public static String getUrlForAttendees(int version, String baseUrl, String token) {
  167. return getUrlForRoom(version, baseUrl, token) + "/attendees";
  168. }
  169. public static String getUrlForParticipants(int version, String baseUrl, String token) {
  170. if (token == null || token.isEmpty()) {
  171. Log.e(TAG, "token was null or empty");
  172. }
  173. return getUrlForRoom(version, baseUrl, token) + "/participants";
  174. }
  175. public static String getUrlForParticipantsActive(int version, String baseUrl, String token) {
  176. return getUrlForParticipants(version, baseUrl, token) + "/active";
  177. }
  178. public static String getUrlForParticipantsSelf(int version, String baseUrl, String token) {
  179. return getUrlForParticipants(version, baseUrl, token) + "/self";
  180. }
  181. public static String getUrlForParticipantsResendInvitations(int version, String baseUrl, String token) {
  182. return getUrlForParticipants(version, baseUrl, token) + "/resend-invitations";
  183. }
  184. public static String getUrlForRoomFavorite(int version, String baseUrl, String token) {
  185. return getUrlForRoom(version, baseUrl, token) + "/favorite";
  186. }
  187. public static String getUrlForRoomModerators(int version, String baseUrl, String token) {
  188. return getUrlForRoom(version, baseUrl, token) + "/moderators";
  189. }
  190. public static String getUrlForRoomNotificationLevel(int version, String baseUrl, String token) {
  191. return getUrlForRoom(version, baseUrl, token) + "/notify";
  192. }
  193. public static String getUrlForRoomPublic(int version, String baseUrl, String token) {
  194. return getUrlForRoom(version, baseUrl, token) + "/public";
  195. }
  196. public static String getUrlForRoomPassword(int version, String baseUrl, String token) {
  197. return getUrlForRoom(version, baseUrl, token) + "/password";
  198. }
  199. public static String getUrlForRoomReadOnlyState(int version, String baseUrl, String token) {
  200. return getUrlForRoom(version, baseUrl, token) + "/read-only";
  201. }
  202. public static String getUrlForRoomWebinaryLobby(int version, String baseUrl, String token) {
  203. return getUrlForRoom(version, baseUrl, token) + "/webinary/lobby";
  204. }
  205. public static String getUrlForRoomNotificationCalls(int version, String baseUrl, String token) {
  206. return getUrlForRoom(version, baseUrl, token) + "/notify-calls";
  207. }
  208. public static String getUrlForCall(int version, String baseUrl, String token) {
  209. return getUrlForApi(version, baseUrl) + "/call/" + token;
  210. }
  211. public static String getUrlForChat(int version, String baseUrl, String token) {
  212. return getUrlForApi(version, baseUrl) + "/chat/" + token;
  213. }
  214. public static String getUrlForMentionSuggestions(int version, String baseUrl, String token) {
  215. return getUrlForChat(version, baseUrl, token) + "/mentions";
  216. }
  217. public static String getUrlForChatMessage(int version, String baseUrl, String token, String messageId) {
  218. return getUrlForChat(version, baseUrl, token) + "/" + messageId;
  219. }
  220. public static String getUrlForChatSharedItems(int version, String baseUrl, String token) {
  221. return getUrlForChat(version, baseUrl, token) + "/share";
  222. }
  223. public static String getUrlForChatSharedItemsOverview(int version, String baseUrl, String token) {
  224. return getUrlForChatSharedItems(version, baseUrl, token) + "/overview";
  225. }
  226. public static String getUrlForSignaling(int version, String baseUrl) {
  227. return getUrlForApi(version, baseUrl) + "/signaling";
  228. }
  229. public static String getUrlForSignalingBackend(int version, String baseUrl) {
  230. return getUrlForSignaling(version, baseUrl) + "/backend";
  231. }
  232. public static String getUrlForSignalingSettings(int version, String baseUrl) {
  233. return getUrlForSignaling(version, baseUrl) + "/settings";
  234. }
  235. public static String getUrlForSignaling(int version, String baseUrl, String token) {
  236. return getUrlForSignaling(version, baseUrl) + "/" + token;
  237. }
  238. public static String getUrlForOpenConversations(int version, String baseUrl) {
  239. return getUrlForApi(version, baseUrl) + "/listed-room";
  240. }
  241. public static RetrofitBucket getRetrofitBucketForCreateRoom(int version, String baseUrl, String roomType,
  242. @Nullable String source,
  243. @Nullable String invite,
  244. @Nullable String conversationName) {
  245. RetrofitBucket retrofitBucket = new RetrofitBucket();
  246. retrofitBucket.setUrl(getUrlForRooms(version, baseUrl));
  247. Map<String, String> queryMap = new HashMap<>();
  248. queryMap.put("roomType", roomType);
  249. if (invite != null) {
  250. queryMap.put("invite", invite);
  251. }
  252. if (source != null) {
  253. queryMap.put("source", source);
  254. }
  255. if (conversationName != null) {
  256. queryMap.put("roomName", conversationName);
  257. }
  258. retrofitBucket.setQueryMap(queryMap);
  259. return retrofitBucket;
  260. }
  261. public static RetrofitBucket getRetrofitBucketForAddParticipant(int version, String baseUrl, String token, String user) {
  262. RetrofitBucket retrofitBucket = new RetrofitBucket();
  263. retrofitBucket.setUrl(getUrlForParticipants(version, baseUrl, token));
  264. Map<String, String> queryMap = new HashMap<>();
  265. queryMap.put("newParticipant", user);
  266. retrofitBucket.setQueryMap(queryMap);
  267. return retrofitBucket;
  268. }
  269. public static RetrofitBucket getRetrofitBucketForAddParticipantWithSource(
  270. int version,
  271. String baseUrl,
  272. String token,
  273. String source,
  274. String id
  275. ) {
  276. RetrofitBucket retrofitBucket = getRetrofitBucketForAddParticipant(version, baseUrl, token, id);
  277. retrofitBucket.getQueryMap().put("source", source);
  278. return retrofitBucket;
  279. }
  280. public static String getUrlForUserProfile(String baseUrl) {
  281. return baseUrl + ocsApiVersion + "/cloud/user";
  282. }
  283. public static String getUrlForUserData(String baseUrl, String userId) {
  284. return baseUrl + ocsApiVersion + "/cloud/users/" + userId;
  285. }
  286. public static String getUrlForUserSettings(String baseUrl) {
  287. // FIXME Introduce API version
  288. return baseUrl + ocsApiVersion + spreedApiVersion + "/settings/user";
  289. }
  290. public static String getUrlPostfixForStatus() {
  291. return "/status.php";
  292. }
  293. public static String getUrlForAvatar(String baseUrl, String name, boolean requestBigSize) {
  294. int avatarSize = requestBigSize ? AVATAR_SIZE_BIG : AVATAR_SIZE_SMALL;
  295. return baseUrl + "/index.php/avatar/" + Uri.encode(name) + "/" + avatarSize;
  296. }
  297. public static String getUrlForGuestAvatar(String baseUrl, String name, boolean requestBigSize) {
  298. int avatarSize = requestBigSize ? AVATAR_SIZE_BIG : AVATAR_SIZE_SMALL;
  299. return baseUrl + "/index.php/avatar/guest/" + Uri.encode(name) + "/" + avatarSize;
  300. }
  301. public static String getCredentials(String username, String token) {
  302. if (TextUtils.isEmpty(username) && TextUtils.isEmpty(token)) {
  303. return null;
  304. }
  305. return Credentials.basic(username, token);
  306. }
  307. public static String getUrlNextcloudPush(String baseUrl) {
  308. return baseUrl + ocsApiVersion + "/apps/notifications/api/v2/push";
  309. }
  310. public static String getUrlPushProxy() {
  311. return NextcloudTalkApplication.Companion.getSharedApplication().
  312. getApplicationContext().getResources().getString(R.string.nc_push_server_url) + "/devices";
  313. }
  314. // see https://github.com/nextcloud/notifications/blob/master/docs/ocs-endpoint-v2.md
  315. public static String getUrlForNcNotificationWithId(String baseUrl, String notificationId) {
  316. return baseUrl + ocsApiVersion + "/apps/notifications/api/v2/notifications/" + notificationId;
  317. }
  318. public static String getUrlForSearchByNumber(String baseUrl) {
  319. return baseUrl + ocsApiVersion + "/cloud/users/search/by-phone";
  320. }
  321. public static String getUrlForFileUpload(String baseUrl, String user, String remotePath) {
  322. return baseUrl + "/remote.php/dav/files/" + user + remotePath;
  323. }
  324. public static String getUrlForChunkedUpload(String baseUrl, String user) {
  325. return baseUrl + "/remote.php/dav/uploads/" + user;
  326. }
  327. public static String getUrlForFileDownload(String baseUrl, String user, String remotePath) {
  328. return baseUrl + "/remote.php/dav/files/" + user + "/" + remotePath;
  329. }
  330. public static String getUrlForTempAvatar(String baseUrl) {
  331. return baseUrl + ocsApiVersion + "/apps/spreed/temp-user-avatar";
  332. }
  333. public static String getUrlForUserFields(String baseUrl) {
  334. return baseUrl + ocsApiVersion + "/cloud/user/fields";
  335. }
  336. public static String getUrlToSendLocation(int version, String baseUrl, String roomToken) {
  337. return getUrlForChat(version, baseUrl, roomToken) + "/share";
  338. }
  339. public static String getUrlForHoverCard(String baseUrl, String userId) {
  340. return baseUrl + ocsApiVersion +
  341. "/hovercard/v1/" + userId;
  342. }
  343. public static String getUrlForChatReadMarker(int version, String baseUrl, String roomToken) {
  344. return getUrlForChat(version, baseUrl, roomToken) + "/read";
  345. }
  346. /*
  347. * OCS Status API
  348. */
  349. public static String getUrlForStatus(String baseUrl) {
  350. return baseUrl + ocsApiVersion + "/apps/user_status/api/v1/user_status";
  351. }
  352. public static String getUrlForSetStatusType(String baseUrl) {
  353. return getUrlForStatus(baseUrl) + "/status";
  354. }
  355. public static String getUrlForPredefinedStatuses(String baseUrl) {
  356. return baseUrl + ocsApiVersion + "/apps/user_status/api/v1/predefined_statuses";
  357. }
  358. public static String getUrlForStatusMessage(String baseUrl) {
  359. return getUrlForStatus(baseUrl) + "/message";
  360. }
  361. public static String getUrlForSetCustomStatus(String baseUrl) {
  362. return baseUrl + ocsApiVersion + "/apps/user_status/api/v1/user_status/message/custom";
  363. }
  364. public static String getUrlForSetPredefinedStatus(String baseUrl) {
  365. return baseUrl + ocsApiVersion + "/apps/user_status/api/v1/user_status/message/predefined";
  366. }
  367. public static String getUrlForUserStatuses(String baseUrl) {
  368. return baseUrl + ocsApiVersion + "/apps/user_status/api/v1/statuses";
  369. }
  370. public static String getUrlForMessageReaction(String baseUrl,
  371. String roomToken,
  372. String messageId) {
  373. return baseUrl + ocsApiVersion + spreedApiVersion + "/reaction/" + roomToken + "/" + messageId;
  374. }
  375. @NonNull
  376. public static String getUrlForUnifiedSearch(@NonNull String baseUrl, @NonNull String providerId) {
  377. return baseUrl + ocsApiVersion + "/search/providers/" + providerId + "/search";
  378. }
  379. public static String getUrlForPoll(String baseUrl,
  380. String roomToken,
  381. String pollId) {
  382. return getUrlForPoll(baseUrl, roomToken) + "/" + pollId;
  383. }
  384. public static String getUrlForPoll(String baseUrl,
  385. String roomToken) {
  386. return baseUrl + ocsApiVersion + spreedApiVersion + "/poll/" + roomToken;
  387. }
  388. public static String getUrlForMessageExpiration(int version, String baseUrl, String token) {
  389. return getUrlForRoom(version, baseUrl, token) + "/message-expiration";
  390. }
  391. public static String getUrlForOpenGraph(String baseUrl) {
  392. return baseUrl + ocsApiVersion + "/references/resolve";
  393. }
  394. public static String getUrlForRecording(int version, String baseUrl, String token) {
  395. return getUrlForApi(version, baseUrl) + "/recording/" + token;
  396. }
  397. public static String getUrlForRequestAssistance(int version, String baseUrl, String token) {
  398. return getUrlForApi(version, baseUrl) + "/breakout-rooms/" + token + "/request-assistance";
  399. }
  400. }