ApiUtils.java 17 KB

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